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

org.apache.camel.component.rest.swagger.RestSwaggerEndpoint Maven / Gradle / Ivy

The newest version!
/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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 org.apache.camel.component.rest.swagger;

import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Objects;
import java.util.Optional;
import java.util.function.Function;
import java.util.stream.Collectors;
import java.util.stream.Stream;

import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import io.swagger.models.HttpMethod;
import io.swagger.models.Operation;
import io.swagger.models.Path;
import io.swagger.models.Scheme;
import io.swagger.models.Swagger;
import io.swagger.models.auth.ApiKeyAuthDefinition;
import io.swagger.models.auth.In;
import io.swagger.models.auth.SecuritySchemeDefinition;
import io.swagger.models.parameters.Parameter;
import io.swagger.models.parameters.QueryParameter;
import io.swagger.parser.SwaggerParser;
import io.swagger.util.Json;
import org.apache.camel.CamelContext;
import org.apache.camel.Category;
import org.apache.camel.Consumer;
import org.apache.camel.Endpoint;
import org.apache.camel.ExchangePattern;
import org.apache.camel.Processor;
import org.apache.camel.Producer;
import org.apache.camel.component.http.HttpComponent;
import org.apache.camel.component.http.HttpEndpoint;
import org.apache.camel.component.http.HttpProducer;
import org.apache.camel.spi.Metadata;
import org.apache.camel.spi.RestConfiguration;
import org.apache.camel.spi.UriEndpoint;
import org.apache.camel.spi.UriParam;
import org.apache.camel.spi.UriPath;
import org.apache.camel.support.CamelContextHelper;
import org.apache.camel.support.DefaultEndpoint;
import org.apache.camel.support.ResourceHelper;
import org.apache.camel.support.jsse.SSLContextParameters;
import org.apache.camel.util.ObjectHelper;
import org.apache.camel.util.StringHelper;
import org.apache.camel.util.UnsafeUriCharactersEncoder;
import org.apache.http.client.methods.HttpGet;

import static java.util.Optional.ofNullable;
import static org.apache.camel.component.rest.swagger.RestSwaggerHelper.isHostParam;
import static org.apache.camel.component.rest.swagger.RestSwaggerHelper.isMediaRange;
import static org.apache.camel.util.ObjectHelper.isNotEmpty;
import static org.apache.camel.util.ObjectHelper.notNull;
import static org.apache.camel.util.StringHelper.after;
import static org.apache.camel.util.StringHelper.before;
import static org.apache.camel.util.StringHelper.notEmpty;

/**
 * Configure REST producers based on a Swagger (OpenAPI) specification document delegating to a component implementing
 * the RestProducerFactory interface.
 */
@UriEndpoint(firstVersion = "2.19.0", scheme = "rest-swagger", title = "REST Swagger",
             syntax = "rest-swagger:specificationUri#operationId",
             category = { Category.REST, Category.SWAGGER, Category.HTTP }, producerOnly = true)
public final class RestSwaggerEndpoint extends DefaultEndpoint {

    /**
     * Remaining parameters specified in the Endpoint URI.
     */
    Map parameters = Collections.emptyMap();

    @UriParam(
              description = "API basePath, for example \"`/v2`\". Default is unset, if set overrides the value present in"
                            + " Swagger specification and in the component configuration.",
              defaultValue = "", label = "producer")
    private String basePath;

    @UriParam(description = "Name of the Camel component that will perform the requests. The component must be present"
                            + " in Camel registry and it must implement RestProducerFactory service provider interface. If not set"
                            + " CLASSPATH is searched for single component that implements RestProducerFactory SPI. Overrides"
                            + " component configuration.",
              label = "producer")
    private String componentName;

    @UriParam(
              description = "What payload type this component capable of consuming. Could be one type, like `application/json`"
                            + " or multiple types as `application/json, application/xml; q=0.5` according to the RFC7231. This equates"
                            + " to the value of `Accept` HTTP header. If set overrides any value found in the Swagger specification and."
                            + " in the component configuration",
              label = "producer")
    private String consumes;

    @UriParam(description = "Scheme hostname and port to direct the HTTP requests to in the form of"
                            + " `http[s]://hostname[:port]`. Can be configured at the endpoint, component or in the corresponding"
                            + " REST configuration in the Camel Context. If you give this component a name (e.g. `petstore`) that"
                            + " REST configuration is consulted first, `rest-swagger` next, and global configuration last. If set"
                            + " overrides any value found in the Swagger specification, RestConfiguration. Overrides all other "
                            + " configuration.",
              label = "producer")
    private String host;

    @UriPath(description = "ID of the operation from the Swagger specification.", label = "producer")
    @Metadata(required = true)
    private String operationId;

    @UriParam(description = "What payload type this component is producing. For example `application/json`"
                            + " according to the RFC7231. This equates to the value of `Content-Type` HTTP header. If set overrides"
                            + " any value present in the Swagger specification. Overrides all other configuration.",
              label = "producer")
    private String produces;

    @UriParam(label = "security", description = "To configure security using SSLContextParameters.")
    private SSLContextParameters sslContextParameters;

    @UriPath(description = "Path to the Swagger specification file. The scheme, host base path are taken from this"
                           + " specification, but these can be overridden with properties on the component or endpoint level. If not"
                           + " given the component tries to load `swagger.json` resource from the classpath. Note that the `host` defined on the"
                           + " component and endpoint of this Component should contain the scheme, hostname and optionally the"
                           + " port in the URI syntax (i.e. `http://api.example.com:8080`). Overrides component configuration."
                           + " The Swagger specification can be loaded from different sources by prefixing with file: classpath: http: https:."
                           + " Support for https is limited to using the JDK installed UrlHandler, and as such it can be cumbersome to setup"
                           + " TLS/SSL certificates for https (such as setting a number of javax.net.ssl JVM system properties)."
                           + " How to do that consult the JDK documentation for UrlHandler.",
             defaultValue = RestSwaggerComponent.DEFAULT_SPECIFICATION_URI_STR,
             defaultValueNote = "By default loads `swagger.json` file", label = "producer")
    private URI specificationUri = RestSwaggerComponent.DEFAULT_SPECIFICATION_URI;

    @UriParam(description = "Resolve references in Swagger specification.", label = "producer")
    private Boolean resolveReferences;

    public RestSwaggerEndpoint() {
        // help tooling instantiate endpoint
    }

    public RestSwaggerEndpoint(final String uri, final String remaining, final RestSwaggerComponent component,
                               final Map parameters) {
        super(notEmpty(uri, "uri"), notNull(component, "component"));
        this.parameters = parameters;

        final URI componentSpecificationUri = component.getSpecificationUri();

        specificationUri = before(remaining, "#", StringHelper::trimToNull).map(URI::create)
                .orElse(ofNullable(componentSpecificationUri).orElse(RestSwaggerComponent.DEFAULT_SPECIFICATION_URI));

        operationId = ofNullable(after(remaining, "#")).orElse(remaining);

        setExchangePattern(ExchangePattern.InOut);
    }

    @Override
    public Consumer createConsumer(final Processor processor) throws Exception {
        throw new UnsupportedOperationException("Consumer not supported");
    }

    @Override
    public Producer createProducer() throws Exception {
        final CamelContext camelContext = getCamelContext();

        final Swagger swagger = loadSpecificationFrom(camelContext, specificationUri, resolveSslContextParameters(),
                determineResolveReferences());

        final Map paths = swagger.getPaths();

        for (final Entry pathEntry : paths.entrySet()) {
            final Path path = pathEntry.getValue();

            final Optional> maybeOperationEntry = path.getOperationMap().entrySet()
                    .stream().filter(operationEntry -> operationId.equals(operationEntry.getValue().getOperationId()))
                    .findAny();

            if (maybeOperationEntry.isPresent()) {
                final Entry operationEntry = maybeOperationEntry.get();

                final Operation operation = operationEntry.getValue();
                final Map pathParameters = operation.getParameters().stream()
                        .filter(p -> "path".equals(p.getIn()))
                        .collect(Collectors.toMap(Parameter::getName, Function.identity()));
                final String uriTemplate = resolveUri(pathEntry.getKey(), pathParameters);

                final HttpMethod httpMethod = operationEntry.getKey();
                final String method = httpMethod.name();

                return createProducerFor(swagger, operation, method, uriTemplate);
            }
        }

        final String supportedOperations = paths.values().stream().flatMap(p -> p.getOperations().stream())
                .map(Operation::getOperationId).collect(Collectors.joining(", "));

        throw new IllegalArgumentException(
                "The specified operation with ID: `" + operationId
                                           + "` cannot be found in the Swagger specification loaded from `" + specificationUri
                                           + "`. Operations defined in the specification are: " + supportedOperations);
    }

    private SSLContextParameters resolveSslContextParameters() {
        if (sslContextParameters != null) {
            return sslContextParameters;
        }
        if (component().getSslContextParameters() != null) {
            return component().getSslContextParameters();
        }
        return component().retrieveGlobalSslContextParameters();
    }

    public String getBasePath() {
        return basePath;
    }

    public String getComponentName() {
        return componentName;
    }

    public String getConsumes() {
        return consumes;
    }

    public SSLContextParameters getSslContextParameters() {
        return sslContextParameters;
    }

    public String getHost() {
        return host;
    }

    public String getOperationId() {
        return operationId;
    }

    public String getProduces() {
        return produces;
    }

    public URI getSpecificationUri() {
        return specificationUri;
    }

    @Override
    public boolean isLenientProperties() {
        return true;
    }

    public void setBasePath(final String basePath) {
        this.basePath = notEmpty(basePath, "basePath");
    }

    public void setComponentName(final String componentName) {
        this.componentName = notEmpty(componentName, "componentName");
    }

    public void setConsumes(final String consumes) {
        this.consumes = isMediaRange(consumes, "consumes");
    }

    public void setSslContextParameters(SSLContextParameters sslContextParameters) {
        this.sslContextParameters = sslContextParameters;
    }

    public void setHost(final String host) {
        this.host = isHostParam(host);
    }

    public void setOperationId(final String operationId) {
        this.operationId = notEmpty(operationId, "operationId");
    }

    public void setProduces(final String produces) {
        this.produces = isMediaRange(produces, "produces");
    }

    public void setSpecificationUri(final URI specificationUri) {
        this.specificationUri = notNull(specificationUri, "specificationUri");
    }

    public Boolean getResolveReferences() {
        return resolveReferences;
    }

    public void setResolveReferences(Boolean resolveReferences) {
        this.resolveReferences = resolveReferences;
    }

    RestSwaggerComponent component() {
        return (RestSwaggerComponent) getComponent();
    }

    Producer createProducerFor(
            final Swagger swagger, final Operation operation, final String method,
            final String uriTemplate)
            throws Exception {
        final String basePath = determineBasePath(swagger);

        final StringBuilder componentEndpointUri = new StringBuilder(200).append("rest:").append(method).append(":")
                .append(basePath).append(":").append(uriTemplate);

        final CamelContext camelContext = getCamelContext();

        final Endpoint endpoint = camelContext.getEndpoint(componentEndpointUri.toString());

        Map params = determineEndpointParameters(swagger, operation);
        boolean hasHost = params.containsKey("host");
        if (endpoint instanceof DefaultEndpoint) {
            // let the rest endpoint configure itself
            DefaultEndpoint de = (DefaultEndpoint) endpoint;
            de.setProperties(endpoint, params);
        }

        // if there is a host then we should use this hardcoded host instead of any Header that may have an existing
        // Host header from some other HTTP input, and if so then lets remove it
        return new RestSwaggerProducer(endpoint.createProducer(), hasHost);
    }

    String determineBasePath(final Swagger swagger) {
        if (isNotEmpty(basePath)) {
            return basePath;
        }

        final String componentBasePath = component().getBasePath();
        if (isNotEmpty(componentBasePath)) {
            return componentBasePath;
        }

        final String specificationBasePath = swagger.getBasePath();
        if (isNotEmpty(specificationBasePath)) {
            return specificationBasePath;
        }

        final CamelContext camelContext = getCamelContext();
        final RestConfiguration restConfiguration
                = CamelContextHelper.getRestConfiguration(camelContext, null, determineComponentName());
        final String restConfigurationBasePath = restConfiguration.getContextPath();

        if (isNotEmpty(restConfigurationBasePath)) {
            return restConfigurationBasePath;
        }

        return RestSwaggerComponent.DEFAULT_BASE_PATH;
    }

    String determineComponentName() {
        return Optional.ofNullable(componentName).orElse(component().getComponentName());
    }

    Boolean determineResolveReferences() {
        return Stream.of(getResolveReferences(), component().getResolveReferences())
                .filter(Objects::nonNull)
                .findFirst()
                .orElse(false);
    }

    Map determineEndpointParameters(final Swagger swagger, final Operation operation) {
        final Map parameters = new HashMap<>();

        final String componentName = determineComponentName();
        if (componentName != null) {
            parameters.put("producerComponentName", componentName);
        }

        final String host = determineHost(swagger);
        if (host != null) {
            parameters.put("host", host);
        }

        final RestSwaggerComponent component = component();

        // what we consume is what the API defined by Swagger specification
        // produces
        final String determinedConsumes = determineOption(swagger.getProduces(), operation.getProduces(),
                component.getConsumes(), consumes);

        if (isNotEmpty(determinedConsumes)) {
            parameters.put("consumes", determinedConsumes);
        }

        // what we produce is what the API defined by Swagger specification
        // consumes
        final String determinedProduces = determineOption(swagger.getConsumes(), operation.getConsumes(),
                component.getProduces(), produces);

        if (isNotEmpty(determinedProduces)) {
            parameters.put("produces", determinedProduces);
        }

        final String queryParameters = determineQueryParameters(swagger, operation).map(this::queryParameter)
                .collect(Collectors.joining("&"));
        if (isNotEmpty(queryParameters)) {
            parameters.put("queryParameters", queryParameters);
        }

        // pass properties that might be applied if the delegate component is
        // created, i.e. if it's not
        // present in the Camel Context already
        final Map componentParameters = new HashMap<>();

        if (component.isUseGlobalSslContextParameters()) {
            // by default it's false
            componentParameters.put("useGlobalSslContextParameters", component.isUseGlobalSslContextParameters());
        }
        if (component.getSslContextParameters() != null) {
            componentParameters.put("sslContextParameters", component.getSslContextParameters());
        }

        final Map nestedParameters = new HashMap<>();
        if (!componentParameters.isEmpty()) {
            // we're trying to set RestEndpoint.parameters['component']
            nestedParameters.put("component", componentParameters);
        }

        // Add rest endpoint parameters
        addRestEndpointParameters(operation, nestedParameters);

        if (!nestedParameters.isEmpty()) {
            parameters.put("parameters", nestedParameters);
        }

        return parameters;
    }

    private void addRestEndpointParameters(Operation operation, Map nestedParameters) {
        if (this.parameters != null && operation.getParameters() != null) {
            for (Entry entry : this.parameters.entrySet()) {
                for (Parameter param : operation.getParameters()) {
                    // skip parameters that are part of the operation as path as otherwise
                    // it will be duplicated as query parameter as well
                    boolean clash = "path".equals(param.getIn()) && entry.getKey().equals(param.getName());
                    if (!clash) {
                        nestedParameters.put(entry.getKey(), entry.getValue());
                    }
                }
            }
        }
    }

    String determineHost(final Swagger swagger) {
        if (isNotEmpty(host)) {
            return host;
        }

        final String componentHost = component().getHost();
        if (isNotEmpty(componentHost)) {
            return componentHost;
        }

        final String swaggerScheme = pickBestScheme(specificationUri.getScheme(), swagger.getSchemes());
        final String swaggerHost = swagger.getHost();

        if (isNotEmpty(swaggerScheme) && isNotEmpty(swaggerHost)) {
            return swaggerScheme + "://" + swaggerHost;
        }

        final CamelContext camelContext = getCamelContext();
        final RestConfiguration globalRestConfiguration
                = CamelContextHelper.getRestConfiguration(camelContext, null, determineComponentName());
        final String globalConfigurationHost = hostFrom(globalRestConfiguration);

        if (globalConfigurationHost != null) {
            return globalConfigurationHost;
        }

        final String specificationScheme = specificationUri.getScheme();
        // Perform a case insensitive "startsWith" check that works for different locales
        String prefix = Scheme.HTTP.toValue();
        if (specificationUri.isAbsolute() && specificationScheme.regionMatches(true, 0, prefix, 0, prefix.length())) {
            try {
                return new URI(
                        specificationUri.getScheme(), specificationUri.getUserInfo(), specificationUri.getHost(),
                        specificationUri.getPort(), null, null, null).toString();
            } catch (final URISyntaxException e) {
                throw new IllegalStateException("Unable to create a new URI from: " + specificationUri, e);
            }
        }

        throw new IllegalStateException(
                "Unable to determine destination host for requests. The Swagger specification"
                                        + " does not specify `scheme` and `host` parameters, the specification URI is not absolute with `http` or"
                                        + " `https` scheme, and no RestConfigurations configured with `scheme`, `host` and `port` were found for `"
                                        + (determineComponentName() != null
                                                ? determineComponentName() : "default" + "` component")
                                        + " and there is no global RestConfiguration with those properties");
    }

    String literalPathParameterValue(final Parameter parameter) {
        final String name = parameter.getName();

        final String valueStr = String.valueOf(parameters.get(name));
        final String encoded = UnsafeUriCharactersEncoder.encode(valueStr);

        return encoded;
    }

    String literalQueryParameterValue(final Parameter parameter) {
        final String name = parameter.getName();

        final String valueStr = String.valueOf(parameters.get(name));
        final String encoded = UnsafeUriCharactersEncoder.encode(valueStr);

        return name + "=" + encoded;
    }

    String queryParameter(final Parameter parameter) {
        final String name = parameter.getName();
        if (ObjectHelper.isEmpty(name)) {
            return "";
        }

        if (parameters.containsKey(name)) {
            return literalQueryParameterValue(parameter);
        }

        return queryParameterExpression(parameter);
    }

    String resolveUri(final String uriTemplate, final Map pathParameters) {
        if (pathParameters.isEmpty()) {
            return uriTemplate;
        }

        int start = uriTemplate.indexOf('{');

        if (start == -1) {
            return uriTemplate;
        }

        int pos = 0;
        final StringBuilder resolved = new StringBuilder(uriTemplate.length() * 2);
        while (start != -1) {
            resolved.append(uriTemplate, pos, start);

            final int end = uriTemplate.indexOf('}', start);

            final String name = uriTemplate.substring(start + 1, end);

            if (parameters.containsKey(name)) {
                final Parameter parameter = pathParameters.get(name);
                final Object value = literalPathParameterValue(parameter);
                resolved.append(value);
            } else {
                resolved.append('{').append(name).append('}');
            }

            pos = end + 1;
            start = uriTemplate.indexOf('{', pos);
        }

        if (pos < uriTemplate.length()) {
            resolved.append(uriTemplate, pos, uriTemplate.length());
        }

        return resolved.toString();
    }

    static String determineOption(
            final List specificationLevel, final List operationLevel,
            final String componentLevel, final String endpointLevel) {
        if (isNotEmpty(endpointLevel)) {
            return endpointLevel;
        }

        if (isNotEmpty(componentLevel)) {
            return componentLevel;
        }

        if (operationLevel != null && !operationLevel.isEmpty()) {
            return String.join(", ", operationLevel);
        }

        if (specificationLevel != null && !specificationLevel.isEmpty()) {
            return String.join(", ", specificationLevel);
        }

        return null;
    }

    static Stream determineQueryParameters(final Swagger swagger, final Operation operation) {
        final List>> securityRequirements = operation.getSecurity();
        final List apiKeyQueryParameters = new ArrayList<>();
        if (securityRequirements != null) {
            final Map securityDefinitions = swagger.getSecurityDefinitions();

            for (final Map> securityRequirement : securityRequirements) {
                for (final String securityRequirementName : securityRequirement.keySet()) {
                    final SecuritySchemeDefinition securitySchemeDefinition = securityDefinitions
                            .get(securityRequirementName);
                    if (securitySchemeDefinition instanceof ApiKeyAuthDefinition) {
                        final ApiKeyAuthDefinition apiKeyDefinition = (ApiKeyAuthDefinition) securitySchemeDefinition;

                        if (apiKeyDefinition.getIn() == In.QUERY) {
                            apiKeyQueryParameters.add(new QueryParameter().name(apiKeyDefinition.getName())
                                    .required(true).type("string").description(apiKeyDefinition.getDescription()));
                        }
                    }
                }
            }
        }

        return Stream.concat(apiKeyQueryParameters.stream(),
                operation.getParameters().stream().filter(p -> "query".equals(p.getIn())));
    }

    static String hostFrom(final RestConfiguration restConfiguration) {
        if (restConfiguration == null) {
            return null;
        }

        final String scheme = restConfiguration.getScheme();
        final String host = restConfiguration.getHost();
        final int port = restConfiguration.getPort();

        if (scheme == null || host == null) {
            return null;
        }

        final StringBuilder answer = new StringBuilder(scheme).append("://").append(host);
        if (port > 0 && !(Scheme.HTTP.toValue().equalsIgnoreCase(scheme) && port == 80)
                && !(Scheme.HTTPS.toValue().equalsIgnoreCase(scheme) && port == 443)) {
            answer.append(':').append(port);
        }

        return answer.toString();
    }

    /**
     * Loads the Swagger definition model from the given path. Tries to resolve the resource using Camel's resource
     * loading support, if it fails uses Swagger's resource loading support instead.
     *
     * @param  uri          URI of the specification
     * @param  camelContext context to use
     * @return              the specification
     * @throws IOException
     */
    static Swagger loadSpecificationFrom(
            final CamelContext camelContext, final URI uri, SSLContextParameters sslContextParameters,
            boolean resolveReferences)
            throws IOException {
        final ObjectMapper mapper = Json.mapper();

        final SwaggerParser swaggerParser = new SwaggerParser();

        final String uriAsString = uri.toString();

        if (sslContextParameters == null) {
            try (InputStream stream = ResourceHelper.resolveMandatoryResourceAsInputStream(camelContext, uriAsString)) {
                return parseInputStream(swaggerParser, mapper, stream, resolveReferences);
            } catch (final Exception e) {
                return loadSpecificationFallback(swaggerParser, uriAsString, e, resolveReferences);
            }
        }

        HttpComponent httpComponent = new HttpComponent();
        httpComponent.setSslContextParameters(sslContextParameters);
        httpComponent.setCamelContext(camelContext);
        try (HttpEndpoint e = (HttpEndpoint) httpComponent.createEndpoint(uriAsString);
             HttpProducer p = (HttpProducer) e.createProducer();
             InputStream stream = p.getHttpClient().execute(new HttpGet(uri)).getEntity().getContent()) {

            return parseInputStream(swaggerParser, mapper, stream, resolveReferences);
        } catch (final Exception e) {
            return loadSpecificationFallback(swaggerParser, uriAsString, e, resolveReferences);
        }
    }

    static Swagger loadSpecificationFallback(
            SwaggerParser swaggerParser, String uriAsString, Exception originalException, boolean resolveReferences) {
        // try Swaggers loader
        final Swagger swagger = swaggerParser.read(uriAsString, null, resolveReferences);

        if (swagger != null) {
            return swagger;
        }

        throw new IllegalArgumentException(
                "The given Swagger specification could not be loaded from `" + uriAsString
                                           + "`. Tried loading using Camel's resource resolution and using Swagger's own resource resolution."
                                           + " Swagger tends to swallow exceptions while parsing, try specifying Java system property `debugParser`"
                                           + " (e.g. `-DdebugParser=true`), the exception that occurred when loading using Camel's resource"
                                           + " loader follows",
                originalException);
    }

    static Swagger parseInputStream(
            SwaggerParser swaggerParser, ObjectMapper mapper, InputStream stream, boolean resolveReferences)
            throws IOException {
        final JsonNode node = mapper.readTree(stream);

        return swaggerParser.read(node, resolveReferences);
    }

    static String pickBestScheme(final String specificationScheme, final List schemes) {
        if (schemes != null && !schemes.isEmpty()) {
            if (schemes.contains(Scheme.HTTPS)) {
                return Scheme.HTTPS.toValue();
            }

            if (schemes.contains(Scheme.HTTP)) {
                return Scheme.HTTP.toValue();
            }
        }

        if (specificationScheme != null && (Scheme.HTTP.toValue().contains(specificationScheme)
                || Scheme.HTTPS.toValue().contains(specificationScheme))) {
            return specificationScheme;
        }

        // there is no support for WebSocket (Scheme.WS, Scheme.WSS)

        return null;
    }

    static String queryParameterExpression(final Parameter parameter) {
        final String name = parameter.getName();

        final StringBuilder expression = new StringBuilder(name).append("={").append(name);
        if (!parameter.getRequired()) {
            expression.append('?');
        }
        expression.append('}');

        return expression.toString();
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy