Изменение файла XML - Разделение значений

Я хочу изменить значения, безусловно, в файле XML, разделив их.

Мне нужно разделить значения для se:SvgParameter[name="stroke-width"] а также se:Size на 3,6. Я нашел это руководство на Python.org, Изменение XML-файла, но, похоже, оно не работает, и я не смог найти что-то похожее здесь. Любая помощь или точка будет принята с благодарностью!

Вот несколько примеров XML, но я хочу иметь возможность выводить сценарий в папку с несколькими файлами и преобразовывать их все.

XML:

<?xml version="1.0" encoding="ISO-8859-1"?>
<StyledLayerDescriptor version="1.1.0" xmlns="http://www.opengis.net/sld" xmlns:ogc="http://www.opengis.net/ogc" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xlink="http://www.w3.org/1999/xlink" xsi:schemaLocation="http://www.opengis.net/sld http://schemas.opengis.net/sld/1.1.0/StyledLayerDescriptor.xsd" xmlns:se="http://www.opengis.net/se">
  <NamedLayer>
    <se:Name>QGIS_SLD_ScaleProblem</se:Name>
    <se:Description>
      <se:Title>QGIS_SLD_ScaleProblem</se:Title>
      <se:Abstract>This SLD is in Pixels, but QGIS will read it as mm</se:Abstract>
    </se:Description>
    <UserStyle>
      <se:Name>QGIS_SLD_ScaleProblem</se:Name>
      <se:FeatureTypeStyle>
        <se:Rule>
          <se:Name>QGIS_SLD_ScaleProblem_Line</se:Name>
          <se:LineSymbolizer>
            <se:Name>Line</se:Name>
            <se:Stroke>
              <se:SvgParameter name="stroke">#FF0000</se:SvgParameter>
              <se:SvgParameter name="stroke-width">36</se:SvgParameter>
            </se:Stroke>
          </se:LineSymbolizer>
        </se:Rule>
        <se:Rule>
          <se:Name>QGIS_SLD_ScaleProblem_Point</se:Name>
          <se:PointSymbolizer>
            <se:Name>Point</se:Name>
            <se:Graphic>
             <se:Mark>
              <se:WellKnownName>circle</se:WellKnownName>
              <se:Fill>
               <se:SvgParameter name="fill">#FF0000</se:SvgParameter>
              </se:Fill>
             </se:Mark>
             <se:Size>36</se:Size>
            </se:Graphic>
          </se:PointSymbolizer>
        </se:Rule>
      </se:FeatureTypeStyle>
    </UserStyle>
  </NamedLayer>
</StyledLayerDescriptor>

Код:

import xml.etree.ElementTree as ET
from lxml import etree

tree = ET.parse('QGIS-SLD-ScaleProblem_Line.sld')
root = tree.getroot()

# To check that the script is reading the correct file
print("SLD file", root[0][0].text, "loaded.")

# Registers namespaces to prevent them being filled with defaults
ET.register_namespace('', "http://www.opengis.net/sld")
ET.register_namespace('ogc', "http://www.opengis.net/ogc")
ET.register_namespace('xsi', "http://www.w3.org/2001/XMLSchema-instance")
ET.register_namespace('xlink', "http://www.w3.org/1999/xlink")
ET.register_namespace('schemaLocation', "http://www.opengis.net/sld http://schemas.opengis.net/sld/1.1.0/StyledLayerDescriptor.xsd")
ET.register_namespace('se', "http://www.opengis.net/se")

# This section "should" search the XML for se:SvgParameter name="stroke-width" and se:Size and divide the value by 3.6
for SvgParameter in root.iter('SvgParameter'):
    new_SvgParameter = int(SvgParameter.text) / 3.6
    SvgParameter.text = str(new_SvgParameter)
for Size in root.iter('Size'):
    new_Size= int(Size.text) / 3.6
    Size.text = str(new_Size)

tree.write("GIS-SLD-ScaleFixed.sld", xml_declaration=True, encoding='ISO-8859-1')

Ответы, на которые я смотрел:

1 ответ

Решение

Рассмотрим XSLT - специальный язык, предназначенный для преобразования файлов XML. И питона lxml Модуль может запускать сценарии XSLT 1.0. Плюс XSLT может быть запущен вне Python, следовательно, его переносимость.

Просто вызовите преобразование идентичности, чтобы скопировать весь документ как есть, а затем запустите разделение на выбранных узлах. нет for петли или if логика нужна с таким подходом.

XSLT (сохранить как файл.xsl, специальный правильно сформированный файл XML)

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
                              xmlns:se="http://www.opengis.net/se">                               
  <xsl:output indent="yes"/>
  <xsl:strip-space elements="*"/>

  <xsl:template match="@*|node()">
    <xsl:copy>
      <xsl:apply-templates select="@*|node()"/>
    </xsl:copy>
  </xsl:template>

  <xsl:template match="se:SvgParameter[@name='stroke-width']|se:Size">
    <xsl:copy>
        <xsl:apply-templates select="@*"/>
        <xsl:value-of select="format-number(text(), '#') div 3.6"/>
    </xsl:copy>
  </xsl:template>

</xsl:stylesheet>

Python (итерация ниже процесса в цикле для нескольких файлов)

import lxml.etree as et

# LOAD XML AND XSL
doc = et.parse('Input.xml')
xsl = et.parse('XSLTScript.xsl')

# TRANSFORM
transform = et.XSLT(xsl)
result = transform(doc)

# OUTPUT TO SCREEN    
print(result)

# OUTPUT TO FILE
with open('Output.xml', 'wb') as f:
    f.write(result)

Выход

<?xml version="1.0"?>
<StyledLayerDescriptor xmlns="http://www.opengis.net/sld" xmlns:ogc="http://www.opengis.net/ogc" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xlink="http://www.w3.org/1999/xlink" xmlns:se="http://www.opengis.net/se" version="1.1.0" xsi:schemaLocation="http://www.opengis.net/sld http://schemas.opengis.net/sld/1.1.0/StyledLayerDescriptor.xsd">
  <NamedLayer>
    <se:Name>QGIS_SLD_ScaleProblem</se:Name>
    <se:Description>
      <se:Title>QGIS_SLD_ScaleProblem</se:Title>
      <se:Abstract>This SLD is in Pixels, but QGIS will read it as mm</se:Abstract>
    </se:Description>
    <UserStyle>
      <se:Name>QGIS_SLD_ScaleProblem</se:Name>
      <se:FeatureTypeStyle>
        <se:Rule>
          <se:Name>QGIS_SLD_ScaleProblem_Line</se:Name>
          <se:LineSymbolizer>
            <se:Name>Line</se:Name>
            <se:Stroke>
              <se:SvgParameter name="stroke">#FF0000</se:SvgParameter>
              <se:SvgParameter name="stroke-width">10</se:SvgParameter>
            </se:Stroke>
          </se:LineSymbolizer>
        </se:Rule>
        <se:Rule>
          <se:Name>QGIS_SLD_ScaleProblem_Point</se:Name>
          <se:PointSymbolizer>
            <se:Name>Point</se:Name>
            <se:Graphic>
              <se:Mark>
                <se:WellKnownName>circle</se:WellKnownName>
                <se:Fill>
                  <se:SvgParameter name="fill">#FF0000</se:SvgParameter>
                </se:Fill>
              </se:Mark>
              <se:Size>10</se:Size>
            </se:Graphic>
          </se:PointSymbolizer>
        </se:Rule>
      </se:FeatureTypeStyle>
    </UserStyle>
  </NamedLayer>
</StyledLayerDescriptor>
Другие вопросы по тегам