Создать XML на основе 2 XML и поиска значений в обоих файлах, используя XSLT2.0 - PullRequest
0 голосов
/ 29 апреля 2020

Я пытаюсь сгенерировать XML на основе 2 других XML. Я опрашиваю БД, которая возвращает данные о людях (в запросе может быть n человек). Конечный XML должен иметь точное количество тегов данных в качестве отдельных тегов имен в XML, поступающих из DB . Например:

1-й XML - Получение этого из БД

<parent>
    <child>
        <name>John</name>
        <city>Boston</city>
    </child>
    <child>
        <name>John</name>
        <city>Seattle</city>
    </child>
    <child>
        <name>Allison</name>
        <city>Houston</city>
    </child>
</parent>

2-й XML - Получение этого из другого источника

<details>
    <parent>
        <detail>
            <city>Boston</city>
            <code>abc</code>
        </detail>
        <detail>
            <city>Houston</city>
            <code>xyz</code>
        </detail>
    </parent>
    <parent>
        <detail>
            <city>Boston</city>
            <code>abc</code>
        </detail>
        <detail>
            <city>Seattle</city>
            <code>mno</code>
        </detail>
    </parent>
    <parent>
        <detail>
            <city>Houston</city>
            <code>xyz</code>
        </detail>
        <detail>
            <city>Seattle</city>
            <code>mno</code>
        </detail>
    </parent>
</details>

Сначала мне нужно создайте 2 тега данных, так как есть два разных имени - Джон и Эллисон (эта часть уже сделана и работает нормально). Затем мне нужно проверить Джона, какие уникальные теги города присутствуют в возвращаемых строках БД. Давайте рассмотрим 1-е XML, у нас есть Джон, связанный с Бостоном и Сиэтлом. Итак, один за другим, я проверю эти города во 2-м XML и для каждого тега parent , который мне соответствует, я создам новый тег details и вставлю все соответствующее содержимое ,

1) Если нет подходящих записей, тег подробностей создавать не следует, так как нет подходящей записи.

2) Придет тег city под тегом parent . Значения тега city будут УНИКАЛЬНЫМИ в теге parent . Я должен сопоставить city один за другим во втором XML и взять значения из всех соответствующих тегов city для всех родителей из второго XML и заполнить таким образом где все совпадения в теге parent помещаются в соответствующий тег detail в выводе XML. PFB примеры XML-файлов, которые объяснят лучше -

Окончательный ожидаемый XML -

<FinalData>
    <Data>
        <name>John</name>
        <details>
            <detail>
                <city value="Boston">abc</city>
            </detail>
            <detail>
                <city value="Boston">abc</city>
                <city value="Seattle">mno</city>
            </detail>
            <detail>
                <city value="Seattle">mno</city>
            </detail>
        </details>
    </Data>
    <Data>
        <name>Allison</name>
        <details>
            <detail>
                <city value="Houston">xyz</city>
            </detail>
            <detail>
                <city value="Houston">xyz</city>
            </detail>
        </details>
    </Data>
</FinalData>

В настоящее время мой XSLT приводит к чему-то вроде ниже -

<FinalData>
    <Data>
        <name>John</name>
        <details>
            <detail>
                <city value="Boston">abc</city>
                <city value="Boston">abc</city>
                <city value="Seattle">mno</city>
                <city value="Seattle">mno</city>
            </detail>
        </details>
    </Data>
    <Data>
        <name>Allison</name>
        <details>
            <detail>
                <city value="Houston">xyz</city>
                <city value="Houston">xyz</city>
            </detail>
        </details>
    </Data>
</FinalData>

Надеюсь, это понятно, потому что я не умею давать объяснения.

1 Ответ

0 голосов
/ 29 апреля 2020

Используйте ключи для разрешения перекрестных ссылок:

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    xmlns:xs="http://www.w3.org/2001/XMLSchema"
    exclude-result-prefixes="#all"
    version="3.0">

  <xsl:param name="details">
<details>
    <parent>
        <detail>
            <city>Boston</city>
            <code>abc</code>
        </detail>
        <detail>
            <city>Houston</city>
            <code>xyz</code>
        </detail>
    </parent>
    <parent>
        <detail>
            <city>Boston</city>
            <code>abc</code>
        </detail>
        <detail>
            <city>Seattle</city>
            <code>mno</code>
        </detail>
    </parent>
    <parent>
        <detail>
            <city>Houston</city>
            <code>xyz</code>
        </detail>
        <detail>
            <city>Seattle</city>
            <code>mno</code>
        </detail>
    </parent>
</details>      
  </xsl:param>

  <xsl:key name="parent-ref" match="parent" use="detail/city"/>
  <xsl:key name="detail-ref" match="parent/detail" use="city"/>

  <xsl:output method="xml" indent="yes" />

  <xsl:template match="parent">
    <FinalData>
        <xsl:for-each-group select="child" group-by="name">
            <Data>
                <xsl:copy-of select="name"/>
            </Data>
            <Details>
                <xsl:apply-templates select="key('parent-ref', current-group()/city, $details)"/>
            </Details>
        </xsl:for-each-group>
    </FinalData>
  </xsl:template>

  <xsl:template match="details/parent">
      <detail>
          <xsl:apply-templates select="key('detail-ref', current-group()/city, .)"/>
      </detail>
  </xsl:template>

  <xsl:template match="detail">
      <city value="{city}">
          <xsl:value-of select="code"/>
      </city>
  </xsl:template>

</xsl:stylesheet>

https://xsltfiddle.liberty-development.net/gVhDDyY

Для полноты второй документ встроен, но вы, конечно, можете использовать <xsl:param name="details" select="doc('details.xml')"/> вместо.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...