Kader Yildirim
A la découverte de BizTalk Server 2006 1/3
Développer un désassembleur pour BizTalk Server 2006 R2
Par Kader Yildirim publié le 02/03/2008 à 21:58
 
La première étape consiste à développer une fonction qui prend en paramètre un document XML et une liste de XSD et qui retourne une liste de documents XML compatibles avec au moins l'un des XSD. La fonction peut être schématisée comme suit :
 
/content/af013d85-430c-482b-8450-a88c26dc0881/ExtractDocument.png
 
Pour BizTalk un schéma est caractérisé par son namespace et le nom du noeud racine.
Si le namespace et le nom d'un noeud du document XML donné en entrée correspondent avec ceux d'un schéma alors on peut supposer que ce noeud est potentiellement valide et l'extraire. C'est ce que fait le code ci-dessous :

public Dictionary<Guid, Pair<Schema, Stream>> Find(Stream doc, SchemaList schemas)

        {

            Pair<string, string> root = null;

            Dictionary<Guid, Pair<Schema, Stream>> parts = new Dictionary<Guid, Pair<Schema, Stream>>();

 

            using (XmlReader docReader = XmlReader.Create(doc))

            {

                MemoryStream newDocStream = new MemoryStream();

                using (XmlWriter newDocWriter = XmlWriter.Create(newDocStream))

                {

                    while (docReader.Read())

                    {

                        if (docReader.NodeType == XmlNodeType.Element)

                        {

                            Pair<string, string> p = new Pair<string, string>(docReader.LocalName, docReader.NamespaceURI);

 

                            if (root == null)

                                root = p;

 

                            Schema schema = GetAssociatedSchema(schemas, p);

                            if (null != schema)

                            {

                                if (p.Left == root.Left && p.Right == root.Right)

                                {

                                    foreach (Pair<Schema, Stream> pair in parts.Values)

                                        pair.Right.Close();

                                    parts.Clear();

                                    newDocStream.Close();

                                    parts.Add(Guid.Empty, new Pair<Schema, Stream>(schema, doc));

                                    doc.Position = 0;

                                    return parts;

                                }

 

                                MemoryStream part = new MemoryStream();

                                using (XmlWriter partWriter = XmlWriter.Create(part))

                                {

                                    PipelineUtils.Util.WriteShallowNode(docReader, partWriter);

                                    while (docReader.Read())

                                    {

                                        PipelineUtils.Util.WriteShallowNode(docReader, partWriter);

                                        if (p.Left == docReader.LocalName &&

                                            p.Right == docReader.NamespaceURI &&

                                            docReader.NodeType == XmlNodeType.EndElement)

                                        {

                                            break;

                                        }

                                    }

                                }

 

                                Guid g = Guid.NewGuid();

                                parts.Add(g, new Pair<Schema, Stream>(schema, part));

                                part.Position = 0;

                                newDocWriter.WriteElementString(r_TagName, g.ToString());

                                continue;

                            }

                        }

 

                        PipelineUtils.Util.WriteShallowNode(docReader, newDocWriter);

                    }

                }

                parts.Add(Guid.Empty, new Pair<Schema, Stream>(null, newDocStream));

                newDocStream.Position = 0;

            }

 

            return parts;

        }

La fonction WriteShallowNode permet de copier le contenu d'un reader vers un writer.

Toutefois si l'utilisateur le souhaite il peut forcer la validation complète des fragments ainsi extraits. Dans ce cas le code suivant réinjecte dans la source les documents non valides :

public void ValidateXmlDocumentsAgainstSchema(Dictionary<Guid, Pair<Schema, Stream>> dvn,

            Dictionary<Schema, XmlSchemaCollection> s)

        {

            if (null == dvn)

                throw new ArgumentNullException();

 

            if (1 == dvn.Count)

                return;

 

            Pair<Schema, Stream> body = dvn[Guid.Empty];

            body.Right.Position = 0;

            MemoryStream stream = new MemoryStream();

            using (XmlReader reader = XmlReader.Create(body.Right))

            {

                using (XmlWriter writer = XmlWriter.Create(stream))

                {

                    while (reader.Read())

                    {

                        if (reader.NodeType == XmlNodeType.Element &&

                            reader.LocalName == r_TagName)

                        {

                            reader.Read();

                            Guid g = new Guid(reader.Value);

                            using (XmlReader partReader = Validate(dvn[g].Right, s[dvn[g].Left]))

                            {

                                reader.Read();//read guid end tag

 

                                if (null == partReader)

                                {

                                    writer.WriteElementString(r_TagName, g.ToString());

                                }

                                else

                                {

                                    do

                                    {

                                        PipelineUtils.Util.WriteShallowNode(partReader, writer);

                                    }

                                    while (partReader.Read());

 

                                    dvn[g].Right.Close();

                                    dvn.Remove(g);

                                }

                            }

                        }

                        else PipelineUtils.Util.WriteShallowNode(reader, writer);

                    }

                }

            }

            body.Right.Close();

            body.Right = stream;

            stream.Position = 0;

        }

La validation en elle-même est assez classique :

private XmlReader Validate(Stream doc, XmlSchemaCollection xsc)

        {

            if (null == doc || null == xsc)

                return null;

 

            XmlReaderSettings xrs = new XmlReaderSettings();

            xrs.ValidationType = ValidationType.Schema;

 

            foreach (XmlSchema xs in xsc)

                xrs.Schemas.Add(xs);

 

            doc.Position = 0;

            using (XmlReader reader = XmlReader.Create(doc, xrs))

            {

                try

                {

                    while (reader.Read()) ;

                }

                catch (XmlSchemaValidationException)

                {

                    doc.Position = 0;

                    XmlReader r = XmlReader.Create(doc);

                    r.MoveToContent();

                    return r;

                }

            }

            doc.Position = 0;

            return null;

 

        }

 Commentaire - A la découverte de BizTalk Server 2006 1/3 

 Dernières Publications      

Windows Media Center et WCF : développez votre maison intelligente
  Le développement d'applications pour Windows Media Center est facilité avec l'arrivée du SDK 5.3. Même si l'on sent un modèle objet bien lourd derrière, il devient plus facile d'exposer les fonctionnalités de WMC sous la forme de services WCF.
par Frédéric Colin posté le 23/06/2008 à 08:04, lu 295 fois, #0
Notions avancées avec Biztalk Server 2006 R2
  Utilisation des notions d'interchange, corrélation et convoi avec BizTalk Server 2006 R2
par Kader Yildirim posté le 09/06/2008 à 08:04, lu 288 fois, #0
Lucene Persistence Engine pour Evaluant Universal Storage Services
  Suite à l'article de Laurent Kempé, voici un moteur de stockage pour EUSS permettant l'indexation d'entités métier avec Lucene.
par Nicolas Penin posté le 01/06/2008 à 23:38, lu 510 fois, #1
Tags: C#, Linq
XMLA Trivia : Découverte du XMLA
  Le XMLA (XML for Analysis) est un langage normalisé par plusieurs éditeurs BI pour simplifier l'accès aux données aux cubes et aux métadonnées des bases multidimensionnelles.
par Renaud Harduin posté le 25/05/2008 à 11:57, lu 523 fois, #0
Exploiter les données CSV via Linq en toute simplicité
  A partir du requêteur dynamique fourni en exemple avec Visual Studio 2008, nous allons essayer de remplir les propriétés d'un ensemble d'objets à partir des données d'un fichier CSV. Nous enrichirons aussi le parseur de nos propres fonctions.
par Frédéric Mélantois posté le 17/05/2008 à 11:41, lu 1713 fois, #0
Comment manipuler simplement le contenu d'un fichier WordML ?
  Manipulations autour du format WordML
par Fabien Reinle posté le 14/05/2008 à 23:55, lu 781 fois, #0
Polymorphisme et contrats de données WCF
  WCF aborde les types polymorphes du point de vue de la sérialisation. En effet, la connaissance du type réel potentiel est rendue nécessaire dès la description du contrat de données. Une fois n'est pas coutume, j'ai réalisé l'exemple en VB.NET.
par Frédéric Colin posté le 14/05/2008 à 08:40, lu 1881 fois, #2
A la découverte de BizTalk Server 2006 3/3
  Développer un assembleur pour BizTalk Server 2006 R2
par Kader Yildirim posté le 06/05/2008 à 13:20, lu 511 fois, #0

 Dernières Actualités      

Deep Earth – Une belle utilisation de Virtual Earth et de Silverlight Deep Zoom
  Ce projet très intéressant est disponible sur Codeplex et vous pouvez voir une démo sur la page suivante . Bien entendu comme touts les projets sur Codeplex vous avez accès aux sources....
Tags: Silverlight
Sortie de JetBrains ReSharper 4.0 en version finale, l’outil ultime pour Visual Studio
  Après plusieurs mois de Early Access Program (EAP) , JetBrains met enfin à disposition la version finale de son outil ReSharper 4.0 . Cette nouvelle version est disponible pour Visual Studio 2005 &...
Tags: Visual Studio 2008, Visual Studio 2005, Outils
BoutDuTunnel v1.4
  BoutDuTunnel est un petit logiciel de tunneling réseau écrit en C#. Il permet par exemple d’accéder aux services ftp/smtp/pop/telnet/nntp/… sur des réseaux qui n’autorisent...
BI Framework & sample sur CodePlex
  Après plusieurs demandes, je me suis décidé à déposer l'ensemble des sources et du BI Framework MS proposés dans mes articles sur codeplex : http://www.codeplex.com/BILAB Je le mettrais à jour au fil des...
Injection de code et API de profiling .NET
  Si vous êtes intéressés par la sécurité du Framework, par le reverse engineering et la manipulation/injection de code .NET et les packers, alors jetez un coup d’œil...
NDepend pour l'analyse statique de code .NET
  Pour ceux qui ne connaissent pas NDepend , il s’agit d’un outil d’analyse statique de code .NET qui permet de remonter des informations à toute une équipe de développement. NDepend aide à travailler sur...
Tags: Outils