Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
de.codecentric.boot.admin.server.web.client.LegacyEndpointConverters Maven / Gradle / Ivy
/*
* Copyright 2014-2023 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
*
* 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 de.codecentric.boot.admin.server.web.client;
import java.time.DateTimeException;
import java.time.Instant;
import java.time.OffsetDateTime;
import java.time.format.DateTimeFormatter;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Date;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.function.Function;
import java.util.stream.Collectors;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;
import org.springframework.core.ParameterizedTypeReference;
import org.springframework.core.ResolvableType;
import org.springframework.core.io.buffer.DataBuffer;
import org.springframework.core.io.buffer.DefaultDataBufferFactory;
import org.springframework.http.codec.json.Jackson2JsonDecoder;
import org.springframework.http.codec.json.Jackson2JsonEncoder;
import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
import org.springframework.lang.Nullable;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import de.codecentric.boot.admin.server.domain.values.Endpoint;
import static java.util.Arrays.asList;
import static java.util.Collections.emptyList;
import static java.util.Collections.emptySet;
import static java.util.Collections.singletonList;
import static java.util.Collections.singletonMap;
import static java.util.function.Function.identity;
import static java.util.stream.Collectors.joining;
import static java.util.stream.Collectors.toList;
import static java.util.stream.Collectors.toMap;
public final class LegacyEndpointConverters {
private static final ParameterizedTypeReference> RESPONSE_TYPE_MAP = new ParameterizedTypeReference>() {
};
private static final ParameterizedTypeReference> RESPONSE_TYPE_LIST = new ParameterizedTypeReference>() {
};
private static final ParameterizedTypeReference>> RESPONSE_TYPE_LIST_MAP = new ParameterizedTypeReference>>() {
};
private static final Jackson2JsonDecoder DECODER;
private static final Jackson2JsonEncoder ENCODER;
private static final DateTimeFormatter TIMESTAMP_PATTERN = DateTimeFormatter
.ofPattern("yyyy-MM-dd'T'HH:mm:ss.SSSZ");
static {
ObjectMapper om = Jackson2ObjectMapperBuilder.json()
.featuresToDisable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS)
.build();
DECODER = new Jackson2JsonDecoder(om);
ENCODER = new Jackson2JsonEncoder(om);
}
private LegacyEndpointConverters() {
}
public static LegacyEndpointConverter health() {
return new LegacyEndpointConverter(Endpoint.HEALTH,
convertUsing(RESPONSE_TYPE_MAP, RESPONSE_TYPE_MAP, LegacyEndpointConverters::convertHealth));
}
public static LegacyEndpointConverter env() {
return new LegacyEndpointConverter(Endpoint.ENV,
convertUsing(RESPONSE_TYPE_MAP, RESPONSE_TYPE_MAP, LegacyEndpointConverters::convertEnv));
}
public static LegacyEndpointConverter httptrace() {
return new LegacyEndpointConverter(Endpoint.HTTPTRACE,
convertUsing(RESPONSE_TYPE_LIST_MAP, RESPONSE_TYPE_MAP, LegacyEndpointConverters::convertHttptrace));
}
public static LegacyEndpointConverter threaddump() {
return new LegacyEndpointConverter(Endpoint.THREADDUMP,
convertUsing(RESPONSE_TYPE_LIST, RESPONSE_TYPE_MAP, LegacyEndpointConverters::convertThreaddump));
}
public static LegacyEndpointConverter liquibase() {
return new LegacyEndpointConverter(Endpoint.LIQUIBASE,
convertUsing(RESPONSE_TYPE_LIST_MAP, RESPONSE_TYPE_MAP, LegacyEndpointConverters::convertLiquibase));
}
public static LegacyEndpointConverter flyway() {
return new LegacyEndpointConverter(Endpoint.FLYWAY,
convertUsing(RESPONSE_TYPE_LIST_MAP, RESPONSE_TYPE_MAP, LegacyEndpointConverters::convertFlyway));
}
public static LegacyEndpointConverter info() {
return new LegacyEndpointConverter(Endpoint.INFO, (flux) -> flux);
}
public static LegacyEndpointConverter beans() {
return new LegacyEndpointConverter(Endpoint.BEANS,
convertUsing(RESPONSE_TYPE_LIST_MAP, RESPONSE_TYPE_MAP, LegacyEndpointConverters::convertBeans));
}
public static LegacyEndpointConverter configprops() {
return new LegacyEndpointConverter(Endpoint.CONFIGPROPS,
convertUsing(RESPONSE_TYPE_MAP, RESPONSE_TYPE_MAP, LegacyEndpointConverters::convertConfigprops));
}
public static LegacyEndpointConverter mappings() {
return new LegacyEndpointConverter(Endpoint.MAPPINGS,
convertUsing(RESPONSE_TYPE_MAP, RESPONSE_TYPE_MAP, LegacyEndpointConverters::convertMappings));
}
public static LegacyEndpointConverter startup() {
return new LegacyEndpointConverter(Endpoint.STARTUP, (flux) -> flux);
}
static Map convertMappingHandlerMethod(String methodDeclaration) {
// In order to keep parsing logic sane, parameterized types are dropped early
// and then we split the method declaration parts on space
// Example:
// public java.lang.Object bar.Handler.handle(java.util.List)
// -> "public","java.lang.Object","bar.Handler.handle(java.util.List)"
String[] declarationParts = methodDeclaration
// remove parameterized types using the regex below
.replaceAll("<[a-zA-Z1-9$_\\.,<> ]*>", "")
.replace(" synchronized ", " ")
// and split on single space to get the decl parts
.split(" ");
// method -> "bar.Handler.handle(java.util.List)"
String method = declarationParts[2];
// methodRef -> "bar.Handler.handle"
String methodRef = method.substring(0, method.indexOf('('));
// className -> "bar.Handler"
String className = methodRef.substring(0, methodRef.lastIndexOf('.'));
// methodName -> "handle"
String methodName = methodRef.substring(methodRef.lastIndexOf('.') + 1);
// In order to simulate descriptors' output we need to read the return type and
// the arguments of the handler method. Parameterized types were stripped early
// in the parsing process to maintain parsing simplicity.
// We also assume object types (see L) which is highly likely for spring-web
// @RequestMapping methods but there will be false positives (eg. on void).
//
// returnType -> Ljava/lang/Object;
String returnType = declarationParts[1].replaceFirst("^", "L").replace(".", "/").concat(";");
// In order to simulate descriptors' output we need to read the return type and
// the arguments of the handler method. Parameterized types were stripped early
// in the parsing process to maintain parsing simplicity.
// We also assume object types (see L) which is highly likely for spring-web
// @RequestMapping methods but there will be false positives.
//
// methodArgs -> (Ljava/util/List;)
String methodArgs = Arrays.stream(method
// get what's included in parenthesis
.substring(method.indexOf('('), method.length() - 1)
// now remove the parenthesis
.replaceAll("[()]", "")
// and split on comma
.split(",")
// then for each argument
)
.map((arg) -> arg
// prepend L char - indicated ObjectType
.replaceFirst("^", "L")
// replace dots with slashes
.replace(".", "/")
// and append ;
.concat(";")
// then join back to simulate MethodDescriptors output
)
.collect(joining("", "(", ")"));
Map handlerMethod = new LinkedHashMap<>();
handlerMethod.put("className", className);
handlerMethod.put("descriptor", methodArgs + returnType);
handlerMethod.put("name", methodName);
return handlerMethod;
}
@SuppressWarnings("unchecked")
private static Function, Flux> convertUsing(
ParameterizedTypeReference sourceType, ParameterizedTypeReference targetType,
Function converterFn) {
return (input) -> DECODER.decodeToMono(input, ResolvableType.forType(sourceType), null, null)
.map((body) -> converterFn.apply((S) body))
.flatMapMany((output) -> ENCODER.encode(Mono.just(output), new DefaultDataBufferFactory(),
ResolvableType.forType(targetType), null, null));
}
@SuppressWarnings("unchecked")
private static Map convertHealth(Map body) {
Map converted = new LinkedHashMap<>();
Map details = new LinkedHashMap<>();
body.forEach((key, value) -> {
if ("status".equals(key)) {
converted.put(key, value);
}
else if (value instanceof Map) {
details.put(key, convertHealth((Map) value));
}
else {
details.put(key, value);
}
});
if (!details.isEmpty()) {
converted.put("details", details);
}
return converted;
}
@SuppressWarnings("unchecked")
private static Map convertEnv(Map body) {
Map converted = new LinkedHashMap<>();
List> propertySources = new ArrayList<>(body.size());
body.forEach((key, value) -> {
if ("profiles".equals(key)) {
converted.put("activeProfiles", value);
}
else if (value instanceof Map) {
Map values = (Map) value;
Map properties = new LinkedHashMap<>();
values.forEach((propKey, propValue) -> properties.put(propKey, singletonMap("value", propValue)));
Map propertySource = new LinkedHashMap<>();
propertySource.put("name", key);
propertySource.put("properties", properties);
propertySources.add(propertySource);
}
});
converted.put("propertySources", propertySources);
return converted;
}
private static Map convertHttptrace(List> traces) {
return singletonMap("traces",
traces.stream().sequential().map(LegacyEndpointConverters::convertHttptrace).collect(toList()));
}
@SuppressWarnings("unchecked")
private static Map convertHttptrace(Map in) {
Map out = new LinkedHashMap<>();
out.put("timestamp", getInstant(in.get("timestamp")));
Map in_info = (Map) in.get("info");
if (in_info != null) {
Map request = new LinkedHashMap<>();
request.put("method", in_info.get("method"));
request.put("uri", in_info.get("path"));
out.put("request", request);
Map response = new LinkedHashMap<>();
Map in_headers = (Map) in_info.get("headers");
if (in_headers != null) {
Map in_request_headers = (Map) in_headers.get("request");
if (in_request_headers != null) {
Map requestHeaders = new LinkedHashMap<>();
in_request_headers.forEach((k, v) -> requestHeaders.put(k, singletonList(v)));
request.put("headers", requestHeaders);
}
Map in_response_headers = (Map) in_headers.get("response");
if (in_response_headers != null) {
if (in_response_headers.get("status") instanceof String) {
response.put("status", Long.valueOf(in_response_headers.get("status").toString()));
}
Map responseHeaders = new LinkedHashMap<>();
in_response_headers.forEach((k, v) -> responseHeaders.put(k, singletonList(v)));
responseHeaders.remove("status");
response.put("headers", responseHeaders);
}
}
out.put("response", response);
if (in_info.get("timeTaken") instanceof String) {
out.put("timeTaken", Long.valueOf(in_info.get("timeTaken").toString()));
}
}
return out;
}
private static Map convertThreaddump(List threads) {
return singletonMap("threads", threads);
}
@SuppressWarnings("unchecked")
private static Map convertLiquibase(List> reports) {
Map liquibaseBeans = reports.stream()
.sequential()
.collect(toMap((r) -> (String) r.get("name"), (r) -> singletonMap("changeSets", LegacyEndpointConverters
.convertLiquibaseChangesets((List>) r.get("changeLogs")))));
return singletonMap("contexts", singletonMap("application", singletonMap("liquibaseBeans", liquibaseBeans)));
}
private static List> convertLiquibaseChangesets(List> changeSets) {
return changeSets.stream().map((changeset) -> {
Map converted = new LinkedHashMap<>();
converted.put("id", changeset.get("ID"));
converted.put("author", changeset.get("AUTHOR"));
converted.put("changeLog", changeset.get("FILENAME"));
if (changeset.get("DATEEXECUTED") instanceof Long) {
converted.put("dateExecuted", new Date((Long) changeset.get("DATEEXECUTED")));
}
converted.put("orderExecuted", changeset.get("ORDEREXECUTED"));
converted.put("execType", changeset.get("EXECTYPE"));
converted.put("checksum", changeset.get("MD5SUM"));
converted.put("description", changeset.get("DESCRIPTION"));
converted.put("comments", changeset.get("COMMENTS"));
converted.put("tag", changeset.get("TAG"));
converted.put("contexts", (changeset.get("CONTEXTS") instanceof String)
? new LinkedHashSet<>(asList(((String) changeset.get("CONTEXTS")).split(",\\s*"))) : emptySet());
converted.put("labels", (changeset.get("LABELS") instanceof String)
? new LinkedHashSet<>(asList(((String) changeset.get("LABELS")).split(",\\s*"))) : emptySet());
converted.put("deploymentId", changeset.get("DEPLOYMENT_ID"));
return converted;
}).collect(toList());
}
@SuppressWarnings("unchecked")
private static Map convertFlyway(List> reports) {
Map flywayBeans = reports.stream()
.sequential()
.collect(toMap((r) -> (String) r.get("name"), (r) -> singletonMap("migrations", LegacyEndpointConverters
.convertFlywayMigrations((List>) r.get("migrations")))));
return singletonMap("contexts", singletonMap("application", singletonMap("flywayBeans", flywayBeans)));
}
private static List> convertFlywayMigrations(List> migrations) {
return migrations.stream().map((migration) -> {
Map converted = new LinkedHashMap<>(migration);
if (migration.get("installedOn") instanceof Long) {
converted.put("installedOn", new Date((Long) migration.get("installedOn")));
}
return converted;
}).collect(toList());
}
private static Map convertBeans(List> contextBeans) {
Map convertedContexts = contextBeans.stream().map((context) -> {
String contextName = (String) context.get("context");
String parentId = (String) context.get("parent");
// SB 1.x /beans child application context has
// itself as parent as well. In order to avoid contexts
// with same name we simple append .child in that case.
if (contextName.equals(parentId)) {
contextName = contextName + ".child";
}
List> legacyBeans = (List>) context.get("beans");
Map convertedBeans = legacyBeans.stream()
.collect(toMap((bean) -> (String) bean.get("bean"), identity()));
Map convertedContext = new LinkedHashMap<>();
convertedContext.put("contextName", contextName);
convertedContext.put("parentId", parentId);
convertedContext.put("beans", convertedBeans);
return convertedContext;
}).collect(toMap((context) -> (String) context.get("contextName"), identity()));
return singletonMap("contexts", convertedContexts);
}
private static Map convertConfigprops(Map configProps) {
Map contexts = new LinkedHashMap<>();
// SB 1.x /configprops contains a parent entry which
// contains the configprops of the parent context.
// We put this on a parentContext entry in the
// converted response.
Object parentConfigProps = configProps.get("parent");
if (parentConfigProps != null) {
configProps.remove("parent");
contexts.put("parentContext", singletonMap("beans", parentConfigProps));
}
contexts.put("application", singletonMap("beans", configProps));
return singletonMap("contexts", contexts);
}
private static Map convertMappings(Map mappings) {
List> convertedMappings = mappings.entrySet().stream().map((entry) -> {
Map convertedMapping = new LinkedHashMap<>();
Map convertedMappingDetails = new LinkedHashMap<>();
convertedMapping.put("details", convertedMappingDetails);
String predicate = entry.getKey();
convertedMapping.put("predicate", predicate);
String method = (String) ((Map) entry.getValue()).get("method");
if (method != null) {
convertedMapping.put("handler", method);
convertedMappingDetails.put("handlerMethod", convertMappingHandlerMethod(method));
}
convertedMappingDetails.put("requestMappingConditions", convertMappingConditions(predicate));
return convertedMapping;
}).collect(Collectors.toList());
return singletonMap("contexts", //
singletonMap("application", //
singletonMap("mappings", //
singletonMap("dispatcherServlets", //
singletonMap("dispatcherServlet", convertedMappings)))));
}
private static Map convertMappingConditions(String predicate) {
// Before further processing we need to remove the following occurencies
// {[, ]}, [, ] and split on comma to get condition pairs from the
// predicate string.
// Example:
// {[/scratch/{ticketId}/selectPrize/{prizeId}],methods=[POST]}
// -> "/scratch/{ticketId}/selectPrize/{prizeId}","methods=POST"
String[] conditionPairs = predicate
// remove all {[ and }] pairs
.replaceAll("\\{\\[|\\]\\}", "")
// remove all single brackets [ and ]
.replaceAll("[\\[\\]]", "")
// split on comma
.split(",");
Map conditionsMap = new LinkedHashMap<>();
conditionsMap.put("consumes", emptyList());
conditionsMap.put("headers", emptyList());
conditionsMap.put("methods", emptyList());
conditionsMap.put("params", emptyList());
conditionsMap.put("patterns", emptyList());
conditionsMap.put("produces", emptyList());
Arrays.stream(conditionPairs).forEach((condition) -> {
String[] conditionParts = condition.split("=");
// URI path patterns part of the details doesn't follow the detail=value
// semantics it's just the value so splits to a single item array
boolean isPattern = conditionParts.length == 2;
String conditionKey = isPattern ? conditionParts[0] : "patterns";
String conditionValueStr = isPattern ? conditionParts[1] : conditionParts[0];
// All detail values in SB1.x are in form of 'str1 || str2' so we split
// them on ' || '
List conditionValue = Arrays.asList(conditionValueStr.split(" \\|\\| "));
// Based on conditionKey we may need to apply some transformations,
// mostly wrapping, of the input values
switch (conditionKey) {
case "consumes":
case "produces":
conditionValue = conditionValue.stream()
.map((v) -> singletonMap("mediaType", v))
.collect(Collectors.toList());
break;
case "headers":
case "params":
case "method":
case "patterns":
default:
break;
}
conditionsMap.put(conditionKey, conditionValue);
});
return conditionsMap;
}
@Nullable
private static Instant getInstant(Object o) {
try {
if (o instanceof String) {
return OffsetDateTime.parse((String) o, TIMESTAMP_PATTERN).toInstant();
}
else if (o instanceof Long) {
return Instant.ofEpochMilli((Long) o);
}
}
catch (DateTimeException | ClassCastException ex) {
return null;
}
return null;
}
}