como ignorar namespaces com XPath


111

Meu objetivo é extrair certos nós de vários arquivos xml com vários namespaces usando XPath. Tudo funciona bem, desde que eu conheça os URIs de namespace. O nome do namespace em si permanece constante, mas os esquemas (XSD) às vezes são gerados pelo cliente, ou seja, desconhecidos para mim. Então, fico com basicamente três opções:

  1. use apenas um esquema para o namespace, esperando que nada dê errado (posso ter certeza?)

  2. pegue os nós filhos do documento e procure o primeiro nó com um URI de namespace, esperando que esteja lá e apenas use o URI, esperando que seja o correto. pode dar errado por vários motivos

  3. de alguma forma diga ao xpath: "olha, eu não me importo com os namespaces, apenas encontre TODOS os nós com este nome, posso até dizer o nome do namespace, mas não o URI". E esta é a questão aqui ...

Esta não é uma reiteração de várias perguntas "my xpath expression não funciona porque não estou ciente da consciência de namespace", conforme encontradas aqui ou aqui . Eu sei como usar o reconhecimento de namespace. Só não como se livrar disso.


2
Se você não conhece os esquemas, como sabe quais elementos deseja?
Paul Butcher


1
obrigado por apontar, Alejandro. A pesquisa por "ignorar namespace xpath" deveria ter revelado este, mas não revelou
kostja

2
@kostja: Não pesquise com a caixa de pesquisa do SO, é inútil ... Experimente o Google da próxima vez. Na verdade, isso é incentivado pela equipe SO.

1
O Google sitesearch realmente faz um trabalho melhor em encontrar coisas úteis no SO. Eu me pergunto por que não é uma opção por padrão. Obrigado novamente, Alejandro
kostja

Respostas:


164

Você pode usar a local-name()função XPath. Em vez de selecionar um nó como

/path/to/x:somenode

você pode selecionar todos os nós e filtrar aquele com o nome local correto:

/path/to/*[local-name() = 'somenode']

9
Você também pode usar o local-name()para se referir a atributos, de uma maneira sem conhecimento de namespace, consulte: stackoverflow.com/q/21239181/274677
Marcus Junius Brutus


1
Tão simples. salvou minha tarde.
C Johnson


2

Você pode usar Namespace = false em um XmlTextReader

[TestMethod]
public void MyTestMethod()
{
    string _withXmlns = @"<?xml version=""1.0"" encoding=""utf-8""?>
<ParentTag xmlns=""http://anyNamespace.com"">
<Identification value=""ID123456"" />
</ParentTag>
";

    var xmlReader = new XmlTextReader(new MemoryStream(Encoding.Default.GetBytes(_withXmlns)));

    xmlReader.Namespaces = false;

    var content = XElement.Load(xmlReader);

    XElement elem = content.XPathSelectElement("/Identification");

    elem.Should().NotBeNull();
    elem.Attribute("value").Value.Should().Be("ID123456");
}

com:

using System;
using System.IO;
using System.Linq;
using System.Text;
using System.Xml;
using System.Xml.Linq;
using System.Xml.XPath;
using FluentAssertions;
using Microsoft.VisualStudio.TestTools.UnitTesting;

Para selecionar um nó por meio do XPath, isso funciona; Infelizmente, você não pode salvar o documento devido a um 'The 'xmlns' attribute is bound to the reserved namespaceerro.
AutomatedChaos

2

Ou você pode usar o nome ():

/path/to/*[name() = 'somenode']

Ou apenas atributos de pesquisa:

//*[@attribute="this one"]

Se você abrir o xml como um objeto PowerShell, ele ignorará os namespaces:

[xml]$xml = get-content file.xml
$xml.path.to.somenode

0

É meu exemplo em Qt C ++. Qt suporta XPath 2.0:

    QString planePath = ":/Models/Plane.dae";
    QFile f(planePath);
    if (!f.open(QIODevice::ReadOnly))
    {
        std::cerr << "Failed to load the file: " <<
                     planePath.toStdString() << std::endl;
        return;
    }

    QXmlQuery query;
    query.bindVariable("myFile", &f);
//    query.setQuery("doc($myFile)//*[local-name() = 'p']/text()"); // it works too but it is XPath 1.0
    query.setQuery("doc($myFile)//*:p/text()");

    QString result;
    query.evaluateTo(&result);
    qDebug() << result;
    f.close();

Resultado do programa: "1 0 0 2 0 1 0 0 2 1 0 3 3 0 4 2 0 5\n"

Plane.dae

<?xml version="1.0" encoding="utf-8"?>
<COLLADA xmlns="http://www.collada.org/2005/11/COLLADASchema" version="1.4.1" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
  <asset>
    <contributor>
      <author>Blender User</author>
      <authoring_tool>Blender 2.83.3 commit date:2020-07-22, commit time:06:01, hash:353e5bd7493e</authoring_tool>
    </contributor>
    <created>2020-08-03T14:03:19</created>
    <modified>2020-08-03T14:03:19</modified>
    <unit name="meter" meter="1"/>
    <up_axis>Z_UP</up_axis>
  </asset>
  <library_effects>
    <effect id="PlaneMaterial-effect">
      <profile_COMMON>
        <technique sid="common">
          <lambert>
            <emission>
              <color sid="emission">0 0 0 1</color>
            </emission>
            <diffuse>
              <color sid="diffuse">0.01664001 0.8000001 0.01191879 1</color>
            </diffuse>
            <reflectivity>
              <float sid="specular">0.5</float>
            </reflectivity>
          </lambert>
        </technique>
      </profile_COMMON>
    </effect>
  </library_effects>
  <library_images/>
  <library_materials>
    <material id="PlaneMaterial-material" name="PlaneMaterial">
      <instance_effect url="#PlaneMaterial-effect"/>
    </material>
  </library_materials>
  <library_geometries>
    <geometry id="Plane-mesh" name="Plane">
      <mesh>
        <source id="Plane-mesh-positions">
          <float_array id="Plane-mesh-positions-array" count="12">-1 -1 0 1 -1 0 -1 1 0 1 1 0</float_array>
          <technique_common>
            <accessor source="#Plane-mesh-positions-array" count="4" stride="3">
              <param name="X" type="float"/>
              <param name="Y" type="float"/>
              <param name="Z" type="float"/>
            </accessor>
          </technique_common>
        </source>
        <source id="Plane-mesh-normals">
          <float_array id="Plane-mesh-normals-array" count="3">0 0 1</float_array>
          <technique_common>
            <accessor source="#Plane-mesh-normals-array" count="1" stride="3">
              <param name="X" type="float"/>
              <param name="Y" type="float"/>
              <param name="Z" type="float"/>
            </accessor>
          </technique_common>
        </source>
        <source id="Plane-mesh-map-0">
          <float_array id="Plane-mesh-map-0-array" count="12">1 0 0 1 0 0 1 0 1 1 0 1</float_array>
          <technique_common>
            <accessor source="#Plane-mesh-map-0-array" count="6" stride="2">
              <param name="S" type="float"/>
              <param name="T" type="float"/>
            </accessor>
          </technique_common>
        </source>
        <vertices id="Plane-mesh-vertices">
          <input semantic="POSITION" source="#Plane-mesh-positions"/>
        </vertices>
        <triangles material="PlaneMaterial-material" count="2">
          <input semantic="VERTEX" source="#Plane-mesh-vertices" offset="0"/>
          <input semantic="NORMAL" source="#Plane-mesh-normals" offset="1"/>
          <input semantic="TEXCOORD" source="#Plane-mesh-map-0" offset="2" set="0"/>
          <p>1 0 0 2 0 1 0 0 2 1 0 3 3 0 4 2 0 5</p>
        </triangles>
      </mesh>
    </geometry>
  </library_geometries>
  <library_visual_scenes>
    <visual_scene id="Scene" name="Scene">
      <node id="Plane" name="Plane" type="NODE">
        <matrix sid="transform">1 0 0 0 0 1 0 0 0 0 1 0 0 0 0 1</matrix>
        <instance_geometry url="#Plane-mesh" name="Plane">
          <bind_material>
            <technique_common>
              <instance_material symbol="PlaneMaterial-material" target="#PlaneMaterial-material">
                <bind_vertex_input semantic="UVMap" input_semantic="TEXCOORD" input_set="0"/>
              </instance_material>
            </technique_common>
          </bind_material>
        </instance_geometry>
      </node>
    </visual_scene>
  </library_visual_scenes>
  <scene>
    <instance_visual_scene url="#Scene"/>
  </scene>
</COLLADA>
Ao utilizar nosso site, você reconhece que leu e compreendeu nossa Política de Cookies e nossa Política de Privacidade.
Licensed under cc by-sa 3.0 with attribution required.