Merge "Create a custom StringField that can process functions"
This commit is contained in:
commit
d538381ebe
@ -14,6 +14,8 @@
|
|||||||
# under the License.
|
# under the License.
|
||||||
|
|
||||||
import ast
|
import ast
|
||||||
|
import hashlib
|
||||||
|
import inspect
|
||||||
import six
|
import six
|
||||||
|
|
||||||
from oslo_versionedobjects import fields as object_fields
|
from oslo_versionedobjects import fields as object_fields
|
||||||
@ -33,6 +35,37 @@ class StringField(object_fields.StringField):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class StringAcceptsCallable(object_fields.String):
|
||||||
|
@staticmethod
|
||||||
|
def coerce(obj, attr, value):
|
||||||
|
if callable(value):
|
||||||
|
value = value()
|
||||||
|
return super(StringAcceptsCallable, StringAcceptsCallable).coerce(
|
||||||
|
obj, attr, value)
|
||||||
|
|
||||||
|
|
||||||
|
class StringFieldThatAcceptsCallable(object_fields.StringField):
|
||||||
|
"""Custom StringField object that allows for functions as default
|
||||||
|
|
||||||
|
In some cases we need to allow for dynamic defaults based on configuration
|
||||||
|
options, this StringField object allows for a function to be passed as a
|
||||||
|
default, and will only process it at the point the field is coerced
|
||||||
|
"""
|
||||||
|
|
||||||
|
AUTO_TYPE = StringAcceptsCallable()
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
default = self._default
|
||||||
|
if (self._default != object_fields.UnspecifiedDefault and
|
||||||
|
callable(self._default)):
|
||||||
|
default = "%s-%s" % (
|
||||||
|
self._default.__name__,
|
||||||
|
hashlib.md5(inspect.getsource(
|
||||||
|
self._default).encode()).hexdigest())
|
||||||
|
return '%s(default=%s,nullable=%s)' % (self._type.__class__.__name__,
|
||||||
|
default, self._nullable)
|
||||||
|
|
||||||
|
|
||||||
class DateTimeField(object_fields.DateTimeField):
|
class DateTimeField(object_fields.DateTimeField):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
@ -117,15 +117,10 @@ class Node(base.IronicObject, object_base.VersionedObjectDictCompat):
|
|||||||
|
|
||||||
'extra': object_fields.FlexibleDictField(nullable=True),
|
'extra': object_fields.FlexibleDictField(nullable=True),
|
||||||
|
|
||||||
'network_interface': object_fields.StringField(
|
'network_interface': object_fields.StringFieldThatAcceptsCallable(
|
||||||
nullable=False, default=_default_network_interface()),
|
nullable=False, default=_default_network_interface),
|
||||||
}
|
}
|
||||||
|
|
||||||
def __init__(self, context=None, **kwargs):
|
|
||||||
self.fields['network_interface']._default = (
|
|
||||||
_default_network_interface())
|
|
||||||
super(Node, self).__init__(context, **kwargs)
|
|
||||||
|
|
||||||
def _validate_property_values(self, properties):
|
def _validate_property_values(self, properties):
|
||||||
"""Check if the input of local_gb, cpus and memory_mb are valid.
|
"""Check if the input of local_gb, cpus and memory_mb are valid.
|
||||||
|
|
||||||
|
@ -13,6 +13,8 @@
|
|||||||
# License for the specific language governing permissions and limitations
|
# License for the specific language governing permissions and limitations
|
||||||
# under the License.
|
# under the License.
|
||||||
|
|
||||||
|
import hashlib
|
||||||
|
import inspect
|
||||||
|
|
||||||
from ironic.common import exception
|
from ironic.common import exception
|
||||||
from ironic.objects import fields
|
from ironic.objects import fields
|
||||||
@ -61,3 +63,45 @@ class TestFlexibleDictField(test_base.TestCase):
|
|||||||
# nullable
|
# nullable
|
||||||
self.field = fields.FlexibleDictField(nullable=True)
|
self.field = fields.FlexibleDictField(nullable=True)
|
||||||
self.assertEqual({}, self.field.coerce('obj', 'attr', None))
|
self.assertEqual({}, self.field.coerce('obj', 'attr', None))
|
||||||
|
|
||||||
|
|
||||||
|
class TestStringFieldThatAcceptsCallable(test_base.TestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestStringFieldThatAcceptsCallable, self).setUp()
|
||||||
|
|
||||||
|
def test_default_function():
|
||||||
|
return "default value"
|
||||||
|
|
||||||
|
self.test_default_function_hash = hashlib.md5(
|
||||||
|
inspect.getsource(test_default_function).encode()).hexdigest()
|
||||||
|
self.field = fields.StringFieldThatAcceptsCallable(
|
||||||
|
default=test_default_function)
|
||||||
|
|
||||||
|
def test_coerce_string(self):
|
||||||
|
self.assertEqual("value", self.field.coerce('obj', 'attr', "value"))
|
||||||
|
|
||||||
|
def test_coerce_function(self):
|
||||||
|
def test_function():
|
||||||
|
return "value"
|
||||||
|
self.assertEqual("value",
|
||||||
|
self.field.coerce('obj', 'attr', test_function))
|
||||||
|
|
||||||
|
def test_coerce_invalid_type(self):
|
||||||
|
self.assertRaises(ValueError, self.field.coerce,
|
||||||
|
'obj', 'attr', ('invalid', 'tuple'))
|
||||||
|
|
||||||
|
def test_coerce_function_invalid_type(self):
|
||||||
|
def test_function():
|
||||||
|
return ('invalid', 'tuple',)
|
||||||
|
self.assertRaises(ValueError,
|
||||||
|
self.field.coerce, 'obj', 'attr', test_function)
|
||||||
|
|
||||||
|
def test_coerce_default_as_function(self):
|
||||||
|
self.assertEqual("default value",
|
||||||
|
self.field.coerce('obj', 'attr', None))
|
||||||
|
|
||||||
|
def test__repr__includes_default_function_name_and_source_hash(self):
|
||||||
|
expected = ('StringAcceptsCallable(default=test_default_function-%s,'
|
||||||
|
'nullable=False)' % self.test_default_function_hash)
|
||||||
|
self.assertEqual(expected, repr(self.field))
|
||||||
|
@ -404,7 +404,7 @@ class TestObject(_LocalTest, _TestObject):
|
|||||||
# version bump. It is md5 hash of object fields and remotable methods.
|
# version bump. It is md5 hash of object fields and remotable methods.
|
||||||
# The fingerprint values should only be changed if there is a version bump.
|
# The fingerprint values should only be changed if there is a version bump.
|
||||||
expected_object_fingerprints = {
|
expected_object_fingerprints = {
|
||||||
'Node': '1.18-8cdb6010014b29f17ca636bef72b7800',
|
'Node': '1.18-37a1d39ba8a4957f505dda936ac9146b',
|
||||||
'MyObj': '1.5-4f5efe8f0fcaf182bbe1c7fe3ba858db',
|
'MyObj': '1.5-4f5efe8f0fcaf182bbe1c7fe3ba858db',
|
||||||
'Chassis': '1.3-d656e039fd8ae9f34efc232ab3980905',
|
'Chassis': '1.3-d656e039fd8ae9f34efc232ab3980905',
|
||||||
'Port': '1.6-609504503d68982a10f495659990084b',
|
'Port': '1.6-609504503d68982a10f495659990084b',
|
||||||
|
Loading…
x
Reference in New Issue
Block a user