com.adobe.acs.commons.synth.children.ChildrenAsPropertyResource Maven / Gradle / Ivy
/*
* ACS AEM Commons
*
* Copyright (C) 2013 - 2023 Adobe
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.adobe.acs.commons.synth.children;
import com.adobe.acs.commons.json.JsonObjectUtil;
import com.google.gson.Gson;
import com.google.gson.JsonElement;
import com.google.gson.JsonObject;
import org.apache.jackrabbit.JcrConstants;
import org.apache.sling.api.resource.ModifiableValueMap;
import org.apache.sling.api.resource.Resource;
import org.apache.sling.api.resource.ResourceWrapper;
import org.apache.sling.api.resource.ValueMap;
import org.apache.sling.api.wrappers.ValueMapDecorator;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.jcr.RepositoryException;
import java.io.Serializable;
import java.lang.reflect.InvocationTargetException;
import java.time.Instant;
import java.time.LocalDateTime;
import java.time.ZoneId;
import java.time.format.DateTimeFormatter;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.Collections;
import java.util.Comparator;
import java.util.Date;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import java.util.TreeSet;
/**
* Class to wrapper a real resource to facilitate the persistence of children resources in a property (serialized as
* JSON).
*
* Can be used as follows...
*
* To write data:
*
* Resource real = resolve.getResource("/content/real");
* ChildrenAsPropertyResource wrapper = new ChildrenAsPropertyResource(real);
* Resource child = wrapper.create("child-1", "nt:unstructured");
* ModifiableValueMap mvm = child.adaptTo(ModifiableValueMap.class);
* mvm.put("prop-1", "some data");
* mvm.put("prop-2", Calendar.getInstance());
* wrapper.persist();
* resolver.commit();
*
* To read data:
*
* Resource real = resolve.getResource("/content/real");
* ChildrenAsPropertyResource wrapper = new ChildrenAsPropertyResource(real);
* for(Resource child : wrapper.getChildren()) {
* child.getValueMap().get("prop-1", String.class);
* }
*
*/
public class ChildrenAsPropertyResource extends ResourceWrapper {
private static final Logger log = LoggerFactory.getLogger(ChildrenAsPropertyResource.class);
private static final String EMPTY_JSON = "{}";
private static final String DEFAULT_PROPERTY_NAME = "children";
private final Resource resource;
private final String propertyName;
private Map lookupCache = null;
private Set orderedCache = null;
private Comparator comparator = null;
public static final Comparator RESOURCE_NAME_COMPARATOR = new ResourceNameComparator();
/**
* ResourceWrapper that allows resource children to be modeled in data stored into a property using the default
* property name of "children".
*
* @param resource the resource to store the children as properties on
* @throws InvalidDataFormatException
*/
public ChildrenAsPropertyResource(Resource resource) throws InvalidDataFormatException {
this(resource, DEFAULT_PROPERTY_NAME, null);
}
/**
* ResourceWrapper that allows resource children to be modeled in data stored into a property.
*
* @param resource the resource to store the children as properties on
* @param propertyName the property name to store the children as properties in
*/
public ChildrenAsPropertyResource(Resource resource, String propertyName) throws InvalidDataFormatException {
this(resource, propertyName, null);
}
/**
* ResourceWrapper that allows resource children to be modeled in data stored into a property.
*
* @param resource the resource to store the children as properties on
* @param propertyName the property name to store the children as properties in
* @param comparator the comparator used to order the serialized children
* @throws InvalidDataFormatException
*/
public ChildrenAsPropertyResource(Resource resource, String propertyName, Comparator comparator)
throws InvalidDataFormatException {
super(resource);
this.resource = resource;
this.propertyName = propertyName;
this.comparator = comparator;
if (this.comparator == null) {
this.orderedCache = new LinkedHashSet();
} else {
this.orderedCache = new TreeSet(this.comparator);
}
this.lookupCache = new HashMap();
for (SyntheticChildAsPropertyResource r : this.deserialize()) {
this.orderedCache.add(r);
this.lookupCache.put(r.getName(), r);
}
}
/**
* {@inheritDoc}
**/
@Override
public final Iterator listChildren() {
return this.orderedCache.iterator();
}
/**
* {@inheritDoc}
**/
@Override
public final Iterable getChildren() {
return Collections.unmodifiableSet(this.orderedCache);
}
/**
* {@inheritDoc}
**/
@Override
public final Resource getChild(String name) {
return this.lookupCache.get(name);
}
/**
* {@inheritDoc}
**/
@Override
public final Resource getParent() {
return this.resource;
}
public final Resource create(String name, String primaryType) throws RepositoryException {
return create(name, primaryType, null);
}
public final Resource create(String name, String primaryType, Map data) throws RepositoryException {
if (data == null) {
data = new HashMap();
}
if (data.containsKey(JcrConstants.JCR_PRIMARYTYPE) && primaryType != null) {
data.put(JcrConstants.JCR_PRIMARYTYPE, primaryType);
}
final SyntheticChildAsPropertyResource child =
new SyntheticChildAsPropertyResource(this.resource, name, data);
if (this.lookupCache.containsKey(child.getName())) {
log.info("Existing synthetic child [ {} ] overwritten", name);
}
this.lookupCache.put(child.getName(), child);
this.orderedCache.add(child);
return child;
}
/**
* Deletes the named child.
*
* Requires subsequent call to persist().
*
* @param name the child node name to delete
* @throws RepositoryException
*/
public final void delete(String name) throws RepositoryException {
if (this.lookupCache.containsKey(name)) {
Resource tmp = this.lookupCache.get(name);
this.orderedCache.remove(tmp);
this.lookupCache.remove(name);
}
}
/**
* Delete all children.
*
* Requires subsequent call to persist().
*
* @throws InvalidDataFormatException
*/
public final void deleteAll() throws InvalidDataFormatException {
// Clear the caches; requires serialize
if (this.comparator == null) {
this.orderedCache = new LinkedHashSet();
} else {
this.orderedCache = new TreeSet(this.comparator);
}
this.lookupCache = new HashMap();
}
/**
* Persist changes to the underlying valuemap so they are available for persisting to the JCR.
*
* @throws RepositoryException
*/
public final void persist() throws RepositoryException {
this.serialize();
}
/**
* Serializes all children data as JSON to the resource's propertyName.
*
* @throws InvalidDataFormatException
*/
private void serialize() throws InvalidDataFormatException {
final long start = System.currentTimeMillis();
final ModifiableValueMap modifiableValueMap = this.resource.adaptTo(ModifiableValueMap.class);
JsonObject childrenJSON = new JsonObject();
try {
// Add the new entries to the JSON
for (Resource childResource : this.orderedCache) {
childrenJSON.add(childResource.getName(), this.serializeToJSON(childResource));
}
if (childrenJSON.entrySet().size() > 0) {
// Persist the JSON back to the Node
modifiableValueMap.put(this.propertyName, childrenJSON.toString());
} else {
// Nothing to persist; delete the property
modifiableValueMap.remove(this.propertyName);
}
log.debug("Persist operation for [ {} ] in [ {} ms ]",
this.resource.getPath() + "/" + this.propertyName,
System.currentTimeMillis() - start);
} catch (NoSuchMethodException e) {
throw new InvalidDataFormatException(this.resource, this.propertyName, childrenJSON.toString());
} catch (IllegalAccessException e) {
throw new InvalidDataFormatException(this.resource, this.propertyName, childrenJSON.toString());
} catch (InvocationTargetException e) {
throw new InvalidDataFormatException(this.resource, this.propertyName, childrenJSON.toString());
}
}
/**
* Convert the serialized JSON data found in the node property to Resources.
*
* @return the list of children sorting using the comparator.
* @throws InvalidDataFormatException
*/
private List deserialize() throws InvalidDataFormatException {
final long start = System.currentTimeMillis();
final String propertyData = this.resource.getValueMap().get(this.propertyName, EMPTY_JSON);
List resources;
resources = deserializeToSyntheticChildResources(JsonObjectUtil.toJsonObject(propertyData));
if (this.comparator != null) {
Collections.sort(resources, this.comparator);
}
log.debug("Get operation for [ {} ] in [ {} ms ]",
this.resource.getPath() + "/" + this.propertyName,
System.currentTimeMillis() - start);
return resources;
}
/**
* Converts a list of SyntheticChildAsPropertyResource to their JSON representation, keeping the provided order.
*
* @param resourceToSerialize the resource to serialize to JSON.
* @return the JSONObject representing the resources.
*/
protected final JsonObject serializeToJSON(final Resource resourceToSerialize)
throws IllegalAccessException, NoSuchMethodException, InvocationTargetException {
DateTimeFormatter formatter = DateTimeFormatter.ISO_OFFSET_DATE_TIME;
final Map serializedData = new HashMap();
for (Map.Entry entry : resourceToSerialize.getValueMap().entrySet()) {
if (entry.getValue() instanceof Calendar) {
final Calendar cal = (Calendar) entry.getValue();
LocalDateTime date = LocalDateTime.ofInstant(Instant.ofEpochMilli(cal.getTimeInMillis()),
ZoneId.systemDefault());
serializedData.put(entry.getKey(), date.format(formatter));
} else if (entry.getValue() instanceof Date) {
final Date date = (Date) entry.getValue();
LocalDateTime ldt = LocalDateTime.ofInstant(date.toInstant(), ZoneId.systemDefault());
serializedData.put(entry.getKey(), ldt.format(formatter));
} else {
serializedData.put(entry.getKey(), entry.getValue());
}
}
Gson gson = new Gson();
return gson.toJsonTree(serializedData).getAsJsonObject();
}
/**
* Converts a JSONObject to the list of SyntheticChildAsPropertyResources.
*
* @param jsonObject the JSONObject to deserialize.
* @return the list of SyntheticChildAsPropertyResources the jsonObject represents.
*/
protected final List deserializeToSyntheticChildResources(JsonObject jsonObject) {
final List resources = new ArrayList<>();
for (Entry elem : jsonObject.entrySet()) {
final String nodeName = elem.getKey();
JsonObject entryJSON = elem.getValue().getAsJsonObject();
if (entryJSON == null) {
continue;
}
final ValueMap properties = new ValueMapDecorator(new HashMap<>());
for (Entry prop : entryJSON.entrySet()) {
final String propName = prop.getKey();
properties.put(propName, prop.getValue().getAsString());
}
resources.add(new SyntheticChildAsPropertyResource(this.getParent(), nodeName, properties));
}
return resources;
}
/**
* Sort by resource name ascending (resource.getName()).
*/
private static final class ResourceNameComparator implements Comparator, Serializable {
private static final long serialVersionUID = 0L;
@Override
public int compare(final Resource o1, final Resource o2) {
return o1.getName().compareTo(o2.getName().toString());
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy