fish.payara.microprofile.metrics.cdi.MetricsResolver Maven / Gradle / Ivy
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright (c) [2018] Payara Foundation and/or its affiliates. All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common Development
* and Distribution License("CDDL") (collectively, the "License"). You
* may not use this file except in compliance with the License. You can
* obtain a copy of the License at
* https://github.com/payara/Payara/blob/master/LICENSE.txt
* See the License for the specific
* language governing permissions and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file at glassfish/legal/LICENSE.txt.
*
* GPL Classpath Exception:
* The Payara Foundation designates this particular file as subject to the "Classpath"
* exception as provided by the Payara Foundation in the GPL Version 2 section of the License
* file that accompanied this code.
*
* Modifications:
* If applicable, add the following below the License Header, with the fields
* enclosed by brackets [] replaced by your own identifying information:
* "Portions Copyright [year] [name of copyright owner]"
*
* Contributor(s):
* If you wish your version of this file to be governed by only the CDDL or
* only the GPL Version 2, indicate your decision by adding "[Contributor]
* elects to include this software in this distribution under the [CDDL or GPL
* Version 2] license." If you don't indicate a single choice of license, a
* recipient has the option to distribute your version of this file under
* either the CDDL, the GPL Version 2 or to extend the choice of license to
* its licensees as provided above. However, if you add GPL Version 2 code
* and therefore, elected the GPL Version 2 license, then the option applies
* only if the new code is made subject to such option by the copyright
* holder.
*
* *****************************************************************************
* Copyright (c) 2013 Antonin Stefanutti ([email protected])
*
* 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 fish.payara.microprofile.metrics.cdi;
import java.lang.annotation.Annotation;
import java.lang.reflect.AnnotatedElement;
import java.lang.reflect.Constructor;
import java.lang.reflect.Member;
import java.lang.reflect.Method;
import javax.enterprise.context.ApplicationScoped;
import javax.enterprise.inject.Vetoed;
import org.eclipse.microprofile.metrics.Metadata;
import org.eclipse.microprofile.metrics.MetricRegistry;
import static org.eclipse.microprofile.metrics.MetricType.COUNTER;
import static org.eclipse.microprofile.metrics.MetricType.GAUGE;
import static org.eclipse.microprofile.metrics.MetricType.METERED;
import static org.eclipse.microprofile.metrics.MetricType.TIMER;
import org.eclipse.microprofile.metrics.annotation.Counted;
import org.eclipse.microprofile.metrics.annotation.Gauge;
import org.eclipse.microprofile.metrics.annotation.Metered;
import org.eclipse.microprofile.metrics.annotation.Timed;
@ApplicationScoped
public class MetricsResolver {
public Of counted(Class> bean, E element) {
return resolverOf(bean, element, Counted.class);
}
public Of gauge(Class> bean, Method method) {
return resolverOf(bean, method, Gauge.class);
}
public Of metered(Class> bean, E element) {
return resolverOf(bean, element, Metered.class);
}
public Of timed(Class> bean, E element) {
return resolverOf(bean, element, Timed.class);
}
private Of resolverOf(Class> bean, E element, Class metric) {
if (element.isAnnotationPresent(metric)) {
return elementResolverOf(element, metric);
} else {
return beanResolverOf(bean, element, metric);
}
}
private Of elementResolverOf(E element, Class metric) {
T annotation = element.getAnnotation(metric);
String name = metricName(element, annotation);
return new ValidMetric<>(annotation, name, getMetadata(name, annotation));
}
private Of beanResolverOf(Class> bean, E element, Class metric) {
if (bean.isAnnotationPresent(metric)) {
T annotation = bean.getAnnotation(metric);
String name = metricName(bean, element, annotation);
return new ValidMetric<>(annotation, name, getMetadata(name, annotation));
} else if (bean.getSuperclass() != null) {
return beanResolverOf(bean.getSuperclass(), element, metric);
}
return new InvalidMetric<>();
}
private String metricName(E element, Annotation annotation) {
String name = metricName(annotation);
String metric = name.isEmpty() ? defaultName(element) : name;
return isMetricAbsolute(annotation) ? metric
: MetricRegistry.name(element.getDeclaringClass(), metric);
}
private String metricName(Class> bean, E element, Annotation annotation) {
String name = metricName(annotation);
String metric = name.isEmpty() ? bean.getSimpleName() : name;
return isMetricAbsolute(annotation) ? MetricRegistry.name(metric, defaultName(element))
: MetricRegistry.name(bean.getPackage().getName(), metric, defaultName(element));
}
private String defaultName(E element) {
if (element instanceof Constructor) {
return element.getDeclaringClass().getSimpleName();
} else {
return element.getName();
}
}
private String metricName(Annotation annotation) {
if (Counted.class.isInstance(annotation)) {
return ((Counted) annotation).name();
} else if (Gauge.class.isInstance(annotation)) {
return ((Gauge) annotation).name();
} else if (Metered.class.isInstance(annotation)) {
return ((Metered) annotation).name();
} else if (Timed.class.isInstance(annotation)) {
return ((Timed) annotation).name();
} else {
throw new IllegalArgumentException("Unsupported Metrics [" + annotation.getClass().getName() + "]");
}
}
private boolean isMetricAbsolute(Annotation annotation) {
if (Counted.class.isInstance(annotation)) {
return ((Counted) annotation).absolute();
} else if (Gauge.class.isInstance(annotation)) {
return ((Gauge) annotation).absolute();
} else if (Metered.class.isInstance(annotation)) {
return ((Metered) annotation).absolute();
} else if (Timed.class.isInstance(annotation)) {
return ((Timed) annotation).absolute();
} else {
throw new IllegalArgumentException("Unsupported Metrics [" + annotation.getClass().getName() + "]");
}
}
public Metadata getMetadata(String name, T annotation) {
Metadata metadata;
String[] tags;
if (Counted.class.isInstance(annotation)) {
Counted counted = (Counted) annotation;
metadata = new Metadata(name, counted.displayName(), counted.description(), COUNTER, counted.unit());
metadata.setReusable(counted.reusable());
tags = counted.tags();
} else if (Gauge.class.isInstance(annotation)) {
Gauge gauge = (Gauge) annotation;
metadata = new Metadata(name, gauge.displayName(), gauge.description(), GAUGE, gauge.unit());
tags = gauge.tags();
} else if (Metered.class.isInstance(annotation)) {
Metered metered = (Metered) annotation;
metadata = new Metadata(name, metered.displayName(), metered.description(), METERED, metered.unit());
metadata.setReusable(metered.reusable());
tags = metered.tags();
} else if (Timed.class.isInstance(annotation)) {
Timed timed = (Timed) annotation;
metadata = new Metadata(name, timed.displayName(), timed.description(), TIMER, timed.unit());
metadata.setReusable(timed.reusable());
tags = timed.tags();
} else {
throw new IllegalArgumentException("Unsupported Metrics [" + annotation.getClass().getName() + "]");
}
for (String tag : tags) {
metadata.addTag(tag);
}
return metadata;
}
@Vetoed
private static final class ValidMetric implements Of {
private final T annotation;
private final String name;
private final Metadata metadata;
private ValidMetric(T annotation, String name, Metadata metadata) {
this.annotation = annotation;
this.name = name;
this.metadata = metadata;
}
@Override
public boolean isPresent() {
return true;
}
@Override
public String metricName() {
return name;
}
@Override
public T metricAnnotation() {
return annotation;
}
@Override
public Metadata metadata() {
return metadata;
}
}
@Vetoed
private static final class InvalidMetric implements Of {
private InvalidMetric() {
}
@Override
public boolean isPresent() {
return false;
}
@Override
public String metricName() {
throw new UnsupportedOperationException();
}
@Override
public T metricAnnotation() {
throw new UnsupportedOperationException();
}
@Override
public Metadata metadata() {
return null;
}
}
public interface Of {
boolean isPresent();
String metricName();
T metricAnnotation();
Metadata metadata();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy