Création et exploitation de fichiers XML avec Powershell

Je vais aborder dans cet article les possibilités qu’offre Powershell sur les fichiers XML. Récemment, j’ai développé un script réalisant un audit de l’ActiveDirectory au format HTML (que je partagerai éventuellement plus tard) et qui exporte certaines données dans un fichier XML, afin qu’elles puissent être exploitées plus facilement, par exemple pour avoir un historique et tracer des courbes d’évolution.

Nous allons donc utiliser les diverses méthodes de la classe Xml de .NET pour créer les différentes entrées, la hiérarchie et les attributs de ce que nous souhaitons stocker. Le script présenté dans cet article va récupérer les membres d’un groupe ActiveDirectory ainsi que des informations concernant ces membres, afin de tout stocker dans un fichier XML. Voici à quoi va ressembler mon fichier à la fin de l’exécution du code :

Tout d’abord, la déclaration de l’objet et l’écriture de la déclaration XML, qui est nécessaire pour que Powershell puisse gérer le fichier. Afin d’éviter d’écrire dans la console, j’ai besoin de piper out-null à chaque appel de AppendChild.

[xml]$xmlfile = New-Object System.Xml.XmlDocument
$xmldecl = $xmlfile.CreateXmlDeclaration("1.0","UTF-8",$null)
$xmlfile.AppendChild($xmldecl) | out-null

On créé le nœud racine <users>.

$xmlroot = $xmlfile.CreateNode("element","users",$null)

Les requêtes ActiveDirectory pour récupérer les membres du groupe en question, et la récupération des informations de ces utilisateurs :

$adGroupM = Get-AdGroupMember -identity "sales"
$adUsers = $adGroupM | foreach { Get-AdUser $_ -Properties DisplayName, mail, TelephoneNumber }

C’est ensuite dans une boucle ForEach que tout le traitement va avoir lieu. En effet, il va être nécessaire de jouer pour chaque utilisateur la création des éléments DisplayName, Mail et TelephoneNumber.

$adUsers | foreach {
	$userNode = $xmlfile.CreateNode("element","user",$null)
	$userNode.SetAttribute("Name",$_.samAccountName)
	$element = $xmlfile.CreateElement("DisplayName")
	$element.InnerText = $_.DisplayName
	$userNode.AppendChild($element) | out-null
	$element = $xmlfile.CreateElement("Mail")
	$element.InnerText = $_.mail
	$userNode.AppendChild($element) | out-null
	$element = $xmlfile.CreateElement("TelephoneNumber")
	$element.InnerText = $_.TelephoneNumber
	$userNode.AppendChild($element) | out-null
	$xmlroot.AppendChild($userNode) | out-null
}

Explications de ce snippet :

  • $UserNode désigne un utilisateur. Au niveau XML, il s’agit donc de <user>, qui contient <DisplayName>, <Mail> et <TelephoneNumber>.
  • Afin de pouvoir afficher le nom de l’utilisateur dans la balise XML <user> de manière à ce qu’il soit écrit <user Name= »mscott »>, il faut utiliser SetAttribute. Il est possible d’ajouter plusieurs attributs.
  • $Element désigne ce que l’on va ajouter à la balise parente. Il s’agit donc ici de <DisplayName>, <Mail> et <TelephoneNumber>. Ces éléments n’ont pas d’enfants, à l’inverse de <user>. On considère que <user> est un nœud (ou node) et <DisplayName> un élément. La fonction InnerText permet de spécifier la valeur : ici il s’agira du DisplayName, de l’adresse mail et du numéro de téléphone, que l’on récupère grâce à la variable temporaire $_ de la boucle ForEach.
  • Pour finir, on rattache l’élément créé au nœud parent correspondant, dans notre cas $UserNode. Lorsque l’élément a été rattaché, on obtient donc un nœud <User> complet. De la même manière, on le rattache au nœud racine du fichier XML, dans notre cas $xmlroot, qui s’appelle <users>.

En reprenant la capture d’écran plus haut, voici comment est découpé le fichier XML. En rouge le nœud racine $xmlroot ; en vert les noeuds $UserNode ; en orange les éléments $Element qui dépendent donc de $UserNode, lui-même dépendant de $xmlroot, dépendant finalement de $xmlfile.

Une fois que tous les nodes ont bien été traités, il faut préparer le fichier final. XML étant un format hiérarchique, $xmlroot n’est ni plus ni moins qu’un nœud de l’objet XML que nous avons déclaré en début de script, $xmlfile.

$xmlfile.AppendChild($xmlroot) | out-null

Pour conclure, l’écriture du fichier sur le disque :

$xmlfile.Save("sales-dundermifflin.xml")

Importons ce fichier XML dans Powershell :

[xml]$sales = Get-Content sales-dundermifflin.xml

Ensuite, appeler $sales n’aura aucun intérêt. Par contre, en utilisant les noms des nœuds et des éléments, on peut donc afficher et manipuler les informations comme on le souhaite :

Et pour reprendre le même code couleur que précédemment :

En rouge, l’ensemble du node Users, en vert les 4 noeuds User et en orange, les éléments composant chaque User, une ligne par node.
Lien pour marque-pages : Permaliens.

Un Commentaire

  1. Nicolas GIULIANI

    Bonjour enguerrand, je me permets d’envoyer ce message, je me demande si cela est possible de me partager le script indique en début d’article ( pour auditer AD). Merci par avance, Nicolas

Laisser un commentaire

Ce site utilise Akismet pour réduire les indésirables. En savoir plus sur comment les données de vos commentaires sont utilisées.