Merge branch 'dataforms' into 'master'

Replace @nested_property with @property in dataforms

See merge request !124
This commit is contained in:
Yann Leboulanger 2017-08-27 22:08:48 +02:00
commit aa087c34ad
1 changed files with 268 additions and 297 deletions

View File

@ -49,14 +49,6 @@ class ExtendedNode(nbxmpp.Node, object):
extend.__class__ = cls extend.__class__ = cls
return extend return extend
# helper decorator to create properties in cleaner way
def nested_property(f):
ret = f()
p = {'doc': f.__doc__}
for v in ('fget', 'fset', 'fdel', 'doc'):
if v in ret.keys(): p[v]=ret[v]
return property(**p)
# helper to create fields from scratch # helper to create fields from scratch
def Field(typ, **attrs): def Field(typ, **attrs):
''' Helper function to create a field of given type. ''' ''' Helper function to create a field of given type. '''
@ -131,124 +123,116 @@ class DataField(ExtendedNode):
self.required = required self.required = required
self.options = options self.options = options
@nested_property @property
def type_(): # pylint: disable=E0202 def type_(self):
""" """
Type of field. Recognized values are: 'boolean', 'fixed', 'hidden', Type of field. Recognized values are: 'boolean', 'fixed', 'hidden',
'jid-multi', 'jid-single', 'list-multi', 'list-single', 'text-multi', 'jid-multi', 'jid-single', 'list-multi', 'list-single', 'text-multi',
'text-private', 'text-single'. If you set this to something different, 'text-private', 'text-single'. If you set this to something different,
DataField will store given name, but treat all data as text-single DataField will store given name, but treat all data as text-single
""" """
def fget(self):
t = self.getAttr('type') t = self.getAttr('type')
if t is None: if t is None:
return 'text-single' return 'text-single'
return t return t
def fset(self, value): @type_.setter
def type_(self, value):
assert isinstance(value, str) assert isinstance(value, str)
self.setAttr('type', value) self.setAttr('type', value)
return locals() @property
def var(self):
@nested_property
def var(): # pylint: disable=E0202
""" """
Field identifier Field identifier
""" """
def fget(self):
return self.getAttr('var') return self.getAttr('var')
def fset(self, value): @var.setter
def var(self, value):
assert isinstance(value, str) assert isinstance(value, str)
self.setAttr('var', value) self.setAttr('var', value)
def fdel(self): @var.deleter
def var(self):
self.delAttr('var') self.delAttr('var')
return locals() @property
def label(self):
@nested_property
def label(): # pylint: disable=E0202
""" """
Human-readable field name Human-readable field name
""" """
def fget(self):
l = self.getAttr('label') l = self.getAttr('label')
if not l: if not l:
l = self.var l = self.var
return l return l
def fset(self, value): @label.setter
def label(self, value):
assert isinstance(value, str) assert isinstance(value, str)
self.setAttr('label', value) self.setAttr('label', value)
def fdel(self): @label.deleter
def label(self):
if self.getAttr('label'): if self.getAttr('label'):
self.delAttr('label') self.delAttr('label')
return locals() @property
def description(self):
@nested_property
def description():
""" """
Human-readable description of field meaning Human-readable description of field meaning
""" """
def fget(self):
return self.getTagData('desc') or '' return self.getTagData('desc') or ''
def fset(self, value): @description.setter
def description(self, value):
assert isinstance(value, str) assert isinstance(value, str)
if value == '': if value == '':
fdel(self) del self.description
else: else:
self.setTagData('desc', value) self.setTagData('desc', value)
def fdel(self): @description.deleter
def description(self):
t = self.getTag('desc') t = self.getTag('desc')
if t is not None: if t is not None:
self.delChild(t) self.delChild(t)
return locals() @property
def required(self):
@nested_property
def required(): # pylint: disable=E0202
""" """
Controls whether this field required to fill. Boolean Controls whether this field required to fill. Boolean
""" """
def fget(self):
return bool(self.getTag('required')) return bool(self.getTag('required'))
def fset(self, value): @required.setter
def required(self, value):
t = self.getTag('required') t = self.getTag('required')
if t and not value: if t and not value:
self.delChild(t) self.delChild(t)
elif not t and value: elif not t and value:
self.addChild('required') self.addChild('required')
return locals() @property
def media(self):
@nested_property
def media():
""" """
Media data Media data
""" """
def fget(self):
media = self.getTag('media', namespace=nbxmpp.NS_DATA_MEDIA) media = self.getTag('media', namespace=nbxmpp.NS_DATA_MEDIA)
if media: if media:
return Media(media) return Media(media)
def fset(self, value): @media.setter
fdel(self) def media(self, value):
del self.media
self.addChild(node=value) self.addChild(node=value)
def fdel(self): @media.deleter
def media(self):
t = self.getTag('media') t = self.getTag('media')
if t is not None: if t is not None:
self.delChild(t) self.delChild(t)
return locals()
def is_valid(self): def is_valid(self):
return True return True
@ -256,68 +240,64 @@ class Uri(nbxmpp.Node):
def __init__(self, uri_tag): def __init__(self, uri_tag):
nbxmpp.Node.__init__(self, node=uri_tag) nbxmpp.Node.__init__(self, node=uri_tag)
@nested_property @property
def type_(): def type_(self):
""" """
uri type uri type
""" """
def fget(self):
return self.getAttr('type') return self.getAttr('type')
def fset(self, value): @type_.setter
def type_(self, value):
self.setAttr('type', value) self.setAttr('type', value)
def fdel(self): @type_.deleter
def type_(self):
self.delAttr('type') self.delAttr('type')
return locals() @property
def uri_data(self):
@nested_property
def uri_data():
""" """
uri data uri data
""" """
def fget(self):
return self.getData() return self.getData()
def fset(self, value): @uri_data.setter
def uri_data(self, value):
self.setData(value) self.setData(value)
def fdel(self): @uri_data.deleter
def uri_data(self):
self.setData(None) self.setData(None)
return locals()
class Media(nbxmpp.Node): class Media(nbxmpp.Node):
def __init__(self, media_tag): def __init__(self, media_tag):
nbxmpp.Node.__init__(self, node=media_tag) nbxmpp.Node.__init__(self, node=media_tag)
@nested_property @property
def uris(): def uris(self):
""" """
URIs of the media element. URIs of the media element.
""" """
def fget(self):
return map(Uri, self.getTags('uri')) return map(Uri, self.getTags('uri'))
def fset(self, value): @uris.setter
fdel(self) def uris(self, value):
del self.uris
for uri in value: for uri in value:
self.addChild(node=uri) self.addChild(node=uri)
def fdel(self): @uris.deleter
def uris(self):
for element in self.getTags('uri'): for element in self.getTags('uri'):
self.delChild(element) self.delChild(element)
return locals()
class BooleanField(DataField): class BooleanField(DataField):
@nested_property @property
def value(): # pylint: disable=E0202 def value(self):
""" """
Value of field. May contain True, False or None Value of field. May contain True, False or None
""" """
def fget(self):
v = self.getTagData('value') v = self.getTagData('value')
if v in ('0', 'false'): if v in ('0', 'false'):
return False return False
@ -327,54 +307,53 @@ class BooleanField(DataField):
return False # default value is False return False # default value is False
raise WrongFieldValue raise WrongFieldValue
def fset(self, value): @value.setter
def value(self, value):
self.setTagData('value', value and '1' or '0') self.setTagData('value', value and '1' or '0')
def fdel(self, value): @value.deleter
def value(self):
t = self.getTag('value') t = self.getTag('value')
if t is not None: if t is not None:
self.delChild(t) self.delChild(t)
return locals()
class StringField(DataField): class StringField(DataField):
""" """
Covers fields of types: fixed, hidden, text-private, text-single Covers fields of types: fixed, hidden, text-private, text-single
""" """
@nested_property @property
def value(): # pylint: disable=E0202 def value(self):
""" """
Value of field. May be any string Value of field. May be any string
""" """
def fget(self):
return self.getTagData('value') or '' return self.getTagData('value') or ''
def fset(self, value): @value.setter
def value(self, value):
assert isinstance(value, str) assert isinstance(value, str)
if value == '' and not self.required: if value == '' and not self.required:
return fdel(self) del self.value
return
self.setTagData('value', value) self.setTagData('value', value)
def fdel(self): @value.deleter
def value(self):
try: try:
self.delChild(self.getTag('value')) self.delChild(self.getTag('value'))
except ValueError: # if there already were no value tag except ValueError: # if there already were no value tag
pass pass
return locals()
class ListField(DataField): class ListField(DataField):
""" """
Covers fields of types: jid-multi, jid-single, list-multi, list-single Covers fields of types: jid-multi, jid-single, list-multi, list-single
""" """
@nested_property @property
def options(): # pylint: disable=E0202 def options(self):
""" """
Options Options
""" """
def fget(self):
options = [] options = []
for element in self.getTags('option'): for element in self.getTags('option'):
v = element.getTagData('value') v = element.getTagData('value')
@ -386,17 +365,17 @@ class ListField(DataField):
options.append((l, v)) options.append((l, v))
return options return options
def fset(self, values): @options.setter
fdel(self) def options(self, values):
del self.options
for value, label in values: for value, label in values:
self.addChild('option', {'label': label}).setTagData('value', value) self.addChild('option', {'label': label}).setTagData('value', value)
def fdel(self): @options.deleter
def options(self):
for element in self.getTags('option'): for element in self.getTags('option'):
self.delChild(element) self.delChild(element)
return locals()
def iter_options(self): def iter_options(self):
for element in self.iterTags('option'): for element in self.iterTags('option'):
v = element.getTagData('value') v = element.getTagData('value')
@ -438,28 +417,27 @@ class ListMultiField(ListField):
Covers list-multi fields Covers list-multi fields
""" """
@nested_property @property
def values(): def values(self):
""" """
Values held in field Values held in field
""" """
def fget(self):
values = [] values = []
for element in self.getTags('value'): for element in self.getTags('value'):
values.append(element.getData()) values.append(element.getData())
return values return values
def fset(self, values): @values.setter
fdel(self) def values(self, values):
del self.values
for value in values: for value in values:
self.addChild('value').setData(value) self.addChild('value').setData(value)
def fdel(self): @values.deleter
def values(self):
for element in self.getTags('value'): for element in self.getTags('value'):
self.delChild(element) self.delChild(element)
return locals()
def iter_values(self): def iter_values(self):
for element in self.getTags('value'): for element in self.getTags('value'):
yield element.getData() yield element.getData()
@ -488,30 +466,29 @@ class JidMultiField(ListMultiField):
return True return True
class TextMultiField(DataField): class TextMultiField(DataField):
@nested_property @property
def value(): # pylint: disable=E0202 def value(self):
""" """
Value held in field Value held in field
""" """
def fget(self):
value = '' value = ''
for element in self.iterTags('value'): for element in self.iterTags('value'):
value += '\n' + element.getData() value += '\n' + element.getData()
return value[1:] return value[1:]
def fset(self, value): @value.setter
fdel(self) def value(self, value):
del self.value
if value == '': if value == '':
return return
for line in value.split('\n'): for line in value.split('\n'):
self.addChild('value').setData(line) self.addChild('value').setData(line)
def fdel(self): @value.deleter
def value(self):
for element in self.getTags('value'): for element in self.getTags('value'):
self.delChild(element) self.delChild(element)
return locals()
class DataRecord(ExtendedNode): class DataRecord(ExtendedNode):
""" """
The container for data fields - an xml element which has DataField elements The container for data fields - an xml element which has DataField elements
@ -539,27 +516,26 @@ class DataRecord(ExtendedNode):
self.delChild(field) self.delChild(field)
self.fields = fields self.fields = fields
@nested_property @property
def fields(): # pylint: disable=E0202 def fields(self):
""" """
List of fields in this record List of fields in this record
""" """
def fget(self):
return self.getTags('field') return self.getTags('field')
def fset(self, fields): @fields.setter
fdel(self) def fields(self, fields):
del self.fields
for field in fields: for field in fields:
if not isinstance(field, DataField): if not isinstance(field, DataField):
ExtendField(field) ExtendField(field)
self.addChild(node=field) self.addChild(node=field)
def fdel(self): @fields.deleter
def fields(self):
for element in self.getTags('field'): for element in self.getTags('field'):
self.delChild(element) self.delChild(element)
return locals()
def iter_fields(self): def iter_fields(self):
""" """
Iterate over fields in this record. Do not take associated into account Iterate over fields in this record. Do not take associated into account
@ -597,69 +573,65 @@ class DataForm(ExtendedNode):
if instructions is not None: if instructions is not None:
self.instructions=instructions self.instructions=instructions
@nested_property @property
def type_(): # pylint: disable=E0202 def type_(self):
""" """
Type of the form. Must be one of: 'form', 'submit', 'cancel', 'result'. Type of the form. Must be one of: 'form', 'submit', 'cancel', 'result'.
'form' - this form is to be filled in; you will be able soon to do: 'form' - this form is to be filled in; you will be able soon to do:
filledform = DataForm(replyto=thisform) filledform = DataForm(replyto=thisform)
""" """
def fget(self):
return self.getAttr('type') return self.getAttr('type')
def fset(self, type_): @type_.setter
def type_(self, type_):
assert type_ in ('form', 'submit', 'cancel', 'result') assert type_ in ('form', 'submit', 'cancel', 'result')
self.setAttr('type', type_) self.setAttr('type', type_)
return locals() @property
def title(self):
@nested_property
def title(): # pylint: disable=E0202
""" """
Title of the form Title of the form
Human-readable, should not contain any \\r\\n. Human-readable, should not contain any \\r\\n.
""" """
def fget(self):
return self.getTagData('title') return self.getTagData('title')
def fset(self, title): @title.setter
def title(self, title):
self.setTagData('title', title) self.setTagData('title', title)
def fdel(self): @title.deleter
def title(self):
try: try:
self.delChild('title') self.delChild('title')
except ValueError: except ValueError:
pass pass
return locals() @property
def instructions(self):
@nested_property
def instructions(): # pylint: disable=E0202
""" """
Instructions for this form Instructions for this form
Human-readable, may contain \\r\\n. Human-readable, may contain \\r\\n.
""" """
# TODO: the same code is in TextMultiField. join them # TODO: the same code is in TextMultiField. join them
def fget(self):
value = '' value = ''
for valuenode in self.getTags('instructions'): for valuenode in self.getTags('instructions'):
value += '\n' + valuenode.getData() value += '\n' + valuenode.getData()
return value[1:] return value[1:]
def fset(self, value): @instructions.setter
fdel(self) def instructions(self, value):
del self.instructions
if value == '': return if value == '': return
for line in value.split('\n'): for line in value.split('\n'):
self.addChild('instructions').setData(line) self.addChild('instructions').setData(line)
def fdel(self): @instructions.deleter
def instructions(self):
for value in self.getTags('instructions'): for value in self.getTags('instructions'):
self.delChild(value) self.delChild(value)
return locals()
class SimpleDataForm(DataForm, DataRecord): class SimpleDataForm(DataForm, DataRecord):
def __init__(self, type_=None, title=None, instructions=None, fields=None, \ def __init__(self, type_=None, title=None, instructions=None, fields=None, \
extend=None): extend=None):
@ -711,39 +683,39 @@ class MultipleDataForm(DataForm):
reported_tag = self.getTag('reported') reported_tag = self.getTag('reported')
self.reported = DataRecord(extend=reported_tag) self.reported = DataRecord(extend=reported_tag)
@nested_property @property
def items(): # pylint: disable=E0202 def items(self):
""" """
A list of all records A list of all records
""" """
def fget(self):
return list(self.iter_records()) return list(self.iter_records())
def fset(self, records): @items.setter
fdel(self) def items(self, records):
del self.items
for record in records: for record in records:
if not isinstance(record, DataRecord): if not isinstance(record, DataRecord):
DataRecord(extend=record) DataRecord(extend=record)
self.addChild(node=record) self.addChild(node=record)
def fdel(self): @items.deleter
def items(self):
for record in self.getTags('item'): for record in self.getTags('item'):
self.delChild(record) self.delChild(record)
return locals()
def iter_records(self): def iter_records(self):
for record in self.getTags('item'): for record in self.getTags('item'):
yield record yield record
# @nested_property # @property
# def reported(): # def reported(self):
# """ # """
# DataRecord that contains descriptions of fields in records # DataRecord that contains descriptions of fields in records
# """ # """
# def fget(self):
# return self.getTag('reported') # return self.getTag('reported')
# def fset(self, record): #
# @reported.setter
# def reported(self, record):
# try: # try:
# self.delChild('reported') # self.delChild('reported')
# except: # except:
@ -751,4 +723,3 @@ class MultipleDataForm(DataForm):
# #
# record.setName('reported') # record.setName('reported')
# self.addChild(node=record) # self.addChild(node=record)
# return locals()