blob: 9e7c8499b75a759fa422e917294b7c9b2291e625 [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.Set;
import java.util.TreeMap;
import javax.management.MBeanAttributeInfo;
import javax.management.MBeanServer;
import javax.management.ObjectName;
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.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 MBeanServer server;
private final ObjectName on;
public AttributeResource(final MBeanServer server,
final ObjectName on,
final ResourceResolver resolver, final String p, final MBeanAttributeInfo mai) {
this.resourceResolver = resolver;
this.path = p;
this.info = mai;
this.on = on;
this.server = server;
}
/**
* @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 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 = server.getAttribute(this.on, info.getName());
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);
}
return null;
}
private Map<String, Object> convertData() {
try {
final Object value = server.getAttribute(this.on, info.getName());
if ( value instanceof TabularData ) {
return convertObject((TabularData)value);
} else if ( value instanceof CompositeData ) {
return convertObject((CompositeData)value);
}
} catch (final Exception ignore) {
// ignore and return null
}
return null;
}
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(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;
}
}
final Iterator<Map.Entry<String, Object>> removeIter = current.entrySet().iterator();
while ( removeIter.hasNext() ) {
final Map.Entry<String, Object> c = removeIter.next();
if ( !(c.getValue() instanceof Map) ) {
removeIter.remove();
}
}
if ( current.size() == 0 ) {
return null;
}
final Iterator<Map.Entry<String, Object>> childIter = current.entrySet().iterator();
return new Iterator<Resource>() {
public void remove() {
throw new UnsupportedOperationException("remove");
}
public Resource next() {
final Map.Entry<String, Object> props = childIter.next();
return new MapResource(getResourceResolver(), getPath() + '/' + props.getKey(), (Map)props.getValue());
}
public boolean hasNext() {
return childIter.hasNext();
}
};
}
return null;
}
}