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

com.palantir.conjure.java.undertow.runtime.ConjureBodySerDe Maven / Gradle / Ivy

/*
 * (c) Copyright 2019 Palantir Technologies Inc. All rights reserved.
 *
 * 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 com.palantir.conjure.java.undertow.runtime;

import com.google.common.base.Splitter;
import com.google.common.collect.ImmutableList;
import com.palantir.conjure.java.undertow.lib.BinaryResponseBody;
import com.palantir.conjure.java.undertow.lib.BodySerDe;
import com.palantir.conjure.java.undertow.lib.Deserializer;
import com.palantir.conjure.java.undertow.lib.Endpoint;
import com.palantir.conjure.java.undertow.lib.Serializer;
import com.palantir.conjure.java.undertow.lib.TypeMarker;
import com.palantir.logsafe.Preconditions;
import com.palantir.logsafe.SafeArg;
import com.palantir.logsafe.exceptions.SafeIllegalArgumentException;
import com.palantir.logsafe.logger.SafeLogger;
import com.palantir.logsafe.logger.SafeLoggerFactory;
import com.palantir.tracing.CloseableTracer;
import com.palantir.tracing.TagTranslator;
import com.palantir.tracing.Tracer;
import io.undertow.server.HttpServerExchange;
import io.undertow.util.HeaderValues;
import io.undertow.util.Headers;
import io.undertow.util.Protocols;
import java.io.IOException;
import java.io.InputStream;
import java.io.PushbackInputStream;
import java.util.List;
import java.util.Optional;
import org.xnio.IoUtils;

/** Package private internal API. */
final class ConjureBodySerDe implements BodySerDe {

    private static final SafeLogger log = SafeLoggerFactory.get(ConjureBodySerDe.class);
    private static final String BINARY_CONTENT_TYPE = "application/octet-stream";
    private static final Splitter ACCEPT_VALUE_SPLITTER =
            Splitter.on(',').trimResults().omitEmptyStrings();

    private final List encodings;

    /**
     * Selects the first (based on input order) of the provided encodings that
     * {@link Encoding#supportsContentType supports} the serialization format {@link Headers#ACCEPT accepted} by a given
     * request, or the first serializer if no such serializer can be found.
     */
    ConjureBodySerDe(List encodings) {
        // Defensive copy
        this.encodings =
                encodings.stream().map(LazilyInitializedEncoding::new).collect(ImmutableList.toImmutableList());
        Preconditions.checkArgument(encodings.size() > 0, "At least one Encoding is required");
    }

    @Override
    public  Serializer serializer(TypeMarker token) {
        return new EncodingSerializerRegistry<>(encodings, token, Optional.empty());
    }

    @Override
    public  Serializer serializer(TypeMarker token, Endpoint endpoint) {
        return new EncodingSerializerRegistry<>(encodings, token, Optional.of(endpoint));
    }

    @Override
    public  Deserializer deserializer(TypeMarker token) {
        return new EncodingDeserializerRegistry<>(encodings, token, Optional.empty());
    }

    @Override
    public  Deserializer deserializer(TypeMarker token, Endpoint endpoint) {
        return new EncodingDeserializerRegistry<>(encodings, token, Optional.of(endpoint));
    }

    @Override
    public void serialize(BinaryResponseBody value, HttpServerExchange exchange) throws IOException {
        Preconditions.checkNotNull(value, "A BinaryResponseBody value is required");
        exchange.getResponseHeaders().put(Headers.CONTENT_TYPE, BINARY_CONTENT_TYPE);
        Tracer.fastStartSpan(TracedEncoding.SERIALIZE_OPERATION);
        try {
            // We wrap the provided outputstream to prevent premature closure, otherwise a common pattern of
            // using try-with-resources structures within BinaryResponseBody prevents failures from being
            // understood by clients. try-with-resource ends up calling OutputStream.close before the
            // exception handler is invoked, which tells the server (and then client) that the response bytes
            // have been fully sent successfully.
            value.write(UnclosableOutputStreams.wrap(exchange.getOutputStream()));
        } finally {
            Tracer.fastCompleteSpan(SerializeBinaryTagTranslator.INSTANCE, SerializeBinaryTagTranslator.INSTANCE);
        }
    }

    @Override
    public InputStream deserializeInputStream(HttpServerExchange exchange) {
        String contentType = getContentType(exchange);
        // Compare using 'String#regionMatches' to avoid allocation
        if (contentType.length() < BINARY_CONTENT_TYPE.length()
                || !contentType.regionMatches(
                        /* ignoreCase = */ true, 0, BINARY_CONTENT_TYPE, 0, BINARY_CONTENT_TYPE.length())) {
            throw FrameworkException.unsupportedMediaType(
                    "Unsupported Content-Type", SafeArg.of("Content-Type", contentType));
        }
        return exchange.getInputStream();
    }

    private static final class EncodingSerializerRegistry implements Serializer {

        private final EncodingSerializerContainer defaultEncoding;
        private final List> encodings;

        EncodingSerializerRegistry(List encodings, TypeMarker token, Optional endpoint) {
            this.encodings = encodings.stream()
                    .map(encoding -> new EncodingSerializerContainer<>(encoding, token, endpoint))
                    .collect(ImmutableList.toImmutableList());
            this.defaultEncoding = this.encodings.get(0);
        }

        @Override
        public void serialize(T value, HttpServerExchange exchange) throws IOException {
            Preconditions.checkNotNull(value, "cannot serialize null value");
            safelyDrainRequestBody(exchange);
            EncodingSerializerContainer container = getResponseSerializer(exchange);
            exchange.getResponseHeaders().put(Headers.CONTENT_TYPE, container.encoding.getContentType());
            container.serializer.serialize(value, exchange.getOutputStream());
        }

        /** Returns the {@link EncodingSerializerContainer} to use for the exchange response. */
        @SuppressWarnings("ForLoopReplaceableByForEach") // performance sensitive code avoids iterator allocation
        EncodingSerializerContainer getResponseSerializer(HttpServerExchange exchange) {
            HeaderValues acceptValues = exchange.getRequestHeaders().get(Headers.ACCEPT);
            if (acceptValues != null) {
                // This implementation prefers the client "Accept" order
                for (int i = 0; i < acceptValues.size(); i++) {
                    for (String acceptValue : ACCEPT_VALUE_SPLITTER.split(acceptValues.get(i))) {
                        for (int j = 0; j < encodings.size(); j++) {
                            EncodingSerializerContainer container = encodings.get(j);
                            if (container.encoding.supportsContentType(acceptValue)) {
                                return container;
                            }
                        }
                    }
                }
            }
            // Fall back to the default
            return defaultEncoding;
        }
    }

    private static final class EncodingSerializerContainer {

        private final Encoding encoding;
        private final Encoding.Serializer serializer;

        EncodingSerializerContainer(Encoding encoding, TypeMarker token, Optional endpoint) {
            this.encoding = encoding;
            this.serializer = endpoint.isPresent()
                    ? TracedEncoding.wrap(encoding).serializer(token, endpoint.get())
                    : TracedEncoding.wrap(encoding).serializer(token);
        }
    }

    private static final class EncodingDeserializerRegistry implements Deserializer {

        private final List> encodings;
        private final boolean optionalType;
        private final TypeMarker marker;

        EncodingDeserializerRegistry(List encodings, TypeMarker token, Optional endpoint) {
            this.encodings = encodings.stream()
                    .map(encoding -> new EncodingDeserializerContainer<>(encoding, token, endpoint))
                    .collect(ImmutableList.toImmutableList());
            this.optionalType = TypeMarkers.isOptional(token);
            this.marker = token;
        }

        @Override
        public T deserialize(HttpServerExchange exchange) throws IOException {
            // If this deserializer is built for an optional root type, Optional, OptionalInt, etc,
            // and the incoming request body might be empty (does not have a content-length greater than zero)
            // we must map from an empty request body to an empty optional.
            // See https://github.com/palantir/conjure/blob/master/docs/spec/wire.md#23-body-parameter
            if (optionalType && maybeEmptyBody(exchange)) {
                return deserializeOptional(exchange);
            }
            return deserializeInternal(exchange, exchange.getInputStream());
        }

        private T deserializeOptional(HttpServerExchange exchange) throws IOException {
            // If the first byte of the request stream is -1 (EOF) we return the empty optional type.
            // We cannot provide the empty stream to jackson because there is no content for jackson
            // to deserialize.
            PushbackInputStream requestStream = new PushbackInputStream(exchange.getInputStream(), 1);
            int firstByte = requestStream.read();
            if (firstByte == -1) {
                return TypeMarkers.getEmptyOptional(marker);
            }
            // Otherwise reset the request stream and deserialize normally.
            requestStream.unread(firstByte);
            return deserializeInternal(exchange, requestStream);
        }

        private T deserializeInternal(HttpServerExchange exchange, InputStream requestStream) throws IOException {
            EncodingDeserializerContainer container = getRequestDeserializer(exchange);
            return container.deserializer.deserialize(requestStream);
        }

        private static boolean maybeEmptyBody(HttpServerExchange exchange) {
            // Content-Length maybe null if "Transfer-Encoding: chunked" is sent with a full body.
            String contentLength = exchange.getRequestHeaders().getFirst(Headers.CONTENT_LENGTH);
            return contentLength == null || "0".equals(contentLength);
        }

        /** Returns the {@link EncodingDeserializerContainer} to use to deserialize the request body. */
        @SuppressWarnings("ForLoopReplaceableByForEach") // performance sensitive code avoids iterator allocation
        EncodingDeserializerContainer getRequestDeserializer(HttpServerExchange exchange) {
            String contentType = getContentType(exchange);
            for (int i = 0; i < encodings.size(); i++) {
                EncodingDeserializerContainer container = encodings.get(i);
                if (container.encoding.supportsContentType(contentType)) {
                    return container;
                }
            }
            throw FrameworkException.unsupportedMediaType(
                    "Unsupported Content-Type", SafeArg.of("Content-Type", contentType));
        }
    }

    private static final class EncodingDeserializerContainer {

        private final Encoding encoding;
        private final Encoding.Deserializer deserializer;

        EncodingDeserializerContainer(Encoding encoding, TypeMarker token, Optional endpoint) {
            this.encoding = encoding;
            this.deserializer = endpoint.isPresent()
                    ? TracedEncoding.wrap(encoding).deserializer(token, endpoint.get())
                    : TracedEncoding.wrap(encoding).deserializer(token);
        }
    }

    /**
     * Ensure the client isn't blocked sending additional data. It's very uncommon for this to be necessary, in most
     * cases exceptional responses are far below the 16k buffer threshold, not even considering socket buffers.
     */
    private static void safelyDrainRequestBody(HttpServerExchange exchange) {
        // No need to impact http/2 which supports out-of-band responses.
        if ((Protocols.HTTP_1_1.equals(exchange.getProtocol()) || Protocols.HTTP_1_0.equals(exchange.getProtocol()))
                && !exchange.isRequestComplete()) {
            try (CloseableTracer ignored = CloseableTracer.startSpan("Undertow: drain request body")) {
                IoUtils.safeClose(exchange.getInputStream());
            }
        }
    }

    private enum SerializeBinaryTagTranslator implements TagTranslator {
        INSTANCE;

        @Override
        public  void translate(TagAdapter adapter, T target, Object _data) {
            adapter.tag(target, "type", "BinaryResponseBody");
            adapter.tag(target, "contentType", BINARY_CONTENT_TYPE);
        }
    }

    /**
     * Gets the request {@code Content-Type} header if exactly one value exists, otherwise logs
     * a warning. This notifies us in the unexpected case when multiple
     * content-type headers are incorrectly sent to the server, it's not clear which should
     * be used.
     * 

* Note that the returned content-type needs to be compared case-insensitive. */ private static String getContentType(HttpServerExchange exchange) { HeaderValues contentTypeValues = exchange.getRequestHeaders().get(Headers.CONTENT_TYPE); if (contentTypeValues == null || contentTypeValues.isEmpty()) { throw new SafeIllegalArgumentException("Request is missing Content-Type header"); } else if (contentTypeValues.size() != 1) { log.warn( "Request has too many Content-Type headers", SafeArg.of("contentTypes", ImmutableList.copyOf(contentTypeValues))); return contentTypeValues.getFirst(); } return contentTypeValues.get(0); } }