From 0045498415225ca355059f939ce39a8b16488fbe Mon Sep 17 00:00:00 2001 From: ziirish Date: Fri, 18 Oct 2019 11:59:16 +0200 Subject: [PATCH] fix: make Wildcard fields support Nested and List fields (fix #728) --- flask_restplus/fields.py | 6 +++++- tests/test_marshalling.py | 31 +++++++++++++++++++++++++++++++ 2 files changed, 36 insertions(+), 1 deletion(-) diff --git a/flask_restplus/fields.py b/flask_restplus/fields.py index ad2e00fe..b75ab5ca 100644 --- a/flask_restplus/fields.py +++ b/flask_restplus/fields.py @@ -281,9 +281,11 @@ def format(self, value): def is_attr(val): return self.container.attribute and hasattr(val, self.container.attribute) + if value is None: + return [] return [ self.container.output(idx, - val if (isinstance(val, dict) or is_attr(val)) and not is_nested else value) + val if (isinstance(val, dict) or is_attr(val)) and not is_nested else value) for idx, val in enumerate(value) ] @@ -793,6 +795,8 @@ def output(self, key, obj, ordered=False): return self.container.format(self.default) return None + if isinstance(self.container, Nested): + return marshal(value, self.container.nested, skip_none=self.container.skip_none, ordered=ordered) return self.container.format(value) def schema(self): diff --git a/tests/test_marshalling.py b/tests/test_marshalling.py index 6e971267..2e0ac6f8 100644 --- a/tests/test_marshalling.py +++ b/tests/test_marshalling.py @@ -25,6 +25,37 @@ def test_marshal(self): assert not isinstance(output, OrderedDict) assert output == {'foo': 'bar'} + def test_marshal_wildcard_nested(self): + nest = fields.Nested(OrderedDict([('thumbnail', fields.String), ('video', fields.String)])) + wild = fields.Wildcard(nest) + wildcard_fields = OrderedDict([('*', wild)]) + model = OrderedDict([('preview', fields.Nested(wildcard_fields))]) + sub_dict = OrderedDict([ + ('9:16', {'thumbnail': 24, 'video': 12}), + ('16:9', {'thumbnail': 25, 'video': 11}), + ('1:1', {'thumbnail': 26, 'video': 10}) + ]) + marshal_dict = OrderedDict([('preview', sub_dict)]) + output = marshal(marshal_dict, model) + assert output == {'preview': {'1:1': {'thumbnail': '26', 'video': '10'}, + '16:9': {'thumbnail': '25', 'video': '11'}, + '9:16': {'thumbnail': '24', 'video': '12'}}} + + def test_marshal_wildcard_list(self): + wild = fields.Wildcard(fields.List(fields.String)) + wildcard_fields = OrderedDict([('*', wild)]) + model = OrderedDict([('preview', fields.Nested(wildcard_fields))]) + sub_dict = OrderedDict([ + ('1:1', [1, 2, 3]), + ('16:9', [4, 5, 6]), + ('9:16', [7, 8, 9]) + ]) + marshal_dict = OrderedDict([('preview', sub_dict)]) + output = marshal(marshal_dict, model) + assert output == {'preview': {'9:16': ['7', '8', '9'], + '16:9': ['4', '5', '6'], + '1:1': ['1', '2', '3']}} + def test_marshal_with_envelope(self): model = OrderedDict([('foo', fields.Raw)]) marshal_dict = OrderedDict([('foo', 'bar'), ('bat', 'baz')])