All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.helidon.tracing.jersey.TracingHelper Maven / Gradle / Ivy

/*
 * Copyright (c) 2018, 2021 Oracle and/or its affiliates.
 *
 * 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 io.helidon.tracing.jersey;

import java.lang.reflect.Method;
import java.util.Optional;
import java.util.function.Function;

import io.helidon.jersey.common.InvokedResource;

import jakarta.ws.rs.Path;
import jakarta.ws.rs.container.ContainerRequestContext;

/**
 * Utilities for tracing in helidon.
 */
public final class TracingHelper {
    private final Function nameFunction;

    private TracingHelper(Function nameFunction) {
        this.nameFunction = nameFunction;
    }

    /**
     * Create helper with default span name function.
     *
     * @return tracing helper
     * @see #classMethodName(ContainerRequestContext)
     */
    public static TracingHelper create() {
        return new TracingHelper(TracingHelper::classMethodName);
    }

    /**
     * Create helper with custom span name function.
     * @param nameFunction function to get span name from context
     * @return tracing helper
     * @see #classMethodName(ContainerRequestContext)
     * @see #httpPathMethodName(ContainerRequestContext)
     */
    public static TracingHelper create(Function nameFunction) {
        return new TracingHelper(nameFunction);
    }

    /**
     * Name is generated from path as {http-method}:{request-path}.
     *
     * @param requestContext context to extract information from
     * @return name of span to use
     */
    public static String httpPathMethodName(ContainerRequestContext requestContext) {
        InvokedResource invokedResource = InvokedResource.create(requestContext);
        Optional method = invokedResource.definitionMethod();

        if (!method.isPresent()) {
            return requestContext.getMethod().toUpperCase() + ":" + requestContext.getUriInfo().getPath();
        }

        StringBuilder fullPath = new StringBuilder();
        fullPath.append(requestContext.getMethod().toUpperCase());
        fullPath.append(":");

        Optional classPathAnnotation = invokedResource.findClassAnnotation(Path.class);
        classPathAnnotation.map(Path::value)
                .ifPresent(resourcePath -> {
                    if (!resourcePath.startsWith("/")) {
                        fullPath.append("/");
                    }
                    fullPath.append(resourcePath);
                });

        Optional methodPathAnnotation = invokedResource.findMethodAnnotation(Path.class);
        methodPathAnnotation.map(Path::value)
                .ifPresent(methodPath -> {
                    if ((fullPath.length() != 0) && (fullPath.charAt(fullPath.length() - 1) != '/')
                            && !methodPath.startsWith("/")) {
                        fullPath.append("/");
                    }
                    fullPath.append(methodPath);
                });

        return fullPath.toString();
    }

    /**
     * Name is generated from class and method as {http-method}:{fully-qualified-class-name}.{method-name}.
     *
     * @param requestContext context to extract information from
     * @return name of span to use
     */
    public static String classMethodName(ContainerRequestContext requestContext) {
        return InvokedResource.create(requestContext)
                .definitionMethod()
                .map(m -> requestContext.getMethod().toUpperCase()
                        + ":" + m.getDeclaringClass().getName()
                        + "." + m.getName())
                .orElseGet(() -> requestContext.getMethod().toUpperCase() + ":404");
    }

    /**
     * Generate span using the function provided by {@link #create(Function)}.
     *
     * @param context request context of the container
     * @return name of the span to use
     */
    public String generateSpanName(ContainerRequestContext context) {
        return nameFunction.apply(context);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy