blob: 64d7e3285d15668be1377da7cef505b949a6cd60 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 org.apache.sling.jmx.provider.impl;
import java.lang.reflect.Array;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Set;
import java.util.TreeMap;
import javax.management.MBeanAttributeInfo;
import javax.management.openmbean.CompositeData;
import javax.management.openmbean.CompositeType;
import javax.management.openmbean.TabularData;
import javax.management.openmbean.TabularType;
import org.apache.sling.api.resource.AbstractResource;
import org.apache.sling.api.resource.Resource;
import org.apache.sling.api.resource.ResourceMetadata;
import org.apache.sling.api.resource.ResourceResolver;
import org.apache.sling.api.resource.ValueMap;
import org.apache.sling.api.wrappers.DeepReadValueMapDecorator;
import org.apache.sling.api.wrappers.ValueMapDecorator;
public class AttributeResource extends AbstractResource {
private final String path;
private final ResourceResolver resourceResolver;
private final ResourceMetadata metadata = new ResourceMetadata();
private final MBeanAttributeInfo info;
private final Object attrValue;
private final AttributesResource parent;
public AttributeResource(final ResourceResolver resolver,
final String path,
final MBeanAttributeInfo info,
final Object value,
final AttributesResource parent) {
this.resourceResolver = resolver;
this.path = path;
this.info = info;
this.attrValue = value;
this.parent = parent;
}
@Override
public Resource getParent() {
return this.parent;
}
/**
* @see org.apache.sling.api.resource.Resource#getPath()
*/
public String getPath() {
return this.path;
}
/**
* @see org.apache.sling.api.resource.Resource#getResourceType()
*/
public String getResourceType() {
return Constants.TYPE_ATTRIBUTE;
}
/**
* @see org.apache.sling.api.resource.Resource#getResourceSuperType()
*/
public String getResourceSuperType() {
return null;
}
/**
* @see org.apache.sling.api.resource.Resource#getResourceMetadata()
*/
public ResourceMetadata getResourceMetadata() {
return metadata;
}
/**
* @see org.apache.sling.api.resource.Resource#getResourceResolver()
*/
public ResourceResolver getResourceResolver() {
return this.resourceResolver;
}
@Override
public <AdapterType> AdapterType adaptTo(final Class<AdapterType> type) {
if ( type == ValueMap.class || type == Map.class ) {
final Map<String, Object> propMap = this.getPropertiesMap();
return (AdapterType) new DeepReadValueMapDecorator(this, new ValueMapDecorator(propMap));
}
return super.adaptTo(type);
}
private Map<String, Object> getPropertiesMap() {
final Map<String, Object> result = new HashMap<String, Object>();
result.put(Constants.PROP_RESOURCE_TYPE, this.getResourceType());
if ( this.getResourceSuperType() != null ) {
result.put(Constants.PROP_RESOURCE_SUPER_TYPE, this.getResourceSuperType());
}
if ( info.getDescription() != null ) {
result.put(Constants.PROP_DESCRIPTION, info.getDescription());
}
result.put(Constants.PROP_TYPE, info.getType());
try {
final Object value = attrValue;
if ( value != null ) {
if ( value.getClass().isArray() ) {
final int length = Array.getLength(value);
final Object[] values = new Object[length];
for (int i = 0; i < length; i ++) {
final Object o = Array.get(value, i);
values[i] = convert(o);
}
result.put(Constants.PROP_VALUE, values);
} else if (value instanceof TabularData) {
// Nothing to do, value is child resource
} else if (value instanceof CompositeData) {
// Nothing to do, value is child resource
} else {
result.put(Constants.PROP_VALUE, convert(value));
}
}
} catch (final Exception ignore) {
// ignore, but put this as info
result.put("mbean:exception", ignore.getMessage());
}
return result;
}
private Object convert(final Object value) {
if ( value == null ) {
return "";
} else if ( value instanceof String ) {
return value;
} else if ( value instanceof Number ) {
return value;
} else if ( value instanceof Boolean ) {
return value;
} else if ( value instanceof Character ) {
return value;
}
return value.toString();
}
public Resource getChildResource(final String subPath) {
final Map<String, Object> childStructure = this.convertData();
if ( childStructure != null ) {
final String[] segments = subPath.split("/");
Map<String, Object> current = childStructure;
for(final String path : segments) {
final Object child = current.get(path);
if ( child == null ) {
return null;
}
if ( !(child instanceof Map) ) {
return null;
}
current = (Map<String, Object>)child;
}
return new MapResource(this.getResourceResolver(), this.getPath(), current, this);
}
return null;
}
private volatile Map<String, Object> convertedValue;
private Map<String, Object> convertData() {
if ( convertedValue == null ) {
if ( attrValue instanceof TabularData ) {
convertedValue = convertObject((TabularData)attrValue);
} else if ( attrValue instanceof CompositeData ) {
convertedValue = convertObject((CompositeData)attrValue);
}
}
return convertedValue;
}
private Map<String, Object> convertObject(final TabularData td) {
final TabularType type = td.getTabularType();
final Map<String, Object> result = new HashMap<String, Object>();
result.put(Constants.PROP_RESOURCE_SUPER_TYPE, Constants.TYPE_ATTRIBUTES);
result.put(Constants.PROP_RESOURCE_TYPE, type.getTypeName());
final Map<String, Map<String, Object>> rows = new LinkedHashMap<String, Map<String, Object>>();
int rowIndex = 1;
@SuppressWarnings("unchecked")
final List<CompositeData> values = new ArrayList<CompositeData>((Collection<CompositeData>)td.values());
Collections.sort(values, new Comparator<CompositeData>() {
public int compare(final CompositeData o1, final CompositeData o2) {
for(final String name : type.getIndexNames()) {
final Object value1 = o1.get(name);
final Object value2 = o2.get(name);
final int result;
if ( value1 instanceof Comparable ) {
result = ((Comparable)value1).compareTo(value2);
} else {
result = value1.toString().compareTo(value2.toString());
}
if ( result != 0 ) {
return result;
}
}
return 0;
}
});
for(final CompositeData data : values) {
rows.put(String.valueOf(rowIndex), convertObject(data));
rowIndex++;
}
result.put(Constants.RSRC_VALUE, rows);
return result;
}
private Map<String, Object> convertObject(final CompositeData cd) {
final CompositeType type = cd.getCompositeType();
final Map<String, Object> result = new HashMap<String, Object>();
result.put(Constants.PROP_RESOURCE_SUPER_TYPE, Constants.TYPE_ATTRIBUTES);
result.put(Constants.PROP_RESOURCE_TYPE, type.getTypeName());
final Map<String, Object> attrMap = new TreeMap<String, Object>();
attrMap.put(Constants.PROP_RESOURCE_TYPE, Constants.TYPE_ATTRIBUTES);
result.put(Constants.RSRC_ATTRIBUTES, attrMap);
final Set<String> names = type.keySet();
for(final String name : names) {
final Map<String, Object> dataMap = new HashMap<String, Object>();
attrMap.put(name, dataMap);
dataMap.put(ResourceResolver.PROPERTY_RESOURCE_TYPE, type.getType(name));
dataMap.put(Constants.PROP_RESOURCE_SUPER_TYPE, Constants.TYPE_ATTRIBUTE);
if ( type.getDescription() != null ) {
dataMap.put(Constants.PROP_DESCRIPTION, type.getDescription());
}
dataMap.put(Constants.PROP_TYPE, type.getType(name).getTypeName());
final Object value = cd.get(name);
if ( value != null ) {
if ( value.getClass().isArray() ) {
final int length = Array.getLength(value);
final Object[] values = new Object[length];
for (int i = 0; i < length; i ++) {
final Object o = Array.get(value, i);
values[i] = convert(o);
}
dataMap.put(Constants.PROP_VALUE, values);
} else if (value instanceof TabularData) {
dataMap.put(Constants.RSRC_VALUE, convertObject((TabularData)value));
} else if (value instanceof CompositeData) {
dataMap.put(Constants.RSRC_VALUE, convertObject((CompositeData)value));
} else {
dataMap.put(Constants.PROP_VALUE, convert(value));
}
}
}
return result;
}
public Iterator<Resource> getChildren(final String parentPath, final String subPath) {
final Map<String, Object> childStructure = this.convertData();
if ( childStructure != null ) {
Map<String, Object> current = childStructure;
if ( subPath != null ) {
final String[] segments = subPath.split("/");
for(final String path : segments) {
final Object child = current.get(path);
if ( child == null ) {
return null;
}
if ( !(child instanceof Map) ) {
return null;
}
current = (Map<String, Object>)child;
}
}
if ( current.size() == 0 ) {
return null;
}
final Iterator<Map.Entry<String, Object>> childIter = current.entrySet().iterator();
return new Iterator<Resource>() {
private Map.Entry<String, Object> next = this.seek();
private Map.Entry<String, Object> seek() {
while ( childIter.hasNext() ) {
final Map.Entry<String, Object> c = childIter.next();
if ( c.getValue() instanceof Map ) {
return c;
}
}
return null;
}
public void remove() {
throw new UnsupportedOperationException("remove");
}
public Resource next() {
final Map.Entry<String, Object> props = next;
if ( props == null ) {
throw new NoSuchElementException();
}
next = seek();
return new MapResource(getResourceResolver(), parentPath + '/' + props.getKey(), (Map)props.getValue(), AttributeResource.this);
}
public boolean hasNext() {
return next != null;
}
};
}
return null;
}
}