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

io.airlift.http.server.HttpRequestEvent Maven / Gradle / Ivy

There is a newer version: 285
Show newest version
/*
 * Copyright 2010 Proofpoint, 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 io.airlift.http.server;

import com.google.common.base.Splitter;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Lists;
import io.airlift.event.client.EventField;
import io.airlift.event.client.EventType;
import io.airlift.http.server.jetty.RequestTiming;
import io.airlift.tracetoken.TraceTokenManager;
import org.eclipse.jetty.server.Request;
import org.eclipse.jetty.server.Response;

import java.security.Principal;
import java.time.Instant;
import java.util.Enumeration;

import static io.airlift.event.client.EventField.EventFieldMapping.TIMESTAMP;
import static io.airlift.http.server.TraceTokenFilter.TRACETOKEN_HEADER;

@EventType("HttpRequest")
public record HttpRequestEvent(
        @EventField(fieldMapping = TIMESTAMP) Instant timeStamp,
        @EventField String traceToken,
        @EventField String clientAddress,
        @EventField String protocol,
        @EventField String method,
        @EventField String requestUri,
        @EventField String user,
        @EventField String agent,
        @EventField String referrer,
        @EventField long requestSize,
        @EventField String requestContentType,
        @EventField long responseSize,
        @EventField int responseCode,
        @EventField String responseContentType,
        @EventField long timeToDispatch,
        @EventField long timeToHandle,
        @EventField long timeToFirstByte,
        @EventField long timeToLastByte,
        @EventField long timeToCompletion,
        @EventField long timeFromFirstToLastContent,
        @EventField DoubleSummaryStats responseContentInterarrivalStats,
        @EventField String protocolVersion)
{
    public static HttpRequestEvent createHttpRequestEvent(Request request, Response response, TraceTokenManager traceTokenManager, RequestTiming timing)
    {
        String user = null;
        Request.AuthenticationState authenticationState = Request.getAuthenticationState(request);
        if (authenticationState != null) {
            Principal principal = authenticationState.getUserPrincipal();
            if (principal != null) {
                user = principal.getName();
            }
        }

        // This is required, because async responses are processed in a different thread.
        String token = request.getHeaders().get(TRACETOKEN_HEADER);
        if (token == null && traceTokenManager != null) {
            token = traceTokenManager.getCurrentRequestToken();
        }

        ImmutableList.Builder builder = ImmutableList.builder();
        if (Request.getRemoteAddr(request) != null) {
            builder.add(Request.getRemoteAddr(request));
        }
        for (Enumeration e = request.getHeaders().getValues("X-FORWARDED-FOR"); e != null && e.hasMoreElements(); ) {
            String forwardedFor = e.nextElement();
            builder.addAll(Splitter.on(',').trimResults().omitEmptyStrings().split(forwardedFor));
        }
        String clientAddress = null;
        ImmutableList clientAddresses = builder.build();
        for (String address : Lists.reverse(clientAddresses)) {
            try {
                if (!Inet4Networks.isPrivateNetworkAddress(address)) {
                    clientAddress = address;
                    break;
                }
            }
            catch (IllegalArgumentException ignored) {
            }
        }
        if (clientAddress == null) {
            clientAddress = Request.getRemoteAddr(request);
        }

        String requestUri = null;
        if (request.getHttpURI() != null) {
            requestUri = request.getHttpURI().getPath();
            String parameters = request.getHttpURI().getQuery();
            if (parameters != null) {
                requestUri += "?" + parameters;
            }
        }

        String method = request.getMethod();
        if (method != null) {
            method = method.toUpperCase();
        }

        String protocol = request.getHeaders().get("X-FORWARDED-PROTO");
        if (protocol == null) {
            protocol = request.getHttpURI().getScheme();
        }
        if (protocol != null) {
            protocol = protocol.toLowerCase();
        }

        return new HttpRequestEvent(
                timing.requestStarted(),
                token,
                clientAddress,
                protocol,
                method,
                requestUri,
                user,
                request.getHeaders().get("User-Agent"),
                request.getHeaders().get("Referer"),
                Request.getContentBytesRead(request),
                request.getHeaders().get("Content-Type"),
                Response.getContentBytesWritten(response),
                response.getStatus(),
                response.getHeaders().get("Content-Type"),
                timing.timeToDispatch().toMillis(),
                timing.timeToHandling().toMillis(),
                timing.timeToFirstByte().toMillis(),
                timing.timeToLastByte().toMillis(),
                timing.timeToCompletion().toMillis(),
                timing.timeToLastByte().toMillis() - timing.timeToFirstByte().toMillis(),
                timing.responseContentInterarrivalStats(),
                request.getConnectionMetaData().getHttpVersion().asString());
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy