Преобразовать строку нотации точек в объект для доступа к l xml objects python - PullRequest
2 голосов
/ 12 марта 2020

Я пытаюсь обновить xml, используя обозначение объекта с помощью l xml objectify.

<xml>
  <fruit>
    <citrus>
        <lemon />
    </citrus>
  </fruit>  
</xml>

Я пытаюсь добавить еще один фрукт с именем man go, используя l xml objectify like

root = lxml.objectify.fromstring(xml_string)
root.fruit.citrus = 'orange'

def update(path, value):
    // code

update('fruit.citrus', 'orange')

Я хотел бы передать строку типа 'fruit.citrus', потому что я не могу передать объект fruit.citrus.

Как мне добиться этого в Python ie Как выполнить код 'root .fruit.citrus =' orange 'внутри функции обновления. Как преобразовать строку в объект?

Ответы [ 3 ]

1 голос
/ 12 марта 2020

Если вы настаиваете на использовании objectify, возможно, вам это не понравится, но я думаю, что это довольно чистое решение с использованием lxml etree:

from lxml import etree

doc = etree.fromstring("""<xml>
  <fruit>
    <citrus>
        <lemon />
    </citrus>
  </fruit>  
</xml>""")


def update(root, path, item):
    elems = root.xpath(path)
    for elem in elems:
        elem.append(etree.Element(item))


update(doc, 'fruit/citrus', 'orange')
print(etree.tostring(doc).decode())
1 голос
/ 12 марта 2020

Попробуйте следующее решение:

import lxml.objectify, lxml.etree

xml = '<xml>  <fruit>    <citrus>        <lemon />    </citrus>  </fruit> </xml>'

root = lxml.objectify.fromstring(xml)

print("Before:")
print(lxml.etree.tostring(root))


def update(path, value):
    parent = None
    lst = path.split('.')
    while lst:
        ele = lst.pop(0)
        parent = getattr(root, ele) if parent is None else getattr(parent, ele)
    lxml.etree.SubElement(parent, value)


update('fruit.citrus', 'orange')

print("After:")
print(lxml.etree.tostring(root))

Вывод:

Before:
b'<xml><fruit><citrus><lemon/></citrus></fruit></xml>'
After:
b'<xml><fruit><citrus><lemon/><orange/></citrus></fruit></xml>'
0 голосов
/ 13 марта 2020

Приведенные выше ответы были частично правильными. У него не было возможности обрабатывать индексы. Приведенный ниже код обрабатывает все случаи с ObjectPath (https://lxml.de/objectify.html).

import lxml.objectify, lxml.etree

from robot.api.deco import keyword

class ConfigXML(object):
    def get_xml(self, filename):
        self.root = lxml.objectify.fromstring(open(filename).read())

    def add_attribute(self, path, **kwargs):
        path_obj = lxml.objectify.ObjectPath(path)
        for key in kwargs:
            path_obj.find(self.root).set(key, kwargs[key])

    def add_value(self, path, value):
        path_obj = lxml.objectify.ObjectPath(path)
        path_obj.setattr(self.root, value)

    def add_tag(self, path, tag):
        path_obj = lxml.objectify.ObjectPath(path)
        lxml.objectify.SubElement(path_obj.find(self.root), tag)

    def generate_xml(self):
        lxml.objectify.deannotate(self.root, cleanup_namespaces=True, xsi_nil=True)
        return lxml.etree.tostring(self.root).decode('utf-8')
...