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

de.codecentric.boot.admin.server.web.InstanceWebProxy Maven / Gradle / Ivy

/*
 * Copyright 2014-2019 the original author or authors.
 *
 * 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 de.codecentric.boot.admin.server.web;

import java.io.IOException;
import java.net.URI;
import java.util.concurrent.TimeoutException;

import javax.annotation.Nullable;

import com.fasterxml.jackson.annotation.JsonInclude;
import io.netty.handler.timeout.ReadTimeoutException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpMethod;
import org.springframework.http.HttpStatus;
import org.springframework.http.client.reactive.ClientHttpRequest;
import org.springframework.web.reactive.function.BodyInserter;
import org.springframework.web.reactive.function.client.ClientResponse;
import org.springframework.web.reactive.function.client.ExchangeStrategies;
import org.springframework.web.reactive.function.client.WebClient;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import reactor.util.function.Tuples;

import de.codecentric.boot.admin.server.domain.entities.Instance;
import de.codecentric.boot.admin.server.domain.values.InstanceId;
import de.codecentric.boot.admin.server.web.client.InstanceWebClient;
import de.codecentric.boot.admin.server.web.client.exception.ResolveEndpointException;

/**
 * Forwards a request to a single instances endpoint and will respond with: - 502 (Bad
 * Gateway) when any error occurs during the request - 503 (Service unavailable) when the
 * instance is not found - 504 (Gateway timeout) when the request exceeds the timeout
 *
 * @author Johannes Edmeier
 */
public class InstanceWebProxy {

	private static final Logger log = LoggerFactory.getLogger(InstanceWebProxy.class);

	private final InstanceWebClient instanceWebClient;

	private final ExchangeStrategies strategies = ExchangeStrategies.withDefaults();

	public InstanceWebProxy(InstanceWebClient instanceWebClient) {
		this.instanceWebClient = instanceWebClient;
	}

	public Mono forward(Mono instance, URI uri, HttpMethod method, HttpHeaders headers,
			BodyInserter bodyInserter) {
		return instance.flatMap((i) -> this.forward(i, uri, method, headers, bodyInserter)).switchIfEmpty(Mono
				.fromSupplier(() -> ClientResponse.create(HttpStatus.SERVICE_UNAVAILABLE, this.strategies).build()));
	}

	public Mono forward(Instance instance, URI uri, HttpMethod method, HttpHeaders headers,
			BodyInserter bodyInserter) {
		log.trace("Proxy-Request for instance {} with URL '{}'", instance.getId(), uri);
		WebClient.RequestBodySpec bodySpec = this.instanceWebClient.instance(instance).method(method).uri(uri)
				.headers((h) -> h.addAll(headers));

		WebClient.RequestHeadersSpec headersSpec = bodySpec;
		if (requiresBody(method)) {
			headersSpec = bodySpec.body(bodyInserter);
		}

		return headersSpec.exchange()
				.onErrorResume((ex) -> ex instanceof ReadTimeoutException || ex instanceof TimeoutException,
						(ex) -> Mono.fromSupplier(() -> {
							log.trace("Timeout for Proxy-Request for instance {} with URL '{}'", instance.getId(), uri);
							return ClientResponse.create(HttpStatus.GATEWAY_TIMEOUT, this.strategies).build();
						}))
				.onErrorResume(ResolveEndpointException.class, (ex) -> Mono.fromSupplier(() -> {
					log.trace("No Endpoint found for Proxy-Request for instance {} with URL '{}'", instance.getId(),
							uri);
					return ClientResponse.create(HttpStatus.NOT_FOUND, this.strategies).build();
				})).onErrorResume(IOException.class, (ex) -> Mono.fromSupplier(() -> {
					log.trace("Proxy-Request for instance {} with URL '{}' errored", instance.getId(), uri, ex);
					return ClientResponse.create(HttpStatus.BAD_GATEWAY, this.strategies).build();
				}));
	}

	public Flux forward(Flux instances, URI uri, HttpMethod method, HttpHeaders headers,
			BodyInserter bodyInserter) {
		return instances.flatMap((instance) -> this.forward(instance, uri, method, headers, bodyInserter)
				.map((clientResponse) -> Tuples.of(instance.getId(), clientResponse))).flatMap((t) -> {
					ClientResponse clientResponse = t.getT2();
					InstanceResponse.Builder response = InstanceResponse.builder().instanceId(t.getT1())
							.status(clientResponse.rawStatusCode())
							.contentType(String.join(", ", clientResponse.headers().header(HttpHeaders.CONTENT_TYPE)));
					return clientResponse.bodyToMono(String.class).map(response::body).defaultIfEmpty(response)
							.map(InstanceResponse.Builder::build);
				});
	}

	private boolean requiresBody(HttpMethod method) {
		switch (method) {
		case PUT:
		case POST:
		case PATCH:
			return true;
		default:
			return false;
		}
	}

	@lombok.Data
	@lombok.Builder(builderClassName = "Builder")
	public static class InstanceResponse {

		private final InstanceId instanceId;

		private final int status;

		@Nullable
		@JsonInclude(JsonInclude.Include.NON_EMPTY)
		private final String body;

		@Nullable
		@JsonInclude(JsonInclude.Include.NON_EMPTY)
		private final String contentType;

	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy