From debd01c185d19dce26bdc0aff2e8c58980a85ded Mon Sep 17 00:00:00 2001 From: Felipe Zago Date: Fri, 25 Aug 2023 15:35:05 -0300 Subject: [PATCH] [ADD] zeep monkey patch to work with multiple complex types --- src/erpbrasil/monkey_patch/__init__.py | 1 + .../monkey_patch/zeep_monkey_patch.py | 42 +++++++++++++++++++ src/erpbrasil/transmissao/transmissao.py | 4 ++ 3 files changed, 47 insertions(+) create mode 100644 src/erpbrasil/monkey_patch/__init__.py create mode 100644 src/erpbrasil/monkey_patch/zeep_monkey_patch.py diff --git a/src/erpbrasil/monkey_patch/__init__.py b/src/erpbrasil/monkey_patch/__init__.py new file mode 100644 index 0000000..cb9d717 --- /dev/null +++ b/src/erpbrasil/monkey_patch/__init__.py @@ -0,0 +1 @@ +from . import zeep_monkey_patch diff --git a/src/erpbrasil/monkey_patch/zeep_monkey_patch.py b/src/erpbrasil/monkey_patch/zeep_monkey_patch.py new file mode 100644 index 0000000..e6a9623 --- /dev/null +++ b/src/erpbrasil/monkey_patch/zeep_monkey_patch.py @@ -0,0 +1,42 @@ +from lxml import etree +from zeep import xsd + +# Monkey patch relacionado a este PR: https://github.com/mvantellingen/python-zeep/pull/1384 +# +# Foi necessário essa alteração devido ao fato da biblioteca "Zeep" não fornecer o tratamento +# para WSDLs que consistem em multiplos tipos complexos aninhados que esperam uma entrada do +# tipo "any", e a entrada fornecida é um objeto etree. +# +# O problema foi encontrado na implementação de DFe utilizando as bibliotecas nfelib e erpbrasil.edoc. +def custom_render(self, parent, value, render_path): + if not isinstance(value, list): + values = [value] + else: + values = value + + self.validate(values, render_path) + + child_path = render_path + for value in xsd.utils.max_occurs_iter(self.max_occurs, values): + for name, element in self.elements_nested: + if name: + if name in value: + element_value = value[name] + child_path += [name] + elif isinstance(value, etree._Element): + element_value = value + child_path += [name] + else: + element_value = xsd.const.NotSet + else: + element_value = value + + if element_value is xsd.const.SkipValue: + continue + + if element_value is not None or not element.is_optional: + element.render(parent, element_value, child_path) + + +def apply_zeep_monkey_patches(): + xsd.elements.indicators.OrderIndicator.render = custom_render diff --git a/src/erpbrasil/transmissao/transmissao.py b/src/erpbrasil/transmissao/transmissao.py index fb0c342..9ff93af 100644 --- a/src/erpbrasil/transmissao/transmissao.py +++ b/src/erpbrasil/transmissao/transmissao.py @@ -16,6 +16,10 @@ from zeep.cache import SqliteCache from zeep.transports import Transport +from ..monkey_patch.zeep_monkey_patch import apply_zeep_monkey_patches + +apply_zeep_monkey_patches() + ABC = abc.ABCMeta('ABC', (object,), {})