From 47ee55899ec709de6f0dfb9c705e0a336cb2ef76 Mon Sep 17 00:00:00 2001 From: Radoslav Bielik Date: Tue, 3 Mar 2015 11:07:07 +0100 Subject: [PATCH] Added a new include option for JSON (allow only specific properties - issue #12) --- README.md | 1 + .../GenericDomainClassJSONMarshaller.groovy | 170 ++++++++++-------- .../config/MarshallingConfig.groovy | 4 + .../config/MarshallingConfigBuilder.groovy | 5 +- ...icDomainClassJSONMarshallerUnitSpec.groovy | 14 ++ 5 files changed, 114 insertions(+), 80 deletions(-) diff --git a/README.md b/README.md index d0b63c9..b46d52e 100644 --- a/README.md +++ b/README.md @@ -255,6 +255,7 @@ Within the marshalling configuration closure there are several configuration opt * *serializer* is a configuration option which allows us to define closures with custom serialization behavior. This configuration options allows us to customize serialization output for existing property (json,xml) * *virtual* unlike *serializer* which will create completely new property * *ignore* is a comma separated list of properties which should be ignored during serialization process (json,xml) +* *include* is an exclusive comma separated list of properties which will be included, meaning properties not listed will not be included (except for those specified using the should* options). The *include* and *ignore* options are mutually exclusive - if both are defined, include takes priority (json) ###Named and marshaller specific configuration diff --git a/src/groovy/org/grails/plugins/marshallers/GenericDomainClassJSONMarshaller.groovy b/src/groovy/org/grails/plugins/marshallers/GenericDomainClassJSONMarshaller.groovy index 4df11ad..12c80d3 100644 --- a/src/groovy/org/grails/plugins/marshallers/GenericDomainClassJSONMarshaller.groovy +++ b/src/groovy/org/grails/plugins/marshallers/GenericDomainClassJSONMarshaller.groovy @@ -71,86 +71,16 @@ class GenericDomainClassJSONMarshaller implements ObjectMarshaller { GrailsDomainClassProperty[] properties = domainClass.getPersistentProperties() + boolean includeMode = false + if(mc.include?.size() > 0) + includeMode = true + for (GrailsDomainClassProperty property : properties) { - if(!mc.ignore?.contains(property.getName())){ - writer.key(property.getName()) - if(mc.serializer?.containsKey(property.getName())){ - mc.serializer[property.getName()].call(value,writer) - } - else{ - if (!property.isAssociation()) { - // Write non-relation property - Object val = beanWrapper.getPropertyValue(property.getName()) - json.convertAnother(val) - } - else { - Object referenceObject = beanWrapper.getPropertyValue(property.getName()) - if (mc.deep?.contains(property.getName())) { - if (referenceObject == null) { - writer.value(null) - } - else { - referenceObject = proxyHandler.unwrapIfProxy(referenceObject) - if (referenceObject instanceof SortedMap) { - referenceObject = new TreeMap((SortedMap) referenceObject) - } - else if (referenceObject instanceof SortedSet) { - referenceObject = new TreeSet((SortedSet) referenceObject) - } - else if (referenceObject instanceof Set) { - referenceObject = new HashSet((Set) referenceObject) - } - else if (referenceObject instanceof Map) { - referenceObject = new HashMap((Map) referenceObject) - } - else if (referenceObject instanceof Collection) { - referenceObject = new ArrayList((Collection) referenceObject) - } - json.convertAnother(referenceObject) - } - } - else { - if (referenceObject == null) { - json.value(null) - } - else { - GrailsDomainClass referencedDomainClass = property.getReferencedDomainClass() - - // Embedded are now always fully rendered - if (referencedDomainClass == null || property.isEmbedded() || GCU.isJdk5Enum(property.getType())) { - json.convertAnother(referenceObject) - } - else if (property.isOneToOne() || property.isManyToOne() || property.isEmbedded()) { - asShortObject(referenceObject, json, referencedDomainClass.getIdentifier(), referencedDomainClass) - } - else { - GrailsDomainClassProperty referencedIdProperty = referencedDomainClass.getIdentifier() - @SuppressWarnings("unused") - String refPropertyName = referencedDomainClass.getPropertyName() - if (referenceObject instanceof Collection) { - Collection o = (Collection) referenceObject - writer.array() - for (Object el : o) { - asShortObject(el, json, referencedIdProperty, referencedDomainClass) - } - writer.endArray() - } - else if (referenceObject instanceof Map) { - Map map = (Map) referenceObject - for (Map.Entry entry : map.entrySet()) { - String key = String.valueOf(entry.getKey()) - Object o = entry.getValue() - writer.object() - writer.key(key) - asShortObject(o, json, referencedIdProperty, referencedDomainClass) - writer.endObject() - } - } - } - } - } - } - } + if(includeMode && mc.include?.contains(property.getName())) { + serializeProperty(property, mc, beanWrapper, json, writer, value) + } + else if(!includeMode && !mc.ignore?.contains(property.getName())) { + serializeProperty(property, mc, beanWrapper, json, writer, value) } } @@ -161,6 +91,88 @@ class GenericDomainClassJSONMarshaller implements ObjectMarshaller { writer.endObject() } + + protected void serializeProperty(GrailsDomainClassProperty property, MarshallingConfig mc, + BeanWrapper beanWrapper, JSON json, JSONWriter writer, def value) { + writer.key(property.getName()) + if(mc.serializer?.containsKey(property.getName())){ + mc.serializer[property.getName()].call(value,writer) + } + else{ + if (!property.isAssociation()) { + // Write non-relation property + Object val = beanWrapper.getPropertyValue(property.getName()) + json.convertAnother(val) + } + else { + Object referenceObject = beanWrapper.getPropertyValue(property.getName()) + if (mc.deep?.contains(property.getName())) { + if (referenceObject == null) { + writer.value(null) + } + else { + referenceObject = proxyHandler.unwrapIfProxy(referenceObject) + if (referenceObject instanceof SortedMap) { + referenceObject = new TreeMap((SortedMap) referenceObject) + } + else if (referenceObject instanceof SortedSet) { + referenceObject = new TreeSet((SortedSet) referenceObject) + } + else if (referenceObject instanceof Set) { + referenceObject = new HashSet((Set) referenceObject) + } + else if (referenceObject instanceof Map) { + referenceObject = new HashMap((Map) referenceObject) + } + else if (referenceObject instanceof Collection) { + referenceObject = new ArrayList((Collection) referenceObject) + } + json.convertAnother(referenceObject) + } + } + else { + if (referenceObject == null) { + json.value(null) + } + else { + GrailsDomainClass referencedDomainClass = property.getReferencedDomainClass() + + // Embedded are now always fully rendered + if (referencedDomainClass == null || property.isEmbedded() || GCU.isJdk5Enum(property.getType())) { + json.convertAnother(referenceObject) + } + else if (property.isOneToOne() || property.isManyToOne() || property.isEmbedded()) { + asShortObject(referenceObject, json, referencedDomainClass.getIdentifier(), referencedDomainClass) + } + else { + GrailsDomainClassProperty referencedIdProperty = referencedDomainClass.getIdentifier() + @SuppressWarnings("unused") + String refPropertyName = referencedDomainClass.getPropertyName() + if (referenceObject instanceof Collection) { + Collection o = (Collection) referenceObject + writer.array() + for (Object el : o) { + asShortObject(el, json, referencedIdProperty, referencedDomainClass) + } + writer.endArray() + } + else if (referenceObject instanceof Map) { + Map map = (Map) referenceObject + for (Map.Entry entry : map.entrySet()) { + String key = String.valueOf(entry.getKey()) + Object o = entry.getValue() + writer.object() + writer.key(key) + asShortObject(o, json, referencedIdProperty, referencedDomainClass) + writer.endObject() + } + } + } + } + } + } + } + } protected void asShortObject(Object refObj, JSON json, GrailsDomainClassProperty idProperty, GrailsDomainClass referencedDomainClass) throws ConverterException { Object idValue diff --git a/src/groovy/org/grails/plugins/marshallers/config/MarshallingConfig.groovy b/src/groovy/org/grails/plugins/marshallers/config/MarshallingConfig.groovy index 85b8249..aeb929f 100644 --- a/src/groovy/org/grails/plugins/marshallers/config/MarshallingConfig.groovy +++ b/src/groovy/org/grails/plugins/marshallers/config/MarshallingConfig.groovy @@ -26,6 +26,10 @@ class MarshallingConfig { * list of field names which will be ignored during serialization */ List ignore + /** + * exclusive list of field names which will be included during serialization (mutually exclusive with "ignore") + */ + List include /** * configuration option allows us to define closures with custom serialization behavior */ diff --git a/src/groovy/org/grails/plugins/marshallers/config/MarshallingConfigBuilder.groovy b/src/groovy/org/grails/plugins/marshallers/config/MarshallingConfigBuilder.groovy index eb15fbe..f3fc4eb 100644 --- a/src/groovy/org/grails/plugins/marshallers/config/MarshallingConfigBuilder.groovy +++ b/src/groovy/org/grails/plugins/marshallers/config/MarshallingConfigBuilder.groovy @@ -12,7 +12,7 @@ class MarshallingConfigBuilder { } MarshallingConfigBuilder(MarshallingConfig c){ - ['clazz', 'deep','identifier','elementName','attribute','virtual','shouldOutputIdentifier','shouldOutputClass','shouldOutputVersion','ignore'].each{p-> + ['clazz','deep','identifier','elementName','attribute','virtual','shouldOutputIdentifier','shouldOutputClass','shouldOutputVersion','ignore','include'].each{p-> if(c[p]!=null){ config[p]=c[p] } @@ -41,6 +41,9 @@ class MarshallingConfigBuilder { void ignore(String... args){ config.ignore=args as List } + void include(String... args){ + config.include=args as List + } void virtual(Closure arg){ VirtualPropertiesBuilder builder=new VirtualPropertiesBuilder() arg.delegate=builder diff --git a/test/unit/org/grails/plugins/marshallers/config/GenericDomainClassJSONMarshallerUnitSpec.groovy b/test/unit/org/grails/plugins/marshallers/config/GenericDomainClassJSONMarshallerUnitSpec.groovy index ea4e4b9..81f1bd0 100644 --- a/test/unit/org/grails/plugins/marshallers/config/GenericDomainClassJSONMarshallerUnitSpec.groovy +++ b/test/unit/org/grails/plugins/marshallers/config/GenericDomainClassJSONMarshallerUnitSpec.groovy @@ -64,6 +64,20 @@ class GenericDomainClassJSONMarshallerUnitSpec extends Specification { m.id!=null } + def "putting a property to a list of includes should supress other properties serialization"(){ + given: + Invoice.marshalling = { include 'name' } + initialize() + when: + def j=new JSON(invoice) + def m=JSON.parse(j.toString()) + then: + m.name!=null + m.admin==null + m.created==null + m.id!=null + } + def "setting shouldOutputClass and shouldOutputVersion to true should output class and version info"(){ given: Invoice.marshalling = {