se implementa un esquema para modelar el xml
FossilOrigin-Name: e4de658f60fe8fcbb330923e14958a5d8f8e0e6395db4f992ec7da45062fa193
This commit is contained in:
parent
7d060e1786
commit
84996066fa
62
facho/model/__init__.py
Normal file
62
facho/model/__init__.py
Normal file
@ -0,0 +1,62 @@
|
||||
|
||||
class ModelMeta(type):
|
||||
def __new__(cls, name, bases, ns):
|
||||
new = type.__new__(cls, name, bases, ns)
|
||||
if '__name__' in ns:
|
||||
new.__name__ = ns['__name__']
|
||||
if '__namespace__' in ns:
|
||||
new.__namespace__ = ns['__namespace__']
|
||||
else:
|
||||
new.__namespace__ = {}
|
||||
|
||||
return new
|
||||
|
||||
class ModelBase(object, metaclass=ModelMeta):
|
||||
|
||||
def __new__(cls, *args, **kwargs):
|
||||
obj = super().__new__(cls, *args, **kwargs)
|
||||
obj._xml_attributes = {}
|
||||
obj._fields = {}
|
||||
obj._text = ""
|
||||
obj._namespace_prefix = None
|
||||
|
||||
return obj
|
||||
|
||||
def __setitem__(self, key, val):
|
||||
self._xml_attributes[key] = val
|
||||
|
||||
def __getitem__(self, key):
|
||||
return self._xml_attributes[key]
|
||||
|
||||
def to_xml(self):
|
||||
tag = self.__name__
|
||||
ns = ''
|
||||
if self._namespace_prefix is not None:
|
||||
ns = "%s:" % (self._namespace_prefix)
|
||||
|
||||
pair_attributes = ["%s=\"%s\"" % (k, v) for (k, v) in self._xml_attributes.values()]
|
||||
|
||||
for (prefix, url) in self.__namespace__.items():
|
||||
pair_attributes.append("xmlns:%s=\"%s\"" % (prefix, url))
|
||||
attributes = ""
|
||||
if pair_attributes:
|
||||
attributes = " " + " ".join(pair_attributes)
|
||||
|
||||
content = ""
|
||||
|
||||
for name, value in self._fields.items():
|
||||
if hasattr(value, 'to_xml'):
|
||||
print(self._fields)
|
||||
content += value.to_xml()
|
||||
elif isinstance(value, str):
|
||||
content += value
|
||||
content += self._text
|
||||
|
||||
if content == "":
|
||||
return "<%s%s%s/>" % (ns, tag, attributes)
|
||||
else:
|
||||
return "<%s%s%s>%s</%s%s>" % (ns, tag, attributes, content, ns, tag)
|
||||
|
||||
class Model(ModelBase):
|
||||
pass
|
||||
|
5
facho/model/fields/__init__.py
Normal file
5
facho/model/fields/__init__.py
Normal file
@ -0,0 +1,5 @@
|
||||
from .attribute import Attribute
|
||||
from .many2one import Many2One
|
||||
from .model import Model
|
||||
|
||||
__all__ = [Attribute, Many2One, Model]
|
17
facho/model/fields/attribute.py
Normal file
17
facho/model/fields/attribute.py
Normal file
@ -0,0 +1,17 @@
|
||||
from .field import Field
|
||||
|
||||
class Attribute(Field):
|
||||
def __init__(self, tag):
|
||||
self.tag = tag
|
||||
|
||||
def __get__(self, inst, cls):
|
||||
if inst is None:
|
||||
return self
|
||||
|
||||
assert self.name is not None
|
||||
(tag, value) = inst._xml_attributes[self.name]
|
||||
return value
|
||||
|
||||
def __set__(self, inst, value):
|
||||
assert self.name is not None
|
||||
inst._xml_attributes[self.name] = (self.tag, value)
|
21
facho/model/fields/field.py
Normal file
21
facho/model/fields/field.py
Normal file
@ -0,0 +1,21 @@
|
||||
class Field:
|
||||
def __set_name__(self, owner, name):
|
||||
self.name = name
|
||||
|
||||
def __get__(self, inst, cls):
|
||||
if inst is None:
|
||||
return self
|
||||
assert self.name is not None
|
||||
return inst._fields[self.name]
|
||||
|
||||
def __set__(self, inst, value):
|
||||
assert self.name is not None
|
||||
inst._fields[self.name] = value
|
||||
|
||||
def _set_namespace(self, inst, name, namespaces):
|
||||
if name is None:
|
||||
return
|
||||
|
||||
if name not in namespaces:
|
||||
raise KeyError("namespace %s not found" % (name))
|
||||
inst._namespace_prefix = name
|
35
facho/model/fields/many2one.py
Normal file
35
facho/model/fields/many2one.py
Normal file
@ -0,0 +1,35 @@
|
||||
from .field import Field
|
||||
|
||||
class Many2One(Field):
|
||||
def __init__(self, model, setter=None, namespace=None):
|
||||
self.model = model
|
||||
self.setter = setter
|
||||
self.namespace = namespace
|
||||
|
||||
def __set_name__(self, owner, name):
|
||||
self.name = name
|
||||
|
||||
def __get__(self, inst, cls):
|
||||
if inst is None:
|
||||
return self
|
||||
assert self.name is not None
|
||||
return inst._fields[self.name]
|
||||
|
||||
def __set__(self, inst, value):
|
||||
assert self.name is not None
|
||||
class_model = self.model
|
||||
inst_model = class_model()
|
||||
|
||||
self._set_namespace(inst_model, self.namespace, inst.__namespace__)
|
||||
inst._fields[self.name] = inst_model
|
||||
|
||||
# si hay setter manual se ejecuta
|
||||
# de lo contrario se asigna como texto del elemento
|
||||
setter = getattr(inst, self.setter or '', None)
|
||||
if callable(setter):
|
||||
setter(inst_model, value)
|
||||
else:
|
||||
inst_model._text = str(value)
|
||||
|
||||
|
||||
|
26
facho/model/fields/model.py
Normal file
26
facho/model/fields/model.py
Normal file
@ -0,0 +1,26 @@
|
||||
from .field import Field
|
||||
|
||||
class Model(Field):
|
||||
def __init__(self, model, namespace=None):
|
||||
self.model = model
|
||||
self.namespace = namespace
|
||||
|
||||
def __get__(self, inst, cls):
|
||||
if inst is None:
|
||||
return self
|
||||
assert self.name is not None
|
||||
return self._create_model(inst)
|
||||
|
||||
def __set__(self, inst, value):
|
||||
obj = self._create_model(inst)
|
||||
obj._text = str(value)
|
||||
|
||||
def _create_model(self, inst):
|
||||
try:
|
||||
return inst._fields[self.name]
|
||||
except KeyError:
|
||||
obj = self.model()
|
||||
self._set_namespace(obj, self.namespace, inst.__namespace__)
|
||||
inst._fields[self.name] = obj
|
||||
return obj
|
||||
|
167
tests/test_model.py
Normal file
167
tests/test_model.py
Normal file
@ -0,0 +1,167 @@
|
||||
#!/usr/bin/env python
|
||||
# -*- coding: utf-8 -*-
|
||||
# This file is part of facho. The COPYRIGHT file at the top level of
|
||||
# this repository contains the full copyright notices and license terms.
|
||||
|
||||
"""Tests for `facho` package."""
|
||||
|
||||
import pytest
|
||||
|
||||
import facho.model
|
||||
import facho.model.fields as fields
|
||||
|
||||
def test_model_to_element():
|
||||
class Person(facho.model.Model):
|
||||
__name__ = 'Person'
|
||||
|
||||
person = Person()
|
||||
|
||||
assert "<Person/>" == person.to_xml()
|
||||
|
||||
def test_model_to_element_with_attribute():
|
||||
class Person(facho.model.Model):
|
||||
__name__ = 'Person'
|
||||
id = fields.Attribute('id')
|
||||
|
||||
person = Person()
|
||||
person.id = 33
|
||||
|
||||
personb = Person()
|
||||
personb.id = 44
|
||||
|
||||
assert "<Person id=\"33\"/>" == person.to_xml()
|
||||
assert "<Person id=\"44\"/>" == personb.to_xml()
|
||||
|
||||
def test_model_to_element_with_attribute_as_element():
|
||||
class ID(facho.model.Model):
|
||||
__name__ = 'ID'
|
||||
|
||||
class Person(facho.model.Model):
|
||||
__name__ = 'Person'
|
||||
|
||||
id = fields.Many2One(ID)
|
||||
|
||||
person = Person()
|
||||
person.id = 33
|
||||
assert "<Person><ID>33</ID></Person>" == person.to_xml()
|
||||
|
||||
def test_many2one_with_custom_attributes():
|
||||
class TaxAmount(facho.model.Model):
|
||||
__name__ = 'TaxAmount'
|
||||
|
||||
currencyID = fields.Attribute('currencyID')
|
||||
|
||||
class TaxTotal(facho.model.Model):
|
||||
__name__ = 'TaxTotal'
|
||||
|
||||
amount = fields.Many2One(TaxAmount)
|
||||
|
||||
tax_total = TaxTotal()
|
||||
tax_total.amount = 3333
|
||||
tax_total.amount.currencyID = 'COP'
|
||||
assert '<TaxTotal><TaxAmount currencyID="COP">3333</TaxAmount></TaxTotal>' == tax_total.to_xml()
|
||||
|
||||
def test_many2one_with_custom_setter():
|
||||
|
||||
class PhysicalLocation(facho.model.Model):
|
||||
__name__ = 'PhysicalLocation'
|
||||
|
||||
id = fields.Attribute('ID')
|
||||
|
||||
class Party(facho.model.Model):
|
||||
__name__ = 'Party'
|
||||
|
||||
location = fields.Many2One(PhysicalLocation, setter='location_setter')
|
||||
|
||||
def location_setter(self, field, value):
|
||||
field.id = value
|
||||
|
||||
party = Party()
|
||||
party.location = 99
|
||||
assert '<Party><PhysicalLocation ID="99"/></Party>' == party.to_xml()
|
||||
|
||||
def test_field_model():
|
||||
class ID(facho.model.Model):
|
||||
__name__ = 'ID'
|
||||
|
||||
class Person(facho.model.Model):
|
||||
__name__ = 'Person'
|
||||
|
||||
id = fields.Model(ID)
|
||||
|
||||
person = Person()
|
||||
person.id = ID()
|
||||
person.id = 33
|
||||
assert "<Person><ID>33</ID></Person>" == person.to_xml()
|
||||
|
||||
def test_field_model_failed_initialization():
|
||||
class ID(facho.model.Model):
|
||||
__name__ = 'ID'
|
||||
|
||||
class Person(facho.model.Model):
|
||||
__name__ = 'Person'
|
||||
|
||||
id = fields.Model(ID)
|
||||
|
||||
|
||||
person = Person()
|
||||
person.id = 33
|
||||
assert "<Person><ID>33</ID></Person>" == person.to_xml()
|
||||
|
||||
def test_field_model_default_initialization_with_attributes():
|
||||
class ID(facho.model.Model):
|
||||
__name__ = 'ID'
|
||||
|
||||
reference = fields.Attribute('REFERENCE')
|
||||
|
||||
class Person(facho.model.Model):
|
||||
__name__ = 'Person'
|
||||
|
||||
id = fields.Model(ID)
|
||||
|
||||
person = Person()
|
||||
person.id = 33
|
||||
person.id.reference = 'haber'
|
||||
assert '<Person><ID REFERENCE="haber">33</ID></Person>' == person.to_xml()
|
||||
|
||||
def test_model_with_xml_namespace():
|
||||
class Person(facho.model.Model):
|
||||
__name__ = 'Person'
|
||||
__namespace__ = {
|
||||
'facho': 'http://lib.facho.cyou'
|
||||
}
|
||||
|
||||
person = Person()
|
||||
assert '<Person xmlns:facho="http://lib.facho.cyou"/>'
|
||||
|
||||
def test_model_with_xml_namespace_nested():
|
||||
class ID(facho.model.Model):
|
||||
__name__ = 'ID'
|
||||
|
||||
class Person(facho.model.Model):
|
||||
__name__ = 'Person'
|
||||
__namespace__ = {
|
||||
'facho': 'http://lib.facho.cyou'
|
||||
}
|
||||
|
||||
id = fields.Many2One(ID, namespace='facho')
|
||||
|
||||
person = Person()
|
||||
person.id = 33
|
||||
assert '<Person xmlns:facho="http://lib.facho.cyou"><facho:ID>33</facho:ID></Person>' == person.to_xml()
|
||||
|
||||
def test_field_model_with_namespace():
|
||||
class ID(facho.model.Model):
|
||||
__name__ = 'ID'
|
||||
|
||||
class Person(facho.model.Model):
|
||||
__name__ = 'Person'
|
||||
__namespace__ = {
|
||||
"facho": "http://lib.facho.cyou"
|
||||
}
|
||||
id = fields.Model(ID, namespace="facho")
|
||||
|
||||
|
||||
person = Person()
|
||||
person.id = 33
|
||||
assert '<Person xmlns:facho="http://lib.facho.cyou"><facho:ID>33</facho:ID></Person>' == person.to_xml()
|
Loading…
Reference in New Issue
Block a user