/*
* Copyright (C) 2011 Google Inc.
*
* 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 org.ros.namespace;
import org.ros.exception.RosRuntimeException;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
/**
* @author ethan.rublee@gmail.com (Ethan Rublee)
* @author kwc@willowgarage.com (Ken Conley)
*/
public class NameResolver {
private final GraphName namespace;
private final Map<GraphName, GraphName> remappings;
public static NameResolver newFromNamespace(GraphName namespace) {
return new NameResolver(namespace, new HashMap<GraphName, GraphName>());
}
public static NameResolver newFromNamespace(String namespace) {
return newFromNamespace(GraphName.of(namespace));
}
public static NameResolver newRoot() {
return newFromNamespace(GraphName.root());
}
public static NameResolver newRootFromRemappings(Map<GraphName, GraphName> remappings) {
return new NameResolver(GraphName.root(), remappings);
}
public static NameResolver newFromNamespaceAndRemappings(String namespace,
Map<GraphName, GraphName> remappings) {
return new NameResolver(GraphName.of(namespace), remappings);
}
public NameResolver(GraphName namespace, Map<GraphName, GraphName> remappings) {
this.remappings = Collections.unmodifiableMap(remappings);
this.namespace = namespace;
}
public GraphName getNamespace() {
return namespace;
}
/**
* Resolve name relative to namespace. If namespace is not global, it will
* first be resolved to a global name. This method will not resolve private
* ~names.
*
* This does all remappings of both the namespace and name.
*
* @param namespace
* @param name
* @return the fully resolved name relative to the given namespace
*/
public GraphName resolve(GraphName namespace, GraphName name) {
GraphName remappedNamespace = lookUpRemapping(namespace);
if (!remappedNamespace.isGlobal()) {
throw new IllegalArgumentException(String.format(
"Namespace %s (remapped from %s) must be global.", remappedNamespace, namespace));
}
GraphName remappedName = lookUpRemapping(name);
if (remappedName.isGlobal()) {
return remappedName;
}
if (remappedName.isRelative()) {
return remappedNamespace.join(remappedName);
}
if (remappedName.isPrivate()) {
throw new RosRuntimeException("Cannot resolve ~private names in arbitrary namespaces.");
}
throw new RosRuntimeException("Unable to resolve graph name: " + name);
}
/**
* @see #resolve(GraphName, GraphName)
*/
public GraphName resolve(String namespace, String name) {
return resolve(GraphName.of(namespace), GraphName.of(name));
}
/**
* @see #resolve(GraphName, GraphName)
*/
public GraphName resolve(GraphName namespace, String name) {
return resolve(namespace, GraphName.of(name));
}
/**
* @see #resolve(GraphName, GraphName)
*/
public GraphName resolve(String namespace, GraphName name) {
return resolve(GraphName.of(namespace), name);
}
/**
* @param name
* name to resolve
* @return the name resolved relative to the default namespace
*/
public GraphName resolve(GraphName name) {
return resolve(namespace, name);
}
/**
* @see #resolve(GraphName)
*/
public GraphName resolve(String name) {
return resolve(GraphName.of(name));
}
/**
* @return remappings
*/
public Map<GraphName, GraphName> getRemappings() {
return remappings;
}
/**
* Construct a new child {@link NameResolver} with the same remappings as this
* {@link NameResolver}. The namespace of the new child {@link NameResolver}
* will be the resolved in this namespace.
*
* @param namespace
* the namespace of the child {@link NameResolver} relative to this
* {@link NameResolver}'s namespace
* @return a new child {@link NameResolver} whose namespace is relative to the
* parent {@link NameResolver}'s namespace
*/
public NameResolver newChild(GraphName namespace) {
return new NameResolver(resolve(namespace), remappings);
}
/**
* @see #newChild(GraphName)
*/
public NameResolver newChild(String namespace) {
return newChild(GraphName.of(namespace));
}
protected GraphName lookUpRemapping(GraphName name) {
GraphName remappedName = name;
if (remappings.containsKey(name)) {
remappedName = remappings.get(name);
}
return remappedName;
}
}