org.springdoc.api.AbstractOpenApiResource Maven / Gradle / Ivy
/*
*
* *
* * * Copyright 2019-2020 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 org.springdoc.api;
import java.io.UnsupportedEncodingException;
import java.lang.reflect.Method;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URLDecoder;
import java.nio.charset.StandardCharsets;
import java.time.Duration;
import java.time.Instant;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
import java.util.Set;
import java.util.concurrent.Executors;
import java.util.stream.Collectors;
import com.fasterxml.jackson.annotation.JsonView;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.MapperFeature;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;
import com.fasterxml.jackson.dataformat.yaml.YAMLFactory;
import com.fasterxml.jackson.dataformat.yaml.YAMLGenerator.Feature;
import io.swagger.v3.core.filter.SpecFilter;
import io.swagger.v3.core.util.Json;
import io.swagger.v3.core.util.ReflectionUtils;
import io.swagger.v3.core.util.Yaml;
import io.swagger.v3.oas.annotations.Hidden;
import io.swagger.v3.oas.annotations.callbacks.Callback;
import io.swagger.v3.oas.annotations.enums.ParameterIn;
import io.swagger.v3.oas.models.Components;
import io.swagger.v3.oas.models.OpenAPI;
import io.swagger.v3.oas.models.Operation;
import io.swagger.v3.oas.models.PathItem;
import io.swagger.v3.oas.models.PathItem.HttpMethod;
import io.swagger.v3.oas.models.Paths;
import io.swagger.v3.oas.models.media.Schema;
import io.swagger.v3.oas.models.media.StringSchema;
import io.swagger.v3.oas.models.parameters.Parameter;
import io.swagger.v3.oas.models.responses.ApiResponses;
import org.apache.commons.lang3.ArrayUtils;
import org.apache.commons.lang3.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springdoc.api.mixins.SortedOpenAPIMixin;
import org.springdoc.api.mixins.SortedSchemaMixin;
import org.springdoc.core.AbstractRequestService;
import org.springdoc.core.ActuatorProvider;
import org.springdoc.core.GenericParameterService;
import org.springdoc.core.GenericResponseService;
import org.springdoc.core.MethodAttributes;
import org.springdoc.core.OpenAPIService;
import org.springdoc.core.OperationService;
import org.springdoc.core.SpringDocConfigProperties;
import org.springdoc.core.SpringDocConfigProperties.GroupConfig;
import org.springdoc.core.annotations.RouterOperations;
import org.springdoc.core.customizers.OpenApiCustomiser;
import org.springdoc.core.customizers.OperationCustomizer;
import org.springdoc.core.fn.AbstractRouterFunctionVisitor;
import org.springdoc.core.fn.RouterFunctionData;
import org.springdoc.core.fn.RouterOperation;
import org.springframework.aop.support.AopUtils;
import org.springframework.beans.factory.ObjectFactory;
import org.springframework.context.ApplicationContext;
import org.springframework.core.annotation.AnnotatedElementUtils;
import org.springframework.core.annotation.AnnotationUtils;
import org.springframework.core.env.Environment;
import org.springframework.util.AntPathMatcher;
import org.springframework.util.CollectionUtils;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.method.HandlerMethod;
import static org.springdoc.core.Constants.ACTUATOR_DEFAULT_GROUP;
import static org.springdoc.core.Constants.LINKS_SCHEMA_CUSTOMISER;
import static org.springdoc.core.Constants.OPERATION_ATTRIBUTE;
import static org.springdoc.core.Constants.SPRING_MVC_SERVLET_PATH;
import static org.springdoc.core.converters.SchemaPropertyDeprecatingConverter.isDeprecated;
/**
* The type Abstract open api resource.
* @author bnasslahsen
* @author kevinraddatz
*/
public abstract class AbstractOpenApiResource extends SpecFilter {
/**
* The constant LOGGER.
*/
private static final Logger LOGGER = LoggerFactory.getLogger(AbstractOpenApiResource.class);
/**
* The constant ADDITIONAL_REST_CONTROLLERS.
*/
private static final List> ADDITIONAL_REST_CONTROLLERS = new ArrayList<>();
/**
* The constant HIDDEN_REST_CONTROLLERS.
*/
private static final List> HIDDEN_REST_CONTROLLERS = new ArrayList<>();
/**
* The Open api builder.
*/
protected OpenAPIService openAPIService;
/**
* The open api builder object factory.
*/
private final ObjectFactory openAPIBuilderObjectFactory;
/**
* The Spring doc config properties.
*/
protected final SpringDocConfigProperties springDocConfigProperties;
/**
* The Actuator provider.
*/
protected final Optional optionalActuatorProvider;
/**
* The Request builder.
*/
private final AbstractRequestService requestBuilder;
/**
* The Response builder.
*/
private final GenericResponseService responseBuilder;
/**
* The Operation parser.
*/
private final OperationService operationParser;
/**
* The Open api customisers.
*/
private final Optional> openApiCustomisers;
/**
* The Operation customizers.
*/
private final Optional> operationCustomizers;
/**
* The Ant path matcher.
*/
private final AntPathMatcher antPathMatcher = new AntPathMatcher();
/**
* The Group name.
*/
protected final String groupName;
/**
* Instantiates a new Abstract open api resource.
*
* @param groupName the group name
* @param openAPIBuilderObjectFactory the open api builder object factory
* @param requestBuilder the request builder
* @param responseBuilder the response builder
* @param operationParser the operation parser
* @param operationCustomizers the operation customizers
* @param openApiCustomisers the open api customisers
* @param springDocConfigProperties the spring doc config properties
* @param actuatorProvider the actuator provider
*/
protected AbstractOpenApiResource(String groupName, ObjectFactory openAPIBuilderObjectFactory,
AbstractRequestService requestBuilder,
GenericResponseService responseBuilder, OperationService operationParser,
Optional> operationCustomizers,
Optional> openApiCustomisers,
SpringDocConfigProperties springDocConfigProperties,
Optional actuatorProvider) {
super();
this.groupName = Objects.requireNonNull(groupName, "groupName");
this.openAPIBuilderObjectFactory = openAPIBuilderObjectFactory;
this.openAPIService = openAPIBuilderObjectFactory.getObject();
this.requestBuilder = requestBuilder;
this.responseBuilder = responseBuilder;
this.operationParser = operationParser;
this.openApiCustomisers = openApiCustomisers;
//add the default customizers
Map existingOpenApiCustomisers = openAPIService.getContext().getBeansOfType(OpenApiCustomiser.class);
if (!CollectionUtils.isEmpty(existingOpenApiCustomisers) && existingOpenApiCustomisers.containsKey(LINKS_SCHEMA_CUSTOMISER))
openApiCustomisers.ifPresent(openApiCustomisersList -> openApiCustomisersList.add(existingOpenApiCustomisers.get(LINKS_SCHEMA_CUSTOMISER)));
this.springDocConfigProperties = springDocConfigProperties;
operationCustomizers.ifPresent(customizers -> customizers.removeIf(Objects::isNull));
this.operationCustomizers = operationCustomizers;
this.optionalActuatorProvider = actuatorProvider;
if (springDocConfigProperties.isPreLoadingEnabled())
Executors.newSingleThreadExecutor().execute(this::getOpenApi);
}
/**
* Add rest controllers.
*
* @param classes the classes
*/
public static void addRestControllers(Class>... classes) {
ADDITIONAL_REST_CONTROLLERS.addAll(Arrays.asList(classes));
}
/**
* Add hidden rest controllers.
*
* @param classes the classes
*/
public static void addHiddenRestControllers(Class>... classes) {
HIDDEN_REST_CONTROLLERS.addAll(Arrays.asList(classes));
}
/**
* Add hidden rest controllers.
*
* @param classes the classes
*/
public static void addHiddenRestControllers(String... classes) {
Set> hiddenClasses = new HashSet<>();
for (String aClass : classes) {
try {
hiddenClasses.add(Class.forName(aClass));
}
catch (ClassNotFoundException e) {
LOGGER.warn("The following class doesn't exist and cannot be hidden: {}", aClass);
}
}
HIDDEN_REST_CONTROLLERS.addAll(hiddenClasses);
}
/**
* Gets open api.
*
* @return the open api
*/
protected synchronized OpenAPI getOpenApi() {
OpenAPI openApi;
if (openAPIService.getCachedOpenAPI() == null || springDocConfigProperties.isCacheDisabled()) {
Instant start = Instant.now();
openAPIService.build();
Map mappingsMap = openAPIService.getMappingsMap().entrySet().stream()
.filter(controller -> (AnnotationUtils.findAnnotation(controller.getValue().getClass(),
Hidden.class) == null))
.filter(controller -> !AbstractOpenApiResource.isHiddenRestControllers(controller.getValue().getClass()))
.collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue, (a1, a2) -> a1));
Map findControllerAdvice = openAPIService.getControllerAdviceMap();
// calculate generic responses
openApi = openAPIService.getCalculatedOpenAPI();
if (springDocConfigProperties.isOverrideWithGenericResponse() && !CollectionUtils.isEmpty(findControllerAdvice)) {
if (!CollectionUtils.isEmpty(mappingsMap))
findControllerAdvice.putAll(mappingsMap);
responseBuilder.buildGenericResponse(openApi.getComponents(), findControllerAdvice);
}
getPaths(mappingsMap);
if (!CollectionUtils.isEmpty(openApi.getServers()))
openAPIService.setServersPresent(true);
openAPIService.updateServers(openApi);
if (springDocConfigProperties.isRemoveBrokenReferenceDefinitions())
this.removeBrokenReferenceDefinitions(openApi);
// run the optional customisers
openApiCustomisers.ifPresent(apiCustomisers -> apiCustomisers.forEach(openApiCustomiser -> openApiCustomiser.customise(openApi)));
openAPIService.setCachedOpenAPI(openApi);
openAPIService.resetCalculatedOpenAPI();
LOGGER.info("Init duration for springdoc-openapi is: {} ms",
Duration.between(start, Instant.now()).toMillis());
}
else {
if (!CollectionUtils.isEmpty(openAPIService.getCachedOpenAPI().getServers()))
openAPIService.setServersPresent(true);
openApi = openAPIService.updateServers(openAPIService.getCachedOpenAPI());
}
return openApi;
}
/**
* Gets paths.
*
* @param findRestControllers the find rest controllers
*/
protected abstract void getPaths(Map findRestControllers);
/**
* Calculate path.
*
* @param handlerMethod the handler method
* @param routerOperation the router operation
*/
protected void calculatePath(HandlerMethod handlerMethod, RouterOperation routerOperation) {
String operationPath = routerOperation.getPath();
Set requestMethods = new HashSet<>(Arrays.asList(routerOperation.getMethods()));
io.swagger.v3.oas.annotations.Operation apiOperation = routerOperation.getOperation();
String[] methodConsumes = routerOperation.getConsumes();
String[] methodProduces = routerOperation.getProduces();
String[] headers = routerOperation.getHeaders();
Map queryParams = routerOperation.getQueryParams();
OpenAPI openAPI = openAPIService.getCalculatedOpenAPI();
Components components = openAPI.getComponents();
Paths paths = openAPI.getPaths();
Map operationMap = null;
if (paths.containsKey(operationPath)) {
PathItem pathItem = paths.get(operationPath);
operationMap = pathItem.readOperationsMap();
}
for (RequestMethod requestMethod : requestMethods) {
Operation existingOperation = getExistingOperation(operationMap, requestMethod);
Method method = handlerMethod.getMethod();
// skip hidden operations
if (operationParser.isHidden(method))
continue;
RequestMapping reqMappingClass = AnnotatedElementUtils.findMergedAnnotation(handlerMethod.getBeanType(),
RequestMapping.class);
MethodAttributes methodAttributes = new MethodAttributes(springDocConfigProperties.getDefaultConsumesMediaType(), springDocConfigProperties.getDefaultProducesMediaType(), methodConsumes, methodProduces, headers);
methodAttributes.setMethodOverloaded(existingOperation != null);
if (reqMappingClass != null) {
methodAttributes.setClassConsumes(reqMappingClass.consumes());
methodAttributes.setClassProduces(reqMappingClass.produces());
}
methodAttributes.calculateConsumesProduces(method);
Operation operation = (existingOperation != null) ? existingOperation : new Operation();
if (isDeprecated(method))
operation.setDeprecated(true);
// Add documentation from operation annotation
if (apiOperation == null || StringUtils.isBlank(apiOperation.operationId()))
apiOperation = AnnotatedElementUtils.findMergedAnnotation(method,
io.swagger.v3.oas.annotations.Operation.class);
calculateJsonView(apiOperation, methodAttributes, method);
if (apiOperation != null)
openAPI = operationParser.parse(apiOperation, operation, openAPI, methodAttributes);
fillParametersList(operation, queryParams, methodAttributes);
// compute tags
operation = openAPIService.buildTags(handlerMethod, operation, openAPI);
io.swagger.v3.oas.annotations.parameters.RequestBody requestBodyDoc = AnnotatedElementUtils.findMergedAnnotation(method,
io.swagger.v3.oas.annotations.parameters.RequestBody.class);
// RequestBody in Operation
requestBuilder.getRequestBodyBuilder()
.buildRequestBodyFromDoc(requestBodyDoc, methodAttributes, components,
methodAttributes.getJsonViewAnnotationForRequestBody())
.ifPresent(operation::setRequestBody);
// requests
operation = requestBuilder.build(handlerMethod, requestMethod, operation, methodAttributes, openAPI);
// responses
ApiResponses apiResponses = responseBuilder.build(components, handlerMethod, operation, methodAttributes);
operation.setResponses(apiResponses);
Set apiCallbacks = AnnotatedElementUtils.findMergedRepeatableAnnotations(method, io.swagger.v3.oas.annotations.callbacks.Callback.class);
// callbacks
buildCallbacks(openAPI, methodAttributes, operation, apiCallbacks);
// allow for customisation
customiseOperation(operation, handlerMethod);
PathItem pathItemObject = buildPathItem(requestMethod, operation, operationPath, paths);
paths.addPathItem(operationPath, pathItemObject);
}
}
/**
* Build callbacks.
*
* @param openAPI the open api
* @param methodAttributes the method attributes
* @param operation the operation
* @param apiCallbacks the api callbacks
*/
private void buildCallbacks(OpenAPI openAPI, MethodAttributes methodAttributes, Operation operation, Set apiCallbacks) {
if (!CollectionUtils.isEmpty(apiCallbacks))
operationParser.buildCallbacks(apiCallbacks, openAPI, methodAttributes)
.ifPresent(operation::setCallbacks);
}
/**
* Calculate path.
*
* @param routerOperationList the router operation list
*/
protected void calculatePath(List routerOperationList) {
ApplicationContext applicationContext = openAPIService.getContext();
if (!CollectionUtils.isEmpty(routerOperationList)) {
Collections.sort(routerOperationList);
for (RouterOperation routerOperation : routerOperationList) {
if (routerOperation.getBeanClass() != null && !Void.class.equals(routerOperation.getBeanClass())) {
Object handlerBean = applicationContext.getBean(routerOperation.getBeanClass());
HandlerMethod handlerMethod = null;
if (StringUtils.isNotBlank(routerOperation.getBeanMethod())) {
try {
if (ArrayUtils.isEmpty(routerOperation.getParameterTypes())) {
Method[] declaredMethods = AopUtils.getTargetClass(handlerBean).getDeclaredMethods();
Optional methodOptional = Arrays.stream(declaredMethods)
.filter(method -> routerOperation.getBeanMethod().equals(method.getName()) && method.getParameters().length == 0)
.findAny();
if (!methodOptional.isPresent())
methodOptional = Arrays.stream(declaredMethods)
.filter(method1 -> routerOperation.getBeanMethod().equals(method1.getName()))
.findAny();
if (methodOptional.isPresent())
handlerMethod = new HandlerMethod(handlerBean, methodOptional.get());
}
else
handlerMethod = new HandlerMethod(handlerBean, routerOperation.getBeanMethod(), routerOperation.getParameterTypes());
}
catch (NoSuchMethodException e) {
LOGGER.error(e.getMessage());
}
if (handlerMethod != null && isFilterCondition(handlerMethod, routerOperation.getPath(), routerOperation.getProduces(), routerOperation.getConsumes(), routerOperation.getHeaders()))
calculatePath(handlerMethod, routerOperation);
}
}
else if (routerOperation.getOperation() != null && StringUtils.isNotBlank(routerOperation.getOperation().operationId()) && isFilterCondition(routerOperation.getPath(), routerOperation.getProduces(), routerOperation.getConsumes(), routerOperation.getHeaders())) {
calculatePath(routerOperation);
}
else if (routerOperation.getOperationModel() != null && StringUtils.isNotBlank(routerOperation.getOperationModel().getOperationId()) && isFilterCondition(routerOperation.getPath(), routerOperation.getProduces(), routerOperation.getConsumes(), routerOperation.getHeaders())) {
calculatePath(routerOperation);
}
}
}
}
/**
* Calculate path.
*
* @param routerOperation the router operation
*/
protected void calculatePath(RouterOperation routerOperation) {
String operationPath = routerOperation.getPath();
io.swagger.v3.oas.annotations.Operation apiOperation = routerOperation.getOperation();
String[] methodConsumes = routerOperation.getConsumes();
String[] methodProduces = routerOperation.getProduces();
String[] headers = routerOperation.getHeaders();
Map queryParams = routerOperation.getQueryParams();
OpenAPI openAPI = openAPIService.getCalculatedOpenAPI();
Paths paths = openAPI.getPaths();
Map operationMap = null;
if (paths.containsKey(operationPath)) {
PathItem pathItem = paths.get(operationPath);
operationMap = pathItem.readOperationsMap();
}
for (RequestMethod requestMethod : routerOperation.getMethods()) {
Operation existingOperation = getExistingOperation(operationMap, requestMethod);
MethodAttributes methodAttributes = new MethodAttributes(springDocConfigProperties.getDefaultConsumesMediaType(), springDocConfigProperties.getDefaultProducesMediaType(), methodConsumes, methodProduces, headers);
methodAttributes.setMethodOverloaded(existingOperation != null);
Operation operation = getOperation(routerOperation, existingOperation);
if (apiOperation != null)
openAPI = operationParser.parse(apiOperation, operation, openAPI, methodAttributes);
String operationId = operationParser.getOperationId(operation.getOperationId(), openAPI);
operation.setOperationId(operationId);
fillParametersList(operation, queryParams, methodAttributes);
if (!CollectionUtils.isEmpty(operation.getParameters()))
operation.getParameters().forEach(parameter -> {
if (parameter.getSchema() == null)
parameter.setSchema(new StringSchema());
if (parameter.getIn() == null)
parameter.setIn(ParameterIn.QUERY.toString());
}
);
PathItem pathItemObject = buildPathItem(requestMethod, operation, operationPath, paths);
paths.addPathItem(operationPath, pathItemObject);
}
}
/**
* Calculate path.
*
* @param handlerMethod the handler method
* @param operationPath the operation path
* @param requestMethods the request methods
*/
protected void calculatePath(HandlerMethod handlerMethod, String operationPath,
Set requestMethods) {
this.calculatePath(handlerMethod, new RouterOperation(operationPath, requestMethods.toArray(new RequestMethod[requestMethods.size()])));
}
/**
* Gets router function paths.
*
* @param beanName the bean name
* @param routerFunctionVisitor the router function visitor
*/
protected void getRouterFunctionPaths(String beanName, AbstractRouterFunctionVisitor routerFunctionVisitor) {
boolean withRouterOperation = routerFunctionVisitor.getRouterFunctionDatas().stream()
.anyMatch(routerFunctionData -> routerFunctionData.getAttributes().containsKey(OPERATION_ATTRIBUTE));
if (withRouterOperation) {
List operationList = routerFunctionVisitor.getRouterFunctionDatas().stream().map(RouterOperation::new).collect(Collectors.toList());
calculatePath(operationList);
}
else {
List routerOperationList = new ArrayList<>();
ApplicationContext applicationContext = openAPIService.getContext();
RouterOperations routerOperations = applicationContext.findAnnotationOnBean(beanName, RouterOperations.class);
if (routerOperations == null) {
org.springdoc.core.annotations.RouterOperation routerOperation = applicationContext.findAnnotationOnBean(beanName, org.springdoc.core.annotations.RouterOperation.class);
if (routerOperation != null)
routerOperationList.add(routerOperation);
}
else
routerOperationList.addAll(Arrays.asList(routerOperations.value()));
if (routerOperationList.size() == 1)
calculatePath(routerOperationList.stream().map(routerOperation -> new RouterOperation(routerOperation, routerFunctionVisitor.getRouterFunctionDatas().get(0))).collect(Collectors.toList()));
else {
List operationList = routerOperationList.stream().map(RouterOperation::new).collect(Collectors.toList());
mergeRouters(routerFunctionVisitor.getRouterFunctionDatas(), operationList);
calculatePath(operationList);
}
}
}
/**
* Is filter condition boolean.
*
* @param handlerMethod the handler method
* @param operationPath the operation path
* @param produces the produces
* @param consumes the consumes
* @param headers the headers
* @return the boolean
*/
protected boolean isFilterCondition(HandlerMethod handlerMethod, String operationPath, String[] produces, String[] consumes, String[] headers) {
return isPackageToScan(handlerMethod.getBeanType().getPackage())
&& isFilterCondition(operationPath, produces, consumes, headers);
}
/**
* Is condition to match boolean.
*
* @param existingConditions the existing conditions
* @param conditionType the condition type
* @return the boolean
*/
protected boolean isConditionToMatch(String[] existingConditions, ConditionType conditionType) {
List conditionsToMatch = getConditionsToMatch(conditionType);
if (CollectionUtils.isEmpty(conditionsToMatch)) {
Optional optionalGroupConfig = springDocConfigProperties.getGroupConfigs().stream().filter(groupConfig -> this.groupName.equals(groupConfig.getGroup())).findAny();
if (optionalGroupConfig.isPresent())
conditionsToMatch = getConditionsToMatch(conditionType, optionalGroupConfig.get());
}
return CollectionUtils.isEmpty(conditionsToMatch)
|| (!ArrayUtils.isEmpty(existingConditions) && conditionsToMatch.size() == existingConditions.length && conditionsToMatch.containsAll(Arrays.asList(existingConditions)));
}
/**
* Is package to scan boolean.
*
* @param aPackage the a package
* @return the boolean
*/
protected boolean isPackageToScan(Package aPackage) {
if (aPackage == null)
return true;
final String packageName = aPackage.getName();
List packagesToScan = springDocConfigProperties.getPackagesToScan();
List packagesToExclude = springDocConfigProperties.getPackagesToExclude();
if (CollectionUtils.isEmpty(packagesToScan)) {
Optional optionalGroupConfig = springDocConfigProperties.getGroupConfigs().stream().filter(groupConfig -> this.groupName.equals(groupConfig.getGroup())).findAny();
if (optionalGroupConfig.isPresent())
packagesToScan = optionalGroupConfig.get().getPackagesToScan();
}
if (CollectionUtils.isEmpty(packagesToExclude)) {
Optional optionalGroupConfig = springDocConfigProperties.getGroupConfigs().stream().filter(groupConfig -> this.groupName.equals(groupConfig.getGroup())).findAny();
if (optionalGroupConfig.isPresent())
packagesToExclude = optionalGroupConfig.get().getPackagesToExclude();
}
boolean include = CollectionUtils.isEmpty(packagesToScan)
|| packagesToScan.stream().anyMatch(pack -> packageName.equals(pack)
|| packageName.startsWith(pack + "."));
boolean exclude = !CollectionUtils.isEmpty(packagesToExclude)
&& (packagesToExclude.stream().anyMatch(pack -> packageName.equals(pack)
|| packageName.startsWith(pack + ".")));
return include && !exclude;
}
/**
* Is path to match boolean.
*
* @param operationPath the operation path
* @return the boolean
*/
protected boolean isPathToMatch(String operationPath) {
List pathsToMatch = springDocConfigProperties.getPathsToMatch();
List pathsToExclude = springDocConfigProperties.getPathsToExclude();
if (CollectionUtils.isEmpty(pathsToMatch)) {
Optional optionalGroupConfig = springDocConfigProperties.getGroupConfigs().stream().filter(groupConfig -> this.groupName.equals(groupConfig.getGroup())).findAny();
if (optionalGroupConfig.isPresent())
pathsToMatch = optionalGroupConfig.get().getPathsToMatch();
}
if (CollectionUtils.isEmpty(pathsToExclude)) {
Optional optionalGroupConfig = springDocConfigProperties.getGroupConfigs().stream().filter(groupConfig -> this.groupName.equals(groupConfig.getGroup())).findAny();
if (optionalGroupConfig.isPresent())
pathsToExclude = optionalGroupConfig.get().getPathsToExclude();
}
boolean include = CollectionUtils.isEmpty(pathsToMatch) || pathsToMatch.stream().anyMatch(pattern -> antPathMatcher.match(pattern, operationPath));
boolean exclude = !CollectionUtils.isEmpty(pathsToExclude) && pathsToExclude.stream().anyMatch(pattern -> antPathMatcher.match(pattern, operationPath));
return include && !exclude;
}
/**
* Decode string.
*
* @param requestURI the request uri
* @return the string
*/
protected String decode(String requestURI) {
try {
return URLDecoder.decode(requestURI, StandardCharsets.UTF_8.toString());
}
catch (UnsupportedEncodingException e) {
return requestURI;
}
}
/**
* Is additional rest controller boolean.
*
* @param rawClass the raw class
* @return the boolean
*/
protected boolean isAdditionalRestController(Class> rawClass) {
return ADDITIONAL_REST_CONTROLLERS.stream().anyMatch(clazz -> clazz.isAssignableFrom(rawClass));
}
/**
* Is hidden rest controllers boolean.
*
* @param rawClass the raw class
* @return the boolean
*/
public static boolean isHiddenRestControllers(Class> rawClass) {
return HIDDEN_REST_CONTROLLERS.stream().anyMatch(clazz -> clazz.isAssignableFrom(rawClass));
}
/**
* Gets default allowed http methods.
*
* @return the default allowed http methods
*/
protected Set getDefaultAllowedHttpMethods() {
RequestMethod[] allowedRequestMethods = { RequestMethod.GET, RequestMethod.POST, RequestMethod.PUT, RequestMethod.PATCH, RequestMethod.DELETE, RequestMethod.OPTIONS, RequestMethod.HEAD };
return new HashSet<>(Arrays.asList(allowedRequestMethods));
}
/**
* Customise operation operation.
*
* @param operation the operation
* @param handlerMethod the handler method
* @return the operation
*/
protected Operation customiseOperation(Operation operation, HandlerMethod handlerMethod) {
operationCustomizers.ifPresent(customizers -> customizers.forEach(customizer -> customizer.customize(operation, handlerMethod)));
return operation;
}
/**
* Merge routers.
*
* @param routerFunctionDatas the router function datas
* @param routerOperationList the router operation list
*/
protected void mergeRouters(List routerFunctionDatas, List routerOperationList) {
for (RouterOperation routerOperation : routerOperationList) {
if (StringUtils.isNotBlank(routerOperation.getPath())) {
// PATH
List routerFunctionDataList = routerFunctionDatas.stream()
.filter(routerFunctionData1 -> routerFunctionData1.getPath().equals(routerOperation.getPath()))
.collect(Collectors.toList());
if (routerFunctionDataList.size() == 1)
fillRouterOperation(routerFunctionDataList.get(0), routerOperation);
else if (routerFunctionDataList.size() > 1 && ArrayUtils.isNotEmpty(routerOperation.getMethods())) {
// PATH + METHOD
routerFunctionDataList = routerFunctionDatas.stream()
.filter(routerFunctionData1 -> routerFunctionData1.getPath().equals(routerOperation.getPath())
&& isEqualMethods(routerOperation.getMethods(), routerFunctionData1.getMethods()))
.collect(Collectors.toList());
if (routerFunctionDataList.size() == 1)
fillRouterOperation(routerFunctionDataList.get(0), routerOperation);
else if (routerFunctionDataList.size() > 1 && ArrayUtils.isNotEmpty(routerOperation.getProduces())) {
// PATH + METHOD + PRODUCES
routerFunctionDataList = routerFunctionDatas.stream()
.filter(routerFunctionData1 -> routerFunctionData1.getPath().equals(routerOperation.getPath())
&& isEqualMethods(routerOperation.getMethods(), routerFunctionData1.getMethods())
&& isEqualArrays(routerFunctionData1.getProduces(), routerOperation.getProduces()))
.collect(Collectors.toList());
if (routerFunctionDataList.size() == 1)
fillRouterOperation(routerFunctionDataList.get(0), routerOperation);
else if (routerFunctionDataList.size() > 1 && ArrayUtils.isNotEmpty(routerOperation.getConsumes())) {
// PATH + METHOD + PRODUCES + CONSUMES
routerFunctionDataList = routerFunctionDatas.stream()
.filter(routerFunctionData1 -> routerFunctionData1.getPath().equals(routerOperation.getPath())
&& isEqualMethods(routerOperation.getMethods(), routerFunctionData1.getMethods())
&& isEqualArrays(routerFunctionData1.getProduces(), routerOperation.getProduces())
&& isEqualArrays(routerFunctionData1.getConsumes(), routerOperation.getConsumes()))
.collect(Collectors.toList());
if (routerFunctionDataList.size() == 1)
fillRouterOperation(routerFunctionDataList.get(0), routerOperation);
}
}
else if (routerFunctionDataList.size() > 1 && ArrayUtils.isNotEmpty(routerOperation.getConsumes())) {
// PATH + METHOD + CONSUMES
routerFunctionDataList = routerFunctionDatas.stream()
.filter(routerFunctionData1 -> routerFunctionData1.getPath().equals(routerOperation.getPath())
&& isEqualMethods(routerOperation.getMethods(), routerFunctionData1.getMethods())
&& isEqualArrays(routerFunctionData1.getConsumes(), routerOperation.getConsumes()))
.collect(Collectors.toList());
if (routerFunctionDataList.size() == 1)
fillRouterOperation(routerFunctionDataList.get(0), routerOperation);
}
}
else if (routerFunctionDataList.size() > 1 && ArrayUtils.isNotEmpty(routerOperation.getProduces())) {
// PATH + PRODUCES
routerFunctionDataList = routerFunctionDatas.stream()
.filter(routerFunctionData1 -> routerFunctionData1.getPath().equals(routerOperation.getPath())
&& isEqualArrays(routerFunctionData1.getProduces(), routerOperation.getProduces()))
.collect(Collectors.toList());
if (routerFunctionDataList.size() == 1)
fillRouterOperation(routerFunctionDataList.get(0), routerOperation);
else if (routerFunctionDataList.size() > 1 && ArrayUtils.isNotEmpty(routerOperation.getConsumes())) {
// PATH + PRODUCES + CONSUMES
routerFunctionDataList = routerFunctionDatas.stream()
.filter(routerFunctionData1 -> routerFunctionData1.getPath().equals(routerOperation.getPath())
&& isEqualMethods(routerOperation.getMethods(), routerFunctionData1.getMethods())
&& isEqualArrays(routerFunctionData1.getConsumes(), routerOperation.getConsumes())
&& isEqualArrays(routerFunctionData1.getProduces(), routerOperation.getProduces()))
.collect(Collectors.toList());
if (routerFunctionDataList.size() == 1)
fillRouterOperation(routerFunctionDataList.get(0), routerOperation);
}
}
else if (routerFunctionDataList.size() > 1 && ArrayUtils.isNotEmpty(routerOperation.getConsumes())) {
// PATH + CONSUMES
routerFunctionDataList = routerFunctionDatas.stream()
.filter(routerFunctionData1 -> routerFunctionData1.getPath().equals(routerOperation.getPath())
&& isEqualArrays(routerFunctionData1.getConsumes(), routerOperation.getConsumes()))
.collect(Collectors.toList());
if (routerFunctionDataList.size() == 1)
fillRouterOperation(routerFunctionDataList.get(0), routerOperation);
}
}
}
}
/**
* Calculate json view.
*
* @param apiOperation the api operation
* @param methodAttributes the method attributes
* @param method the method
*/
private void calculateJsonView(io.swagger.v3.oas.annotations.Operation apiOperation,
MethodAttributes methodAttributes, Method method) {
JsonView jsonViewAnnotation;
JsonView jsonViewAnnotationForRequestBody;
if (apiOperation != null && apiOperation.ignoreJsonView()) {
jsonViewAnnotation = null;
jsonViewAnnotationForRequestBody = null;
}
else {
jsonViewAnnotation = AnnotatedElementUtils.findMergedAnnotation(method, JsonView.class);
/*
* If one and only one exists, use the @JsonView annotation from the method
* parameter annotated with @RequestBody. Otherwise fall back to the @JsonView
* annotation for the method itself.
*/
jsonViewAnnotationForRequestBody = (JsonView) Arrays.stream(ReflectionUtils.getParameterAnnotations(method))
.filter(arr -> Arrays.stream(arr)
.anyMatch(annotation -> (annotation.annotationType()
.equals(io.swagger.v3.oas.annotations.parameters.RequestBody.class) || annotation.annotationType().equals(RequestBody.class))))
.flatMap(Arrays::stream).filter(annotation -> annotation.annotationType().equals(JsonView.class))
.reduce((a, b) -> null).orElse(jsonViewAnnotation);
}
methodAttributes.setJsonViewAnnotation(jsonViewAnnotation);
methodAttributes.setJsonViewAnnotationForRequestBody(jsonViewAnnotationForRequestBody);
}
/**
* Is equal arrays boolean.
*
* @param array1 the array 1
* @param array2 the array 2
* @return the boolean
*/
private boolean isEqualArrays(String[] array1, String[] array2) {
Arrays.sort(array1);
Arrays.sort(array2);
return Arrays.equals(array1, array2);
}
/**
* Is equal methods boolean.
*
* @param requestMethods1 the request methods 1
* @param requestMethods2 the request methods 2
* @return the boolean
*/
private boolean isEqualMethods(RequestMethod[] requestMethods1, RequestMethod[] requestMethods2) {
Arrays.sort(requestMethods1);
Arrays.sort(requestMethods2);
return Arrays.equals(requestMethods1, requestMethods2);
}
/**
* Fill parameters list.
*
* @param operation the operation
* @param queryParams the query params
* @param methodAttributes the method attributes
*/
private void fillParametersList(Operation operation, Map queryParams, MethodAttributes methodAttributes) {
List parametersList = operation.getParameters();
if (parametersList == null)
parametersList = new ArrayList<>();
Collection headersMap = AbstractRequestService.getHeaders(methodAttributes, new LinkedHashMap<>());
parametersList.addAll(headersMap);
if (!CollectionUtils.isEmpty(queryParams)) {
for (Map.Entry entry : queryParams.entrySet()) {
io.swagger.v3.oas.models.parameters.Parameter parameter = new io.swagger.v3.oas.models.parameters.Parameter();
parameter.setName(entry.getKey());
parameter.setSchema(new StringSchema()._default(entry.getValue()));
parameter.setRequired(true);
parameter.setIn(ParameterIn.QUERY.toString());
GenericParameterService.mergeParameter(parametersList, parameter);
}
operation.setParameters(parametersList);
}
}
/**
* Fill router operation.
*
* @param routerFunctionData the router function data
* @param routerOperation the router operation
*/
private void fillRouterOperation(RouterFunctionData routerFunctionData, RouterOperation routerOperation) {
if (ArrayUtils.isEmpty(routerOperation.getConsumes()))
routerOperation.setConsumes(routerFunctionData.getConsumes());
if (ArrayUtils.isEmpty(routerOperation.getProduces()))
routerOperation.setProduces(routerFunctionData.getProduces());
if (ArrayUtils.isEmpty(routerOperation.getHeaders()))
routerOperation.setHeaders(routerFunctionData.getHeaders());
if (ArrayUtils.isEmpty(routerOperation.getMethods()))
routerOperation.setMethods(routerFunctionData.getMethods());
if (CollectionUtils.isEmpty(routerOperation.getQueryParams()))
routerOperation.setQueryParams(routerFunctionData.getQueryParams());
}
/**
* Build path item path item.
*
* @param requestMethod the request method
* @param operation the operation
* @param operationPath the operation path
* @param paths the paths
* @return the path item
*/
private PathItem buildPathItem(RequestMethod requestMethod, Operation operation, String operationPath,
Paths paths) {
PathItem pathItemObject;
if (paths.containsKey(operationPath))
pathItemObject = paths.get(operationPath);
else
pathItemObject = new PathItem();
switch (requestMethod) {
case POST:
pathItemObject.post(operation);
break;
case GET:
pathItemObject.get(operation);
break;
case DELETE:
pathItemObject.delete(operation);
break;
case PUT:
pathItemObject.put(operation);
break;
case PATCH:
pathItemObject.patch(operation);
break;
case TRACE:
pathItemObject.trace(operation);
break;
case HEAD:
pathItemObject.head(operation);
break;
case OPTIONS:
pathItemObject.options(operation);
break;
default:
// Do nothing here
break;
}
return pathItemObject;
}
/**
* Gets existing operation.
*
* @param operationMap the operation map
* @param requestMethod the request method
* @return the existing operation
*/
private Operation getExistingOperation(Map operationMap, RequestMethod requestMethod) {
Operation existingOperation = null;
if (!CollectionUtils.isEmpty(operationMap)) {
// Get existing operation definition
switch (requestMethod) {
case GET:
existingOperation = operationMap.get(HttpMethod.GET);
break;
case POST:
existingOperation = operationMap.get(HttpMethod.POST);
break;
case PUT:
existingOperation = operationMap.get(HttpMethod.PUT);
break;
case DELETE:
existingOperation = operationMap.get(HttpMethod.DELETE);
break;
case PATCH:
existingOperation = operationMap.get(HttpMethod.PATCH);
break;
case HEAD:
existingOperation = operationMap.get(HttpMethod.HEAD);
break;
case OPTIONS:
existingOperation = operationMap.get(HttpMethod.OPTIONS);
break;
default:
// Do nothing here
break;
}
}
return existingOperation;
}
/**
* Gets operation.
*
* @param routerOperation the router operation
* @param existingOperation the existing operation
* @return the operation
*/
private Operation getOperation(RouterOperation routerOperation, Operation existingOperation) {
Operation operationModel = routerOperation.getOperationModel();
Operation operation;
if (existingOperation != null && operationModel == null) {
operation = existingOperation;
}
else if (existingOperation == null && operationModel != null) {
operation = operationModel;
}
else if (existingOperation != null) {
operation = operationParser.mergeOperation(existingOperation, operationModel);
}
else {
operation = new Operation();
}
return operation;
}
/**
* Init open api builder.
*/
protected void initOpenAPIBuilder() {
if (openAPIService.getCachedOpenAPI() != null && springDocConfigProperties.isCacheDisabled()) {
openAPIService = openAPIBuilderObjectFactory.getObject();
}
}
/**
* Write yaml value string.
*
* @param openAPI the open api
* @return the string
* @throws JsonProcessingException the json processing exception
*/
protected String writeYamlValue(OpenAPI openAPI) throws JsonProcessingException {
String result;
ObjectMapper objectMapper = Yaml.mapper();
if (springDocConfigProperties.isWriterWithOrderByKeys())
sortOutput(objectMapper);
YAMLFactory factory = (YAMLFactory) objectMapper.getFactory();
factory.configure(Feature.USE_NATIVE_TYPE_ID, false);
if (!springDocConfigProperties.isWriterWithDefaultPrettyPrinter())
result = objectMapper.writerFor(OpenAPI.class).writeValueAsString(openAPI);
else
result = objectMapper.writerWithDefaultPrettyPrinter().forType(OpenAPI.class).writeValueAsString(openAPI);
return result;
}
/**
* Gets actuator uri.
*
* @param scheme the scheme
* @param host the host
* @return the actuator uri
*/
protected URI getActuatorURI(String scheme, String host) {
int port;
String path;
URI uri = null;
if (optionalActuatorProvider.isPresent()) {
ActuatorProvider actuatorProvider = optionalActuatorProvider.get();
if (ACTUATOR_DEFAULT_GROUP.equals(this.groupName)) {
port = actuatorProvider.getActuatorPort();
path = actuatorProvider.getActuatorPath();
}
else {
port = actuatorProvider.getApplicationPort();
path = actuatorProvider.getContextPath();
String mvcServletPath = this.openAPIService.getContext().getBean(Environment.class).getProperty(SPRING_MVC_SERVLET_PATH);
if (StringUtils.isNotEmpty(mvcServletPath))
path = path + mvcServletPath;
}
try {
uri = new URI(StringUtils.defaultIfEmpty(scheme, "http"), null, StringUtils.defaultIfEmpty(host, "localhost"), port, path, null, null);
}
catch (URISyntaxException e) {
LOGGER.error("Unable to parse the URL: scheme {}, host {}, port {}, path {}", scheme, host, port, path);
}
}
return uri;
}
/**
* Is show actuator boolean.
*
* @return the boolean
*/
protected boolean isShowActuator() {
return springDocConfigProperties.isShowActuator() && optionalActuatorProvider.isPresent();
}
/**
* Write json value string.
*
* @param openAPI the open api
* @return the string
* @throws JsonProcessingException the json processing exception
*/
protected String writeJsonValue(OpenAPI openAPI) throws JsonProcessingException {
String result;
ObjectMapper objectMapper = Json.mapper();
if (springDocConfigProperties.isWriterWithOrderByKeys())
sortOutput(objectMapper);
if (!springDocConfigProperties.isWriterWithDefaultPrettyPrinter())
result = objectMapper.writerFor(OpenAPI.class).writeValueAsString(openAPI);
else
result = objectMapper.writerWithDefaultPrettyPrinter().forType(OpenAPI.class).writeValueAsString(openAPI);
return result;
}
/**
* Gets conditions to match.
*
* @param conditionType the condition type
* @param groupConfigs the group configs
* @return the conditions to match
*/
private List getConditionsToMatch(ConditionType conditionType, GroupConfig... groupConfigs) {
List conditionsToMatch = null;
GroupConfig groupConfig = null;
if (ArrayUtils.isNotEmpty(groupConfigs))
groupConfig = groupConfigs[0];
switch (conditionType) {
case HEADERS:
conditionsToMatch = (groupConfig != null) ? groupConfig.getHeadersToMatch() : springDocConfigProperties.getHeadersToMatch();
break;
case PRODUCES:
conditionsToMatch = (groupConfig != null) ? groupConfig.getProducesToMatch() : springDocConfigProperties.getProducesToMatch();
break;
case CONSUMES:
conditionsToMatch = (groupConfig != null) ? groupConfig.getConsumesToMatch() : springDocConfigProperties.getConsumesToMatch();
break;
default:
break;
} return conditionsToMatch;
}
/**
* Is filter condition boolean.
*
* @param operationPath the operation path
* @param produces the produces
* @param consumes the consumes
* @param headers the headers
* @return the boolean
*/
private boolean isFilterCondition(String operationPath, String[] produces, String[] consumes, String[] headers) {
return isPathToMatch(operationPath)
&& isConditionToMatch(produces, ConditionType.PRODUCES)
&& isConditionToMatch(consumes, ConditionType.CONSUMES)
&& isConditionToMatch(headers, ConditionType.HEADERS);
}
/**
* The enum Condition type.
*/
enum ConditionType {
/**
*Produces condition type.
*/
PRODUCES,
/**
*Consumes condition type.
*/
CONSUMES,
/**
*Headers condition type.
*/
HEADERS
}
private void sortOutput(ObjectMapper objectMapper) {
objectMapper.configure(SerializationFeature.ORDER_MAP_ENTRIES_BY_KEYS, true);
objectMapper.configure(MapperFeature.SORT_PROPERTIES_ALPHABETICALLY, true);
objectMapper.addMixIn(OpenAPI.class, SortedOpenAPIMixin.class);
objectMapper.addMixIn(Schema.class, SortedSchemaMixin.class);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy