web-dev-qa-db-fra.com

Analyser XML pour obtenir la valeur du nœud dans le script bash?

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.

19
MagePsycho

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
19
paoul

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
6
bahamat

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)
3
Dag Wieers

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:

  • lent
  • recherche uniquement par nom de balise
  • aucun décodage d'entité de caractère
3
manatwork

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.

3
Charles Duffy

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.

0
Bimal Poudel

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)

0
KuLuSz