Je voudrais savoir comment obtenir la valeur d'un nœud avec les chemins suivants:
config/global/resources/default_setup/connection/Host
config/global/resources/default_setup/connection/username
config/global/resources/default_setup/connection/password
config/global/resources/default_setup/connection/dbname
à partir du XML suivant:
<?xml version="1.0"?>
<config>
<global>
<install>
<date><![CDATA[Tue, 11 Dec 2012 12:31:25 +0000]]></date>
</install>
<crypt>
<key><![CDATA[70e75d7969b900b696785f2f81ecb430]]></key>
</crypt>
<disable_local_modules>false</disable_local_modules>
<resources>
<db>
<table_prefix><![CDATA[]]></table_prefix>
</db>
<default_setup>
<connection>
<Host><![CDATA[localhost]]></Host>
<username><![CDATA[root]]></username>
<password><![CDATA[pass123]]></password>
<dbname><![CDATA[testdb]]></dbname>
<initStatements><![CDATA[SET NAMES utf8]]></initStatements>
<model><![CDATA[mysql4]]></model>
<type><![CDATA[pdo_mysql]]></type>
<pdoType><![CDATA[]]></pdoType>
<active>1</active>
</connection>
</default_setup>
</resources>
<session_save><![CDATA[files]]></session_save>
</global>
<admin>
<routers>
<adminhtml>
<args>
<frontName><![CDATA[admin]]></frontName>
</args>
</adminhtml>
</routers>
</admin>
</config>
Je souhaite également attribuer cette valeur à la variable pour une utilisation ultérieure. Faites-moi savoir votre idée.
En utilisant bash
et xmllint
(comme indiqué par les balises):
xmllint --version # xmllint: using libxml version 20703
# Note: Newer versions of libxml / xmllint have a --xpath option which
# makes it possible to use xpath expressions directly as arguments.
# --xpath also enables precise output in contrast to the --Shell & sed approaches below.
#xmllint --help 2>&1 | grep -i 'xpath'
{
# the given XML is in file.xml
Host="$(echo "cat /config/global/resources/default_setup/connection/Host/text()" | xmllint --nocdata --Shell file.xml | sed '1d;$d')"
username="$(echo "cat /config/global/resources/default_setup/connection/username/text()" | xmllint --nocdata --Shell file.xml | sed '1d;$d')"
password="$(echo "cat /config/global/resources/default_setup/connection/password/text()" | xmllint --nocdata --Shell file.xml | sed '1d;$d')"
dbname="$(echo "cat /config/global/resources/default_setup/connection/dbname/text()" | xmllint --nocdata --Shell file.xml | sed '1d;$d')"
printf '%s\n' "Host: $Host" "username: $username" "password: $password" "dbname: $dbname"
}
# output
# Host: localhost
# username: root
# password: pass123
# dbname: testdb
Dans le cas où il n'y a qu'une chaîne XML et que l'utilisation d'un fichier temporaire est à éviter, les descripteurs de fichiers sont la voie à suivre avec xmllint
(qui est donné /dev/fd/3
comme argument de fichier ici):
set +H
{
xmlstr='<?xml version="1.0"?>
<config>
<global>
<install>
<date><![CDATA[Tue, 11 Dec 2012 12:31:25 +0000]]></date>
</install>
<crypt>
<key><![CDATA[70e75d7969b900b696785f2f81ecb430]]></key>
</crypt>
<disable_local_modules>false</disable_local_modules>
<resources>
<db>
<table_prefix><![CDATA[]]></table_prefix>
</db>
<default_setup>
<connection>
<Host><![CDATA[localhost]]></Host>
<username><![CDATA[root]]></username>
<password><![CDATA[pass123]]></password>
<dbname><![CDATA[testdb]]></dbname>
<initStatements><![CDATA[SET NAMES utf8]]></initStatements>
<model><![CDATA[mysql4]]></model>
<type><![CDATA[pdo_mysql]]></type>
<pdoType><![CDATA[]]></pdoType>
<active>1</active>
</connection>
</default_setup>
</resources>
<session_save><![CDATA[files]]></session_save>
</global>
<admin>
<routers>
<adminhtml>
<args>
<frontName><![CDATA[admin]]></frontName>
</args>
</adminhtml>
</routers>
</admin>
</config>
'
# exec issue
#exec 3<&- 3<<<"$xmlstr"
#exec 3<&- 3< <(printf '%s' "$xmlstr")
exec 3<&- 3<<EOF
$(printf '%s' "$xmlstr")
EOF
{ read -r Host; read -r username; read -r password; read -r dbname; } < <(
echo "cat /config/global/resources/default_setup/connection/*[self::Host or self::username or self::password or self::dbname]/text()" |
xmllint --nocdata --Shell /dev/fd/3 |
sed -e '1d;$d' -e '/^ *--* *$/d'
)
printf '%s\n' "Host: $Host" "username: $username" "password: $password" "dbname: $dbname"
exec 3<&-
}
set -H
# output
# Host: localhost
# username: root
# password: pass123
# dbname: testdb
Bien qu'il y ait déjà beaucoup de réponses, je répondrai avec xml2
.
$ xml2 < test.xml
/config/global/install/date=Tue, 11 Dec 2012 12:31:25 +0000
/config/global/crypt/key=70e75d7969b900b696785f2f81ecb430
/config/global/disable_local_modules=false
/config/global/resources/db/table_prefix
/config/global/resources/default_setup/connection/Host=localhost
/config/global/resources/default_setup/connection/username=root
/config/global/resources/default_setup/connection/password=pass123
/config/global/resources/default_setup/connection/dbname=testdb
/config/global/resources/default_setup/connection/initStatements=SET NAMES utf8
/config/global/resources/default_setup/connection/model=mysql4
/config/global/resources/default_setup/connection/type=pdo_mysql
/config/global/resources/default_setup/connection/pdoType
/config/global/resources/default_setup/connection/active=1
/config/global/session_save=files
/config/admin/routers/adminhtml/args/frontName=admin
Avec un peu de magie, vous pouvez même les définir directement en tant que variables:
$ eval $(xml2 < test.xml | tr '/, ' '___' | grep =)
$ echo $_config_global_resources_default_setup_connection_Host
localhost
En utilisant xmllint et l'option - xpath , c'est très facile. Vous pouvez simplement faire ceci:
XML_FILE=/path/to/file.xml
Host=$(xmllint --xpath 'string(/config/global/resources/default_setup/connection/Host)' $XML_FILE
USERNAME=$(xmllint --xpath 'string(/config/global/resources/default_setup/connection/username)' $XML_FILE
PASSWORD=$(xmllint --xpath 'string(/config/global/resources/default_setup/connection/password)' $XML_FILE
DBNAME=$(xmllint --xpath 'string(/config/global/resources/default_setup/connection/dbname)' $XML_FILE
Si vous avez besoin d'accéder à l'attribut d'un élément, c'est aussi facile d'utiliser XPath. Imaginez que vous avez le fichier:
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<addon id="screensaver.turnoff"
name="Turn Off"
version="0.10.0"
provider-name="Dag Wieërs">
..snip..
</addon>
Les déclarations Shell nécessaires seraient:
VERSION=$(xmllint --xpath 'string(/addon/@version)' $ADDON_XML)
AUTHOR=$(xmllint --xpath 'string(/addon/@provider-name)' $ADDON_XML)
Une fonction pure bash
, juste pour le cas malheureux où vous n'êtes pas autorisé à installer quoi que ce soit de approprié. Cela peut, et échouera probablement, sur un XML plus compliqué:
function xmlpath()
{
local expr="${1//\// }"
local path=()
local chunk tag data
while IFS='' read -r -d '<' chunk; do
IFS='>' read -r tag data <<< "$chunk"
case "$tag" in
'?'*) ;;
'!–-'*) ;;
'![CDATA['*) data="${tag:8:${#tag}-10}" ;;
?*'/') ;;
'/'?*) unset path[${#path[@]}-1] ;;
?*) path+=("$tag") ;;
esac
[[ "${path[@]}" == "$expr" ]] && echo "$data"
done
}
Usage:
bash-4.1$ xmlpath 'config/global/resources/default_setup/connection/Host' < MagePsycho.xml
localhost
Problèmes connus:
Les éléments suivants fonctionnent lorsqu'ils sont exécutés sur vos données de test:
{ read -r Host; read -r username; read -r password; read -r dbname; } \
< <(xmlstarlet sel -t -m /config/global/resources/default_setup/connection \
-v ./Host -n \
-v ./username -n \
-v ./password -n \
-v ./dbname -n)
Cela place le contenu dans les variables Host
, username
, password
et dbname
.
Vous pouvez utiliser le codage d'interface de ligne de commande php dans les scripts bash pour gérer plusieurs scripts complexes qui s'étendent sur plusieurs lignes de codage. Tout d'abord, essayez de créer votre solution en utilisant les scripts PHP, puis passez les paramètres en mode CLI. Ainsi, vous pouvez contrôler les superbes utilisations des analyseurs XML.
L'environnement semble que vous pouvez utiliser PHP en mode client via l'accès ssh/Shell.
php -f yourxmlparser.php
Maintenant, faites toutes les choses dans votre fichier php. Utilisez les paramètres de ligne de commande que cela peut prendre.
Vous pouvez même affecter ces valeurs de retour à l'environnement Shell pour continuer le reste de vos scripts Shell.
Et l'autre manière consiste à utiliser l'option | grep pour faire correspondre votre valeur requise dans le fichier xml, si vous êtes assez sûr de la structure de votre fichier xml qui ne change pas avec le temps.
Ce commentaire utilise uniquement les commandes et méthodes sh/bash! /test.xml est votre fichier de type XML à la première question ...
#!/bin/sh
cat /test.xml | while read line;do
[ "$(echo "$line" | grep "<Host>")" ]&& echo "Host: $(echo $line | cut -f3 -d'[' | cut -f1 -d']')"
[ "$(echo "$line" | grep "<username>")" ]&& echo "username: $(echo $line | cut -f3 -d'[' | cut -f1 -d']')"
[ "$(echo "$line" | grep "<password>")" ]&& echo "password: $(echo $line | cut -f3 -d'[' | cut -f1 -d']')"
[ "$(echo "$line" | grep "<dbname")" ]&& echo "dbname: $(echo $line | cut -f3 -d'[' | cut -f1 -d']')"
done
production:
Host: localhost
username: root
password: pass123
dbname: testdb
si vous voulez écrire ces valeurs dans un fichier, utilisez cette méthode:
#!/bin/sh
cat /test.xml | while read line;do
[ "$(echo "$line" | grep "<Host>")" ]&& echo "$line" | cut -f3 -d'[' | cut -f1 -d']' > /config/global/resources/default_setup/connection/Host
[ "$(echo "$line" | grep "<username>")" ]&& echo "$line" | cut -f3 -d'[' | cut -f1 -d']' > /config/global/resources/default_setup/connection/username
[ "$(echo "$line" | grep "<password>")" ]&& echo "$line" | cut -f3 -d'[' | cut -f1 -d']' > /config/global/resources/default_setup/connection/password
[ "$(echo "$line" | grep "<dbname")" ]&& echo "$line" | cut -f3 -d'[' | cut -f1 -d']' > /config/global/resources/default_setup/connection/dbname
done
cette méthode écrasera vos fichiers locaux utilisés uniquement en obtenant des valeurs (vos données seront perdues des fichiers de sortie)