Merge "Create a custom StringField that can process functions"

This commit is contained in:
Jenkins 2016-08-04 11:05:11 +00:00 committed by Gerrit Code Review
commit d538381ebe
4 changed files with 80 additions and 8 deletions

View File

@ -14,6 +14,8 @@
# under the License.
import ast
import hashlib
import inspect
import six
from oslo_versionedobjects import fields as object_fields
@ -33,6 +35,37 @@ class StringField(object_fields.StringField):
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):
pass

View File

@ -117,15 +117,10 @@ class Node(base.IronicObject, object_base.VersionedObjectDictCompat):
'extra': object_fields.FlexibleDictField(nullable=True),
'network_interface': object_fields.StringField(
nullable=False, default=_default_network_interface()),
'network_interface': object_fields.StringFieldThatAcceptsCallable(
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):
"""Check if the input of local_gb, cpus and memory_mb are valid.

View File

@ -13,6 +13,8 @@
# License for the specific language governing permissions and limitations
# under the License.
import hashlib
import inspect
from ironic.common import exception
from ironic.objects import fields
@ -61,3 +63,45 @@ class TestFlexibleDictField(test_base.TestCase):
# nullable
self.field = fields.FlexibleDictField(nullable=True)
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))

View File

@ -404,7 +404,7 @@ class TestObject(_LocalTest, _TestObject):
# 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.
expected_object_fingerprints = {
'Node': '1.18-8cdb6010014b29f17ca636bef72b7800',
'Node': '1.18-37a1d39ba8a4957f505dda936ac9146b',
'MyObj': '1.5-4f5efe8f0fcaf182bbe1c7fe3ba858db',
'Chassis': '1.3-d656e039fd8ae9f34efc232ab3980905',
'Port': '1.6-609504503d68982a10f495659990084b',