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

reactor.netty.http.server.MicrometerHttpServerMetricsHandler Maven / Gradle / Ivy

There is a newer version: 1.2.1
Show newest version
/*
 * Copyright (c) 2022-2024 VMware, Inc. or its affiliates, All Rights Reserved.
 *
 * 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
 *
 *   https://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 reactor.netty.http.server;

import io.micrometer.common.KeyValues;
import io.micrometer.core.instrument.Timer;
import io.micrometer.observation.Observation;
import io.micrometer.observation.transport.RequestReplyReceiverContext;
import io.netty.channel.Channel;
import io.netty.handler.codec.http.HttpRequest;
import io.netty.handler.codec.http.HttpResponse;
import reactor.netty.observability.ReactorNettyHandlerContext;
import reactor.util.annotation.Nullable;
import reactor.util.context.ContextView;

import java.net.InetSocketAddress;
import java.net.SocketAddress;
import java.time.Duration;
import java.util.Objects;
import java.util.function.Function;
import java.util.function.Supplier;

import static reactor.netty.Metrics.OBSERVATION_REGISTRY;
import static reactor.netty.Metrics.RESPONSE_TIME;
import static reactor.netty.Metrics.UNKNOWN;
import static reactor.netty.Metrics.updateChannelContext;
import static reactor.netty.ReactorNetty.setChannelContext;
import static reactor.netty.http.server.HttpServerObservations.ResponseTimeHighCardinalityTags.HTTP_SCHEME;
import static reactor.netty.http.server.HttpServerObservations.ResponseTimeHighCardinalityTags.HTTP_STATUS_CODE;
import static reactor.netty.http.server.HttpServerObservations.ResponseTimeHighCardinalityTags.NET_HOST_NAME;
import static reactor.netty.http.server.HttpServerObservations.ResponseTimeHighCardinalityTags.NET_HOST_PORT;
import static reactor.netty.http.server.HttpServerObservations.ResponseTimeHighCardinalityTags.REACTOR_NETTY_TYPE;
import static reactor.netty.http.server.HttpServerObservations.ResponseTimeLowCardinalityTags.METHOD;
import static reactor.netty.http.server.HttpServerObservations.ResponseTimeLowCardinalityTags.STATUS;
import static reactor.netty.http.server.HttpServerObservations.ResponseTimeLowCardinalityTags.URI;

/**
 * {@link AbstractHttpServerMetricsHandler} for Reactor Netty built-in integration with Micrometer.
 *
 * @author Marcin Grzejszczak
 * @author Violeta Georgieva
 * @since 1.1.0
 */
final class MicrometerHttpServerMetricsHandler extends AbstractHttpServerMetricsHandler {
	final MicrometerHttpServerMetricsRecorder recorder;
	final String responseTimeName;

	ResponseTimeHandlerContext responseTimeHandlerContext;
	Observation responseTimeObservation;
	ContextView parentContextView;

	MicrometerHttpServerMetricsHandler(
			MicrometerHttpServerMetricsRecorder recorder,
			@Nullable Function methodTagValue,
			@Nullable Function uriTagValue) {
		super(methodTagValue, uriTagValue);
		this.recorder = recorder;
		this.responseTimeName = recorder.name() + RESPONSE_TIME;
	}

	MicrometerHttpServerMetricsHandler(MicrometerHttpServerMetricsHandler copy) {
		super(copy);
		this.recorder = copy.recorder;
		this.responseTimeName = copy.responseTimeName;

		this.responseTimeHandlerContext = copy.responseTimeHandlerContext;
		this.responseTimeObservation = copy.responseTimeObservation;
		this.parentContextView = copy.parentContextView;
	}

	@Override
	protected MetricsArgProvider createMetricsArgProvider() {
		return super.createMetricsArgProvider().put(Observation.class, responseTimeObservation);
	}

	@Override
	protected HttpServerMetricsRecorder recorder() {
		return recorder;
	}

	@Override
	protected void recordWrite(Channel channel) {
		recordWrite(dataSent, dataSentTime, method, path, remoteSocketAddress, responseTimeObservation, status);

		setChannelContext(channel, parentContextView);
	}

	@Override
	protected void recordWrite(Channel channel, MetricsArgProvider metricsArgProvider) {
		recordWrite(metricsArgProvider.dataSent, metricsArgProvider.dataSentTime, metricsArgProvider.method, metricsArgProvider.path,
				metricsArgProvider.remoteSocketAddress, metricsArgProvider.get(Observation.class), metricsArgProvider.status);
	}

	void recordWrite(
			long dataSent,
			long dataSentTime,
			String method,
			String path,
			SocketAddress remoteSocketAddress,
			Observation responseTimeObservation,
			String status) {
		Duration dataSentTimeDuration = Duration.ofNanos(System.nanoTime() - dataSentTime);
		recorder().recordDataSentTime(path, method, status, dataSentTimeDuration);

		// Always take the remote address from the operations in order to consider proxy information
		// Use remoteSocketAddress() in order to obtain UDS info
		recorder().recordDataSent(remoteSocketAddress, path, dataSent);

		// Cannot invoke the recorder anymore:
		// 1. The recorder is one instance only, it is invoked for all requests that can happen
		// 2. The recorder does not have knowledge about request lifecycle
		//
		// Move the implementation from the recorder here
		responseTimeObservation.stop();
	}

	@Override
	protected void startRead(HttpServerOperations ops) {
		super.startRead(ops);

		responseTimeHandlerContext = new ResponseTimeHandlerContext(recorder, method, path, ops);
		responseTimeObservation = Observation.createNotStarted(this.responseTimeName, responseTimeHandlerContext, OBSERVATION_REGISTRY);
		parentContextView = updateChannelContext(ops.channel(), responseTimeObservation);
		responseTimeObservation.start();
	}

	// response
	@Override
	protected void startWrite(HttpServerOperations ops) {
		super.startWrite(ops);

		if (responseTimeObservation == null) {
			responseTimeHandlerContext = new ResponseTimeHandlerContext(recorder, method, path, ops);
			responseTimeObservation = Observation.createNotStarted(this.responseTimeName, responseTimeHandlerContext, OBSERVATION_REGISTRY);
			parentContextView = updateChannelContext(ops.channel(), responseTimeObservation);
			responseTimeObservation.start();
		}
		responseTimeHandlerContext.setResponse(ops.nettyResponse);
		responseTimeHandlerContext.status = status;
	}

	@Override
	protected void reset(Channel channel) {
		super.reset(channel);

		if (isHttp11 && LAST_FLUSH_WHEN_NO_READ) {
			setChannelContext(channel, parentContextView);
		}

		responseTimeHandlerContext = null;
		responseTimeObservation = null;
		parentContextView = null;
	}

	/*
	 * Requirements for HTTP servers
	 * 

https://github.com/open-telemetry/opentelemetry-specification/blob/main/specification/trace/api.md#span *

https://github.com/open-telemetry/opentelemetry-specification/blob/main/specification/trace/semantic_conventions/http.md#common-attributes *

https://github.com/open-telemetry/opentelemetry-specification/blob/main/specification/trace/semantic_conventions/http.md#http-server */ static final class ResponseTimeHandlerContext extends RequestReplyReceiverContext implements ReactorNettyHandlerContext, Supplier { static final String TYPE = "server"; final String method; final String netHostName; final String netHostPort; final String path; final MicrometerHttpServerMetricsRecorder recorder; final String scheme; // status might not be known beforehand String status = UNKNOWN; ResponseTimeHandlerContext(MicrometerHttpServerMetricsRecorder recorder, String method, String path, HttpServerOperations ops) { super((carrier, key) -> Objects.requireNonNull(carrier).headers().get(key)); this.recorder = recorder; HttpRequest request = ops.nettyRequest; this.method = method; InetSocketAddress hostAddress = ops.hostAddress(); this.netHostName = hostAddress != null ? hostAddress.getHostString() : ""; this.netHostPort = hostAddress != null ? hostAddress.getPort() + "" : ""; this.path = path; this.scheme = ops.scheme; setCarrier(request); setContextualName(this.method + '_' + this.path.substring(1)); } @Override public Observation.Context get() { return this; } @Override public Timer getTimer() { return recorder.getResponseTimeTimer(getName(), path, method, status); } @Override public KeyValues getHighCardinalityKeyValues() { return KeyValues.of(REACTOR_NETTY_TYPE.asString(), TYPE, HTTP_SCHEME.asString(), scheme, HTTP_STATUS_CODE.asString(), status, NET_HOST_NAME.asString(), netHostName, NET_HOST_PORT.asString(), netHostPort); } @Override public KeyValues getLowCardinalityKeyValues() { return KeyValues.of(METHOD.asString(), method, STATUS.asString(), status, URI.asString(), path); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy