diff --git a/lems/__init__.py b/lems/__init__.py
index 9cfdafb..15cacee 100644
--- a/lems/__init__.py
+++ b/lems/__init__.py
@@ -8,4 +8,4 @@
logger = logging.getLogger('LEMS')
-__version__ = '0.4.4'
+__version__ = '0.4.5'
diff --git a/lems/model/component.py b/lems/model/component.py
index ae5cc7d..65873bb 100644
--- a/lems/model/component.py
+++ b/lems/model/component.py
@@ -96,6 +96,41 @@ def toxml(self):
"""
return ''
+
+
+
+class Property(LEMSBase):
+ """
+ Store the specification of a property.
+ """
+
+ def __init__(self, name, dimension = None, description = ''):
+ """
+ Constructor.
+ """
+
+ self.name = name
+ """ Name of the property.
+ @type: str """
+
+ self.dimension = dimension
+ """ Physical dimensions of the property.
+ @type: str """
+
+ self.description = description
+ """ Description of the property.
+ @type: str """
+
+
+ def toxml(self):
+ """
+ Exports this object into a LEMS XML object
+ """
+
+ return ''
+
class IndexParameter(LEMSBase):
"""
@@ -615,6 +650,10 @@ def __init__(self):
""" Map of parameters in this component type.
@type: Map(str -> lems.model.component.Parameter) """
+ self.properties = Map()
+ """ Map of properties in this component type.
+ @type: Map(str -> lems.model.component.Property) """
+
self.derived_parameters = Map()
""" Map of derived_parameters in this component type.
@type: Map(str -> lems.model.component.Parameter) """
@@ -697,6 +736,17 @@ def add_parameter(self, parameter):
self.parameters[parameter.name] = parameter
+
+ def add_property(self, property):
+ """
+ Adds a property to this component type.
+
+ @param property: Property to be added.
+ @type property: lems.model.component.Property
+ """
+
+ self.properties[property.name] = property
+
def add_derived_parameter(self, derived_parameter):
"""
Adds a derived_parameter to this component type.
@@ -853,6 +903,8 @@ def add(self, child):
if isinstance(child, Parameter):
self.add_parameter(child)
+ elif isinstance(child, Property):
+ self.add_property(child)
elif isinstance(child, DerivedParameter):
self.add_derived_parameter(child)
elif isinstance(child, IndexParameter):
@@ -926,6 +978,9 @@ def toxml(self):
chxmlstr = ''
+ for property in self.properties:
+ chxmlstr += property.toxml()
+
for parameter in self.parameters:
chxmlstr += parameter.toxml()
diff --git a/lems/model/model.py b/lems/model/model.py
index e7366d0..5c68a8a 100644
--- a/lems/model/model.py
+++ b/lems/model/model.py
@@ -346,6 +346,8 @@ def merge_component_types(self, ct, base_ct):
else:
ct.parameters[parameter.name] = base_ct.parameters[parameter.name]
+ merge_maps(ct.properties, base_ct.properties)
+
merge_maps(ct.derived_parameters, base_ct.derived_parameters)
merge_maps(ct.index_parameters, base_ct.index_parameters)
merge_maps(ct.constants, base_ct.constants)
@@ -415,6 +417,11 @@ def fatten_component(self, c):
raise ModelError("Parameter '{0}' not initialized for component '{1}'",
parameter.name, c.id)
+ ### Resolve properties
+ for property in ct.properties:
+ property2 = property.copy()
+ fc.add(property2)
+
### Resolve derived_parameters
for derived_parameter in ct.derived_parameters:
derived_parameter2 = derived_parameter.copy()
diff --git a/lems/parser/LEMS.py b/lems/parser/LEMS.py
index 1403d5a..3860ab2 100644
--- a/lems/parser/LEMS.py
+++ b/lems/parser/LEMS.py
@@ -116,6 +116,7 @@ def init_parser(self):
'componentreference',
'exposure', 'eventport',
'fixed', 'link', 'parameter',
+ 'property',
'indexparameter',
'path', 'requirement',
'componentrequirement',
@@ -187,6 +188,7 @@ def init_parser(self):
self.tag_parse_table['onevent'] = self.parse_on_event
self.tag_parse_table['onstart'] = self.parse_on_start
self.tag_parse_table['parameter'] = self.parse_parameter
+ self.tag_parse_table['property'] = self.parse_property
self.tag_parse_table['path'] = self.parse_path
self.tag_parse_table['record'] = self.parse_record
self.tag_parse_table['regime'] = self.parse_regime
@@ -1181,6 +1183,37 @@ def parse_parameter(self, node):
parameter = Parameter(name, dimension)
self.current_component_type.add_parameter(parameter)
+
+ def parse_property(self, node):
+ """
+ Parses
+
+ @param node: Node containing the element
+ @type node: xml.etree.Element
+
+ @raise ParseError: Raised when the property does not have a name.
+ @raise ParseError: Raised when the property does not have a
+ dimension.
+ """
+
+ if self.current_component_type == None:
+ self.raise_error('Property can only be defined in ' +
+ 'a component type')
+
+ try:
+ name = node.lattrib['name']
+ except:
+ self.raise_error(' must specify a name')
+
+ try:
+ dimension = node.lattrib['dimension']
+ except:
+ self.raise_error("Property '{0}' has no dimension",
+ name)
+
+ property = Property(name, dimension)
+
+ self.current_component_type.add_property(property)
def parse_index_parameter(self, node):
diff --git a/lems/sim/build.py b/lems/sim/build.py
index a8c24c9..6d0c8fe 100644
--- a/lems/sim/build.py
+++ b/lems/sim/build.py
@@ -106,6 +106,10 @@ def build_runnable(self, component, parent = None, id_ = None):
for parameter in component.parameters:
runnable.add_instance_variable(parameter.name, parameter.numeric_value)
+
+
+ for property in component.properties:
+ raise NotImplementedError("Property element is not stable in PyLEMS yet, see https://github.com/LEMS/pylems/issues/16")
derived_parameter_code = []
diff --git a/testpy2.sh b/testpy2.sh
index e45f4e4..c538528 100755
--- a/testpy2.sh
+++ b/testpy2.sh
@@ -5,4 +5,4 @@ python2.7 examples/apitest.py
python2.7 examples/apitest2.py
python2.7 examples/loadtest.py
python2.7 lems/dlems/exportdlems.py
-python2.7 pylems -I ../NeuroML2/NeuroML2CoreTypes ../NeuroML2/LEMSexamples/LEMS_NML2_Ex2_Izh.xml
+python2.7 pylems -I ../NeuroML2/NeuroML2CoreTypes ../NeuroML2/LEMSexamples/LEMS_NML2_Ex5_DetCell.xml