// =================================================================================================
// Copyright 2011 Twitter, Inc.
// -------------------------------------------------------------------------------------------------
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this work except in compliance with the License.
// You may obtain a copy of the License in the LICENSE file, or 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.twitter.common.args;
import java.lang.annotation.Annotation;
import java.util.Map;
import javax.annotation.Nullable;
import com.google.common.collect.ImmutableMap;
import com.google.common.reflect.TypeToken;
import com.twitter.common.args.apt.Configuration;
import com.twitter.common.collections.Pair;
import static com.twitter.common.args.apt.Configuration.ConfigurationException;
import static com.twitter.common.args.apt.Configuration.VerifierInfo;
/**
* Utility class to manage relationships between constraints and types.
*
* @author William Farner
*/
public final class Verifiers {
private final ImmutableMap<Pair<Class<?>, Class<? extends Annotation>>,
Verifier<?>> registry;
private Verifiers(Map<Pair<Class<?>, Class<? extends Annotation>>,
Verifier<?>> registry) {
this.registry = ImmutableMap.copyOf(registry);
}
@Nullable
<T> Verifier<T> get(TypeToken<T> type, Annotation constraint) {
for (Map.Entry<Pair<Class<?>, Class<? extends Annotation>>, Verifier<?>> entry
: registry.entrySet()) {
if (entry.getKey().getSecond() == constraint.annotationType()
&& entry.getKey().getFirst().isAssignableFrom(type.getRawType())) {
// We control the registry which ensures a proper mapping of class -> verifier.
@SuppressWarnings("unchecked")
Verifier<T> verifier = (Verifier<T>) entry.getValue();
return verifier;
}
}
return null;
}
static Verifiers fromConfiguration(Configuration configuration) {
ImmutableMap.Builder<Pair<Class<?>, Class<? extends Annotation>>,
Verifier<?>> registry = ImmutableMap.builder();
for (VerifierInfo info : configuration.verifierInfo()) {
Class<?> verifiedType = forName(info.verifiedType);
Class<? extends Annotation> verifyingAnnotation = forName(info.verifyingAnnotation);
Class<? extends Verifier<?>> verifierClass = forName(info.verifierClass);
try {
registry.put(
Pair.<Class<?>, Class<? extends Annotation>>of(verifiedType, verifyingAnnotation),
verifierClass.newInstance());
} catch (InstantiationException e) {
throw new ConfigurationException(e);
} catch (IllegalAccessException e) {
throw new ConfigurationException(e);
}
}
return new Verifiers(registry.build());
}
@SuppressWarnings("unchecked")
private static <T> Class<T> forName(String name) {
try {
return (Class<T>) Class.forName(name);
} catch (ClassNotFoundException e) {
throw new ConfigurationException(e);
}
}
}