blob: d13b5417f4cd36055ff08add735cea0eae6c2c81 [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.juneau.annotation;
import static org.apache.juneau.internal.ExceptionUtils.*;
import java.lang.annotation.*;
import java.lang.reflect.*;
import org.apache.juneau.collections.*;
import org.apache.juneau.internal.*;
/**
* A concrete implementation of an annotation.
*
* <p>
* Follows the standard Java conventions for equality and hashcode calculation for annotations.
* Equivalent annotations defined programmatically and declaratively should match for equality and hashcode calculation.
*
* <p>
* For performance reasons, the hashcode is calculated one time and cached at the end of object creation.
* Constructors must call the {@link #postConstruct()} method after all fields have been set to trigger this calculation.
*/
public class AnnotationImpl implements Annotation {
private final Class<? extends Annotation> annotationType;
private int hashCode = -1;
/**
* Constructor.
*
* @param b The builder used to instantiate the fields of this class.
*/
public AnnotationImpl(AnnotationBuilder b) {
this.annotationType = b.annotationType;
}
/**
* This method must be called at the end of initialization to calculate the hashCode one time.
*/
protected void postConstruct() {
this.hashCode = AnnotationUtils.hashCode(this);
}
/**
* Implements the {@link Annotation#annotationType()} method for child classes.
*
* @return This class.
*/
@Override /* Annotation */
public Class<? extends Annotation> annotationType() {
return annotationType;
}
@Override /* Object */
public int hashCode() {
if (hashCode == -1)
throw runtimeException("Programming error. postConstruct() was never called on annotation.");
return hashCode;
}
@Override /* Object */
public boolean equals(Object o) {
if (! annotationType.isInstance(o))
return false;
return AnnotationUtils.equals(this, (Annotation)o);
}
/**
* Returns the properties defined on this bean as a simple map for debugging purposes.
*
* <p>
* Use <c>SimpleJson.<jsf>DEFAULT</jsf>.println(<jv>thisBean</jv>)</c> to dump the contents of this bean to the console.
*
* @return A new map containing this bean's properties.
*/
public OMap toMap() {
OMap om = OMap.create();
try {
for (Method m : AnnotationUtils.getSortedAnnotationMethods(annotationType()))
om.a(m.getName(), m.invoke(this));
} catch (Exception e) {
e.printStackTrace();
}
return om;
}
@Override /* Object */
public String toString() {
return toMap().toString();
}
}