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

io.swagger.inflector.controllers.SwaggerOperationController Maven / Gradle / Ivy

There is a newer version: 2.0.12
Show newest version
/*
 *  Copyright 2017 SmartBear Software
 *
 *  Licensed under the Apache License, Version 2.0 (the "License");
 *  you may not use this file except in compliance with the License.
 *  You may obtain a copy of the License at
 *
 *  http://www.apache.org/licenses/LICENSE-2.0
 *
 *  Unless required by applicable law or agreed to in writing, software
 *  distributed under the License is distributed on an "AS IS" BASIS,
 *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 *  See the License for the specific language governing permissions and
 *  limitations under the License.
 */

package io.swagger.inflector.controllers;

import com.fasterxml.jackson.databind.JavaType;
import com.google.common.io.Files;
import io.swagger.inflector.config.Configuration;
import io.swagger.inflector.config.ControllerFactory;
import io.swagger.inflector.converters.ConversionException;
import io.swagger.inflector.converters.InputConverter;
import io.swagger.inflector.examples.ExampleBuilder;
import io.swagger.inflector.examples.models.ArrayExample;
import io.swagger.inflector.examples.models.Example;
import io.swagger.inflector.examples.models.ObjectExample;
import io.swagger.inflector.models.ApiError;
import io.swagger.inflector.models.RequestContext;
import io.swagger.inflector.models.ResponseContext;
import io.swagger.inflector.processors.EntityProcessor;
import io.swagger.inflector.processors.EntityProcessorFactory;
import io.swagger.inflector.schema.SchemaValidator;
import io.swagger.inflector.utils.ApiErrorUtils;
import io.swagger.inflector.utils.ApiException;
import io.swagger.inflector.utils.ContentTypeSelector;
import io.swagger.inflector.utils.ReflectionUtils;
import io.swagger.inflector.validators.ValidationException;
import io.swagger.inflector.validators.ValidationMessage;
import io.swagger.models.Model;
import io.swagger.models.Operation;
import io.swagger.models.parameters.BodyParameter;
import io.swagger.models.parameters.FormParameter;
import io.swagger.models.parameters.Parameter;
import io.swagger.models.parameters.SerializableParameter;
import io.swagger.models.properties.Property;
import io.swagger.util.Json;
import org.apache.commons.csv.CSVFormat;
import org.apache.commons.csv.CSVRecord;
import org.apache.commons.fileupload.MultipartStream;
import org.apache.commons.io.IOUtils;
import org.apache.commons.lang.StringUtils;
import org.glassfish.jersey.process.Inflector;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import javax.inject.Inject;
import javax.inject.Provider;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.ws.rs.container.ContainerRequestContext;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;
import javax.ws.rs.core.Response.ResponseBuilder;
import javax.ws.rs.core.UriInfo;
import javax.ws.rs.ext.ContextResolver;
import javax.ws.rs.ext.Providers;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.StringReader;
import java.io.UnsupportedEncodingException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.net.URLDecoder;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;


public class SwaggerOperationController extends ReflectionUtils implements Inflector {
    private static final Logger LOGGER = LoggerFactory.getLogger(SwaggerOperationController.class);

    private static Set commonHeaders = new HashSet();

    static {
        commonHeaders.add("Host");
        commonHeaders.add("User-Agent");
        commonHeaders.add("Accept");
        commonHeaders.add("Content-Type");
        commonHeaders.add("Content-Length");
    }

    private String path;
    private String httpMethod;
    private Operation operation;
    private Object controller = null;
    private Method method = null;
    private JavaType[] parameterClasses = null;
    private Map definitions;
    private InputConverter validator;
    private String controllerName;
    private String methodName;
    private String operationSignature;
    @Inject
    private Provider providersProvider;
    @Inject
    private Provider requestProvider;
    @Inject
    private Provider responseProvider;
    private ControllerFactory controllerFactoryCache = null;

    public SwaggerOperationController(Configuration config, String path, String httpMethod, Operation operation,
                                      Map definitions) {
        this.setConfiguration(config);
        this.path = path;
        this.httpMethod = httpMethod;
        this.operation = operation;
        this.definitions = definitions;
        this.validator = InputConverter.getInstance();
        this.method = detectMethod(operation);
        if (method == null) {
            LOGGER.debug("no method `" + methodName + "` in `" + controllerName + "` to map to, using mock response");
        }
    }

    //Used for unit testing only
    SwaggerOperationController(Configuration config, String path, String httpMethod, Operation operation,
                                      Map definitions, Provider requestProvider,
                                      Provider responseProvider) {
        this(config, path, httpMethod, operation, definitions);
        this.requestProvider = requestProvider;
        this.responseProvider = responseProvider;
    }

    public Method detectMethod(Operation operation) {
        controllerName = getControllerName(operation);
        methodName = getMethodName(path, httpMethod, operation);
        JavaType[] args = getOperationParameterClasses(operation, this.definitions);

        StringBuilder builder = new StringBuilder();

        builder.append(getMethodName(path, httpMethod, operation))
                .append("(");

        for (int i = 0; i < args.length; i++) {
            if (i == 0) {
                builder.append(RequestContext.class.getCanonicalName()).append(" request");
            } else {
                builder.append(", ");
                if (args[i] == null) {
                    LOGGER.error("didn't expect a null class for " + operation.getParameters().get(i - 1).getName());
                } else if (args[i].getRawClass() != null) {
                    String className = args[i].getRawClass().getName();
                    if (className.startsWith("java.lang.")) {
                        className = className.substring("java.lang.".length());
                    }
                    builder.append(className);
                    builder.append(" ").append(operation.getParameters().get(i - 1).getName());
                }
            }
        }
        builder.append(")");

        operationSignature = "public io.swagger.inflector.models.ResponseContext " + builder.toString();

        LOGGER.info("looking for method: `" + operationSignature + "` in class `" + controllerName + "`");
        this.parameterClasses = args;

        if (controllerName != null && methodName != null) {
            try {
                Class cls;
                try {
                    cls = Class.forName(controllerName);
                } catch (ClassNotFoundException e) {
                    controllerName = controllerName + "Controller";
                    cls = Class.forName(controllerName);
                }

                Method[] methods = cls.getMethods();
                for (Method method : methods) {
                    if (methodName.equals(method.getName())) {
                        Class[] methodArgs = method.getParameterTypes();
                        if (methodArgs.length == args.length) {
                            int i = 0;
                            boolean matched = true;
                            if (!args[i].getRawClass().equals(methodArgs[i])) {
                                LOGGER.debug("failed to match " + args[i] + ", " + methodArgs[i]);
                                matched = false;
                            }
                            if (matched) {
                                this.parameterClasses = args;
                                this.controller = getControllerFactory().instantiateController(cls, operation);
                                LOGGER.debug("found class `" + controllerName + "`");
                                return method;
                            }
                        }
                    }
                }
            } catch (ClassNotFoundException e) {
                LOGGER.debug("didn't find class " + controller);
            } catch (IllegalAccessException e) {
                e.printStackTrace();
            } catch (InstantiationException e) {
                e.printStackTrace();
            }
        }
        return null;
    }

    @Override
    public Response apply(ContainerRequestContext ctx) {
        List parameters = operation.getParameters();
        final RequestContext requestContext = createContext(ctx);

        String path = ctx.getUriInfo().getPath();
        Map> formMap = new HashMap>();
        Map inputStreams = new HashMap();

        Object[] args = new Object[parameters.size() + 1];
        if (parameters != null) {
            int i = 0;

            args[i] = requestContext;
            i += 1;
            List missingParams = new ArrayList();
            UriInfo uri = ctx.getUriInfo();
            String formDataString = null;
            String[] parts = null;
            Set existingKeys = new HashSet();

            for (Iterator x = uri.getQueryParameters().keySet().iterator(); x.hasNext(); ) {
                existingKeys.add(x.next() + ": qp");
            }
            for (Iterator x = uri.getPathParameters().keySet().iterator(); x.hasNext(); ) {
                existingKeys.add(x.next() + ": pp");
            }
            for (Iterator x = ctx.getHeaders().keySet().iterator(); x.hasNext(); ) {
                String key = x.next();
//              if(!commonHeaders.contains(key))
//                existingKeys.add(key);
            }
            MediaType mt = requestContext.getMediaType();

            for (Parameter p : parameters) {
                Map headers = new HashMap();
                String name = null;

                if (p instanceof FormParameter) {
                    if (formDataString == null) {
                        // can only read stream once
                        if (mt.isCompatible(MediaType.MULTIPART_FORM_DATA_TYPE)) {
                            // get the boundary
                            String boundary = mt.getParameters().get("boundary");

                            if (boundary != null) {
                                try {
                                    InputStream output = ctx.getEntityStream();

                                    MultipartStream multipartStream = new MultipartStream(output, boundary.getBytes());
                                    boolean nextPart = multipartStream.skipPreamble();
                                    while (nextPart) {
                                        String header = multipartStream.readHeaders();
                                        // process headers
                                        if (header != null) {
                                            CSVFormat format = CSVFormat.DEFAULT
                                                    .withDelimiter(';')
                                                    .withRecordSeparator("=");

                                            Iterable records = format.parse(new StringReader(header));
                                            for (CSVRecord r : records) {
                                                for (int j = 0; j < r.size(); j++) {
                                                    String string = r.get(j);

                                                    Iterable outerString = CSVFormat.DEFAULT
                                                            .withDelimiter('=')
                                                            .parse(new StringReader(string));
                                                    for (CSVRecord outerKvPair : outerString) {
                                                        if (outerKvPair.size() == 2) {
                                                            String key = outerKvPair.get(0).trim();
                                                            String value = outerKvPair.get(1).trim();
                                                            if ("name".equals(key)) {
                                                                name = value;
                                                            }
                                                            headers.put(key, value);
                                                        } else {
                                                            Iterable innerString = CSVFormat.DEFAULT
                                                                    .withDelimiter(':')
                                                                    .parse(new StringReader(string));
                                                            for (CSVRecord innerKVPair : innerString) {
                                                                if (innerKVPair.size() == 2) {
                                                                    String key = innerKVPair.get(0).trim();
                                                                    String value = innerKVPair.get(1).trim();
                                                                    if ("name".equals(key)) {
                                                                        name = value;
                                                                    }
                                                                    headers.put(key, value);
                                                                }
                                                            }
                                                        }
                                                    }
                                                    if (name != null) {
                                                        formMap.put(name, headers);
                                                    }
                                                }
                                            }
                                        }
                                        String filename = extractFilenameFromHeaders( headers ) ;
                                        if (filename != null) {
                                            try {
                                                File file = new File(Files.createTempDir(), filename);
                                                file.deleteOnExit();
                                                file.getParentFile().deleteOnExit();
                                                FileOutputStream fo = new FileOutputStream(file);
                                                multipartStream.readBodyData(fo);
                                                inputStreams.put(name, file);
                                            }
                                            catch( Exception e){
                                                LOGGER.error("Failed to extract uploaded file", e );
                                            }
                                        } else {
                                            ByteArrayOutputStream bo = new ByteArrayOutputStream();
                                            multipartStream.readBodyData(bo);
                                            String value = bo.toString();
                                            headers.put(name, value);
                                        }
                                        if(name != null) {
                                            formMap.put(name, headers);
                                        }
                                        headers = new HashMap<>();
                                        name = null;
                                        nextPart = multipartStream.readBoundary();
                                    }
                                } catch (IOException e) {
                                    e.printStackTrace();
                                }
                            }
                        } else {
                            try {
                                formDataString = IOUtils.toString(ctx.getEntityStream(), "UTF-8");
                                parts = formDataString.split("&");

                                for (String part : parts) {
                                    String[] kv = part.split("=");
                                    existingKeys.add(kv[0] + ": fp");
                                }
                            } catch (IOException e) {
                                e.printStackTrace();
                            }
                        }
                    }
                }
            }
            for (Parameter parameter : parameters) {
                String in = parameter.getIn();
                Object o = null;

                try {
                    if ("formData".equals(in)) {
                        SerializableParameter sp = (SerializableParameter) parameter;
                        String name = parameter.getName();
                        if (mt.isCompatible(MediaType.MULTIPART_FORM_DATA_TYPE)) {
                            // look in the form map
                            Map headers = formMap.get(name);
                            if (headers != null && headers.size() > 0) {
                                if ("file".equals(sp.getType())) {
                                    o = inputStreams.get(name);
                                } else {
                                    Object obj = headers.get(parameter.getName());
                                    if (obj != null) {
                                        JavaType jt = parameterClasses[i];
                                        Class cls = jt.getRawClass();

                                        List os = Arrays.asList(obj.toString());
                                        try {
                                            o = validator.convertAndValidate(os, parameter, cls, definitions);
                                        } catch (ConversionException e) {
                                            missingParams.add(e.getError());
                                        } catch (ValidationException e) {
                                            missingParams.add(e.getValidationMessage());
                                        }
                                    }
                                }
                            }
                        } else {
                            if (formDataString != null) {
                                for (String part : parts) {
                                    String[] kv = part.split("=");
                                    if (kv != null) {
                                        if (kv.length > 0) {
                                            existingKeys.remove(kv[0] + ": fp");
                                        }
                                        if (kv.length == 2) {
                                            // TODO how to handle arrays here?
                                            String key = kv[0];
                                            try {
                                                String value = URLDecoder.decode(kv[1], "utf-8");
                                                if (parameter.getName().equals(key)) {
                                                    JavaType jt = parameterClasses[i];
                                                    Class cls = jt.getRawClass();
                                                    try {
                                                        o = validator.convertAndValidate(Arrays.asList(value), parameter, cls, definitions);
                                                    } catch (ConversionException e) {
                                                        missingParams.add(e.getError());
                                                    } catch (ValidationException e) {
                                                        missingParams.add(e.getValidationMessage());
                                                    }
                                                }
                                            } catch (UnsupportedEncodingException e) {
                                                LOGGER.error("unable to decode value for " + key);
                                            }
                                        }
                                    }
                                }
                            }
                        }
                    } else {
                        try {
                            String paramName = parameter.getName();
                            if ("query".equals(in)) {
                                existingKeys.remove(paramName + ": qp");
                            }
                            if ("path".equals(in)) {
                                existingKeys.remove(paramName + ": pp");
                            }
                            JavaType jt = parameterClasses[i];
                            Class cls = jt.getRawClass();
                            if ("body".equals(in)) {
                                if (ctx.hasEntity()) {
                                    BodyParameter body = (BodyParameter) parameter;
                                    o = EntityProcessorFactory.readValue(ctx.getMediaType(), ctx.getEntityStream(), cls);
                                    if (o != null) {
                                        validate(o, body.getSchema(), SchemaValidator.Direction.INPUT);
                                    }
                                } else if (parameter.getRequired()) {
                                    ValidationException e = new ValidationException();
                                    e.message(new ValidationMessage()
                                            .message("The input body `" + paramName + "` is required"));
                                    throw e;
                                }
                            }
                            if ("query".equals(in)) {
                                o = validator.convertAndValidate(uri.getQueryParameters().get(parameter.getName()), parameter, cls, definitions);
                            } else if ("path".equals(in)) {
                                o = validator.convertAndValidate(uri.getPathParameters().get(parameter.getName()), parameter, cls, definitions);
                            } else if ("header".equals(in)) {
                                o = validator.convertAndValidate(ctx.getHeaders().get(parameter.getName()), parameter, cls, definitions);
                            }
                        } catch (ConversionException e) {
                            missingParams.add(e.getError());
                        } catch (ValidationException e) {
                            missingParams.add(e.getValidationMessage());
                        }
                    }
                } catch (NumberFormatException e) {
                    LOGGER.error("Couldn't find " + parameter.getName() + " (" + in + ") to " + parameterClasses[i], e);
                }

                args[i] = o;
                i += 1;
            }
            if (existingKeys.size() > 0) {
                LOGGER.debug("unexpected keys: " + existingKeys);
            }
            if (missingParams.size() > 0) {
                StringBuilder builder = new StringBuilder();
                builder.append("Input error");
                if (missingParams.size() > 1) {
                    builder.append("s");
                }
                builder.append(": ");
                int count = 0;
                for (ValidationMessage message : missingParams) {
                    if (count > 0) {
                        builder.append(", ");
                    }
                    if (message != null && message.getMessage() != null) {
                        builder.append(message.getMessage());
                    } else {
                        builder.append("no additional input");
                    }
                    count += 1;
                }
                int statusCode = config.getInvalidRequestStatusCode();
                ApiError error = new ApiError()
                        .code(statusCode)
                        .message(builder.toString());
                throw new ApiException(error);
            }
        }
        try {
            if (method != null) {
                LOGGER.info("calling method " + method + " on controller " + this.controller + " with args " + Arrays.toString(args));
                try {
                    Object response = method.invoke(controller, args);
                    if (response instanceof ResponseContext) {
                        ResponseContext wrapper = (ResponseContext) response;
                        ResponseBuilder builder = Response.status(wrapper.getStatus());

                        // response headers
                        for (String key : wrapper.getHeaders().keySet()) {
                            List v = wrapper.getHeaders().get(key);
                            if (v.size() == 1) {
                                builder.header(key, v.get(0));
                            } else {
                                builder.header(key, v);
                            }
                        }

                        // entity
                        if (wrapper.getEntity() != null) {
                            builder.entity(wrapper.getEntity());
                            // content type
                            if (wrapper.getContentType() != null) {
                                builder.type(wrapper.getContentType());
                            } else {
                                final ContextResolver selector = providersProvider
                                        .get().getContextResolver(ContentTypeSelector.class,
                                                MediaType.WILDCARD_TYPE);
                                if (selector != null) {
                                    selector.getContext(getClass()).apply(ctx.getAcceptableMediaTypes(),
                                            builder);
                                }
                            }

                            if (operation.getResponses() != null) {
                                String responseCode = String.valueOf(wrapper.getStatus());
                                io.swagger.models.Response responseSchema = operation.getResponses().get(responseCode);
                                if (responseSchema == null) {
                                    // try default response schema
                                    responseSchema = operation.getResponses().get("default");
                                }
                                if (responseSchema != null && responseSchema.getResponseSchema() != null) {
                                    validate(wrapper.getEntity(), responseSchema.getResponseSchema(), SchemaValidator.Direction.OUTPUT);
                                } else {
                                    LOGGER.debug("no response schema for code " + responseCode + " to validate against");
                                }
                            }
                        }

                        return builder.build();
                    }
                    return Response.ok().entity(response).build();
                } catch (IllegalArgumentException | IllegalAccessException | InvocationTargetException e) {
                    for (Throwable cause = e.getCause(); cause != null; ) {
                        if (cause instanceof ApiException) {
                            throw (ApiException) cause;
                        }
                        final Throwable next = cause.getCause();
                        cause = next == cause || next == null ? null : next;
                    }
                    throw new ApiException(ApiErrorUtils.createInternalError(), e);
                }
            }
            Map responses = operation.getResponses();
            if (responses != null) {
                String[] keys = new String[responses.keySet().size()];
                Arrays.sort(responses.keySet().toArray(keys));
                int code = 0;
                String defaultKey = null;
                for (String key : keys) {
                    if (key.startsWith("2")) {
                        defaultKey = key;
                        code = Integer.parseInt(key);
                        break;
                    }
                    if ("default".equals(key)) {
                        defaultKey = key;
                        code = 200;
                        break;
                    }
                    if (key.startsWith("3")) {
                        // we use the 3xx responses as defaults
                        defaultKey = key;
                        code = Integer.parseInt(key);
                    }
                }

                if(defaultKey != null) {
                    ResponseBuilder builder = Response.status(code);
                    io.swagger.models.Response response = responses.get(defaultKey);

                    if(response.getHeaders() != null && response.getHeaders().size() > 0) {
                        for(String key: response.getHeaders().keySet()) {
                            Property headerProperty = response.getHeaders().get(key);
                            Object output = ExampleBuilder.fromProperty(headerProperty, definitions);
                            if(output instanceof ArrayExample) {
                                output = ((ArrayExample)output).asString();
                            }
                            else if(output instanceof ObjectExample) {
                                LOGGER.debug("not serializing output example, only primitives or arrays of primitives are supported");
                            }
                            else {
                                output = ((Example)output).asString();
                            }
                            builder.header(key, output);
                        }
                    }

                    Map examples = response.getExamples();
                    if (examples != null) {
                        for (MediaType mediaType : requestContext.getAcceptableMediaTypes()) {
                            for (String key : examples.keySet()) {
                                if (MediaType.valueOf(key).isCompatible(mediaType)) {
                                    builder.entity(examples.get(key))
                                            .type(mediaType);

                                    return builder.build();
                                }
                            }
                        }
                    }

                    Object output = ExampleBuilder.fromProperty(response.getSchema(), definitions);

                    if (output != null) {
                        ResponseContext resp = new ResponseContext().entity(output);
                        setContentType(requestContext, resp, operation);
                        builder.entity(output);
                        if (resp.getContentType() != null) {
                            // this comes from the operation itself
                            builder.type(resp.getContentType());
                        }
                        else {
                            // get acceptable content types
                            List processors = EntityProcessorFactory.getProcessors();

                            MediaType responseMediaType = null;

                            // take first compatible one
                            for (EntityProcessor processor : processors) {
                                if(responseMediaType != null) {
                                    break;
                                }
                                for (MediaType mt : requestContext.getAcceptableMediaTypes()) {
                                    LOGGER.debug("checking type " + mt.toString() + " against " + processor.getClass().getName());
                                    if (processor.supports(mt)) {
                                        builder.type(mt);
                                        responseMediaType = mt;
                                        break;
                                    }
                                }
                            }

                            if(responseMediaType == null) {
                                // no match based on Accept header, use first processor in list
                                for (EntityProcessor processor : processors) {
                                    List supportedTypes = processor.getSupportedMediaTypes();
                                    if (supportedTypes.size() > 0) {
                                        builder.type(supportedTypes.get(0));
                                        break;
                                    }
                                }
                            }
                        }

                        builder.entity(output);
                    }
                    return builder.build();
                }
                else {
                    LOGGER.debug("no response type to map to, assume 200");
                    code = 200;
                }
                return Response.status(code).build();
            }
            return Response.ok().build();
        } finally {
            for (String key : inputStreams.keySet()) {
                File file = inputStreams.get(key);
                if (file != null) {
                    LOGGER.debug("deleting file " + file.getPath());
                    file.delete();
                }
            }
        }
    }

    static String extractFilenameFromHeaders(Map headers) {
        String filename = headers.get("filename");
        if( StringUtils.isBlank( filename )){
            return null;
        }

        filename = filename.trim();

        int ix = filename.lastIndexOf(File.separatorChar);
        if (ix != -1 ) {
            filename = filename.substring(ix + 1).trim();
            if( StringUtils.isBlank(filename)){
                return null;
            }
        }

        return filename;
    }

    public void validate(Object o, Property property, SchemaValidator.Direction direction) throws ApiException {
        doValidation(o, property, direction);
    }

    public void validate(Object o, Model model, SchemaValidator.Direction direction) throws ApiException {
        doValidation(o, model, direction);
    }

    public void setContentType(RequestContext res, ResponseContext resp, Operation operation) {
        // honor what has been set, it may be determined by business logic in the controller
        if (resp.getContentType() != null) {
            return;
        }
        List available = operation.getProduces();
        if (available != null) {
            for (String a : available) {
                MediaType mt = MediaType.valueOf(a);
                for (MediaType acceptable : res.getAcceptableMediaTypes()) {
                    if (mt.isCompatible(acceptable)) {
                        resp.setContentType(mt);
                        return;
                    }
                }
            }
            if (available.size() > 0) {
                resp.setContentType(MediaType.valueOf(available.get(0)));
            }
        }
    }

    public String getOperationSignature() {
        return operationSignature;
    }

    public void setOperationSignature(String operationSignature) {
        this.operationSignature = operationSignature;
    }

    public String getControllerName() {
        return controllerName;
    }

    public void setControllerName(String controllerName) {
        this.controllerName = controllerName;
    }

    public String getMethodName() {
        return methodName;
    }

    public void setMethodName(String methodName) {
        this.methodName = methodName;
    }

    public Method getMethod() {
        return method;
    }

    public void setMethod(Method method) {
        this.method = method;
    }

    // package protected to facilitate unit testing
    RequestContext createContext(ContainerRequestContext from) {
        HttpServletRequest request = getHttpServletRequest();
        HttpServletResponse response = getHttpServletResponse();
        return new RequestContext(from, request, response);
    }

    private HttpServletRequest getHttpServletRequest() {
        if (requestProvider != null) {
            return requestProvider.get();
        } else {
            LOGGER.warn("HttpServletRequest provider was null - returning null request!");
            return null;
        }
    }

    private HttpServletResponse getHttpServletResponse() {
        if (responseProvider != null) {
            return responseProvider.get();
        } else {
            LOGGER.warn("HttpServletResponse provider was null - returning null response!");
            return null;
        }
    }

    private void doValidation(Object value, Object schema, SchemaValidator.Direction direction) throws ApiException {
        if (config.getValidatePayloads().isEmpty()) {
            return;
        }
        switch (direction) {
            case INPUT:
                if (config.getValidatePayloads().contains(Configuration.Direction.IN)
                        && !SchemaValidator.validate(value, Json.pretty(schema), direction)) {
                    throw new ApiException(new ApiError()
                            .code(config.getInvalidRequestStatusCode())
                            .message("Input does not match the expected structure"));
                }
                break;
            case OUTPUT:
                if (config.getValidatePayloads().contains(Configuration.Direction.OUT)
                        && !SchemaValidator.validate(value, Json.pretty(schema), direction)) {
                    throw new ApiException(new ApiError()
                            .code(config.getInvalidRequestStatusCode())
                            .message("The server generated an invalid response"));
                }
                break;
        }
    }
    
    private ControllerFactory getControllerFactory() {
    	if (controllerFactoryCache == null){
    		controllerFactoryCache = config.getControllerFactory();
    	}
    	return controllerFactoryCache;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy