org.apache.xbean.naming.context.ContextUtil Maven / Gradle / Ivy
/**
* 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.xbean.naming.context;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.Properties;
import javax.naming.Binding;
import javax.naming.CompoundName;
import javax.naming.Context;
import javax.naming.Name;
import javax.naming.NameClassPair;
import javax.naming.NameParser;
import javax.naming.NamingEnumeration;
import javax.naming.NamingException;
import javax.naming.Reference;
import javax.naming.spi.NamingManager;
import org.apache.xbean.naming.reference.SimpleReference;
/**
* @version $Rev$ $Date$
*/
public final class ContextUtil {
private ContextUtil() {
}
public final static NameParser NAME_PARSER = new SimpleNameParser();
public static Name parseName(String name) throws NamingException {
return NAME_PARSER.parse(name);
}
public static Object resolve(Object value, String stringName, Name parsedName, Context nameCtx) throws NamingException {
if (!(value instanceof Reference)) {
return value;
}
Reference reference = (Reference) value;
// for SimpleReference we can just call the getContext method
if (reference instanceof SimpleReference) {
try {
return ((SimpleReference) reference).getContent();
} catch (NamingException e) {
throw e;
} catch (Exception e) {
throw (NamingException) new NamingException("Could not look up : " + stringName == null? parsedName.toString(): stringName).initCause(e);
}
}
// for normal References we have to do it the slow way
try {
if (parsedName == null) {
parsedName = NAME_PARSER.parse(stringName);
}
return NamingManager.getObjectInstance(reference, parsedName, nameCtx, nameCtx.getEnvironment());
} catch (NamingException e) {
throw e;
} catch (Exception e) {
throw (NamingException) new NamingException("Could not look up : " + stringName == null? parsedName.toString(): stringName).initCause(e);
}
}
public static Map listToMap(NamingEnumeration enumeration) {
Map result = new HashMap();
while (enumeration.hasMoreElements()) {
NameClassPair nameClassPair = (NameClassPair) enumeration.nextElement();
String name = nameClassPair.getName();
result.put(name, nameClassPair.getClassName());
}
return result;
}
public static Map listBindingsToMap(NamingEnumeration enumeration) {
Map result = new HashMap();
while (enumeration.hasMoreElements()) {
Binding binding = (Binding) enumeration.nextElement();
String name = binding.getName();
result.put(name, binding.getObject());
}
return result;
}
public static final class ListEnumeration implements NamingEnumeration {
private final Iterator iterator;
public ListEnumeration(Map localBindings) {
this.iterator = localBindings.entrySet().iterator();
}
public boolean hasMore() {
return iterator.hasNext();
}
public boolean hasMoreElements() {
return iterator.hasNext();
}
public NameClassPair next() {
return nextElement();
}
public NameClassPair nextElement() {
Map.Entry entry = (Map.Entry) iterator.next();
String name = (String) entry.getKey();
Object value = entry.getValue();
String className;
if (value instanceof Reference) {
Reference reference = (Reference) value;
className = reference.getClassName();
} else {
className = value.getClass().getName();
}
return new NameClassPair(name, className);
}
public void close() {
}
}
public static final class ListBindingEnumeration implements NamingEnumeration {
private final Iterator iterator;
private final Context context;
public ListBindingEnumeration(Map localBindings, Context context) {
this.iterator = localBindings.entrySet().iterator();
this.context = context;
}
public boolean hasMore() {
return iterator.hasNext();
}
public boolean hasMoreElements() {
return iterator.hasNext();
}
public Binding next() {
return nextElement();
}
public Binding nextElement() {
Map.Entry entry = (Map.Entry) iterator.next();
String name = (String) entry.getKey();
Object value = entry.getValue();
return new ReadOnlyBinding(name, value, context);
}
public void close() {
}
}
public static final class ReadOnlyBinding extends Binding {
private final Object value;
private final Context context;
private final boolean isRelative;
public ReadOnlyBinding(String name, Object value, Context context) {
this(name, value, false, context);
}
public ReadOnlyBinding(String name, Object value, boolean isRelative, Context context) {
super(name, value);
this.value = value;
this.context = context;
this.isRelative = isRelative;
}
public void setName(String name) {
throw new UnsupportedOperationException("Context is read only");
}
public String getClassName() {
if (value instanceof Reference) {
Reference reference = (Reference) value;
return reference.getClassName();
}
return value.getClass().getName();
}
public void setClassName(String name) {
throw new UnsupportedOperationException("Context is read only");
}
public Object getObject() {
try {
return resolve(value, getName(), null, context);
} catch (NamingException e) {
throw new RuntimeException(e);
}
}
public void setObject(Object obj) {
throw new UnsupportedOperationException("Context is read only");
}
public boolean isRelative() {
return isRelative;
}
public void setRelative(boolean r) {
throw new UnsupportedOperationException("Context is read only");
}
}
private static final class SimpleNameParser implements NameParser {
private static final Properties PARSER_PROPERTIES = new Properties();
static {
PARSER_PROPERTIES.put("jndi.syntax.direction", "left_to_right");
PARSER_PROPERTIES.put("jndi.syntax.separator", "/");
}
private SimpleNameParser() {
}
public Name parse(String name) throws NamingException {
return new CompoundName(name, PARSER_PROPERTIES);
}
}
public static Map createBindings(Map absoluteBindings, NestedContextFactory factory) throws NamingException {
// create a tree of Nodes using the absolute bindings
Node node = buildMapTree(absoluteBindings);
// convert the node tree into a tree of context objects
return ContextUtil.createBindings(null, node, factory);
}
private static Map createBindings(String nameInNameSpace, Node node, NestedContextFactory factory) throws NamingException {
Map bindings = new HashMap(node.size());
for (Map.Entry entry : node.entrySet()) {
String name = entry.getKey();
Object value = entry.getValue();
// if this is a nested node we need to create a context for the node
if (value instanceof Node) {
Node nestedNode = (Node) value;
// recursive call create bindings to cause building the context depth first
String path = nameInNameSpace == null ? name : nameInNameSpace + "/" + name;
Map nestedBindings = createBindings(path, nestedNode, factory);
Context nestedContext = factory.createNestedSubcontext(path, nestedBindings);
bindings.put(name, nestedContext);
} else {
bindings.put(name, value);
}
}
return bindings;
}
/**
* Do nothing subclass of hashmap used to differentiate between a Map in the tree an a nested element during tree building
*/
public static final class Node extends HashMap {
}
public static Node buildMapTree(Map absoluteBindings) throws NamingException {
Node rootContext = new Node();
for (Map.Entry entry : absoluteBindings.entrySet()) {
String name = entry.getKey();
Object value = entry.getValue();
Node parentContext = rootContext;
Name compoundName = ContextUtil.parseName(name);
for (Enumeration parts = compoundName.getAll(); parts.hasMoreElements();) {
String part = (String) parts.nextElement();
// the last element in the path is the name of the value
if (parts.hasMoreElements()) {
// nest node into parent
Node bindings = (Node) parentContext.get(part);
if (bindings == null) {
bindings = new Node();
parentContext.put(part, bindings);
}
parentContext = bindings;
}
}
parentContext.put(compoundName.get(compoundName.size() - 1), value);
}
return rootContext;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy