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

io.micronaut.http.server.netty.FormDataHttpContentProcessor Maven / Gradle / Ivy

/*
 * Copyright 2017-2020 original 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 io.micronaut.http.server.netty;

import io.micronaut.core.annotation.Internal;
import io.micronaut.http.MediaType;
import io.micronaut.http.exceptions.ContentLengthExceededException;
import io.micronaut.http.server.HttpServerConfiguration;
import io.micronaut.http.server.netty.configuration.NettyHttpServerConfiguration;
import io.netty.buffer.ByteBufHolder;
import io.netty.handler.codec.http.HttpContent;
import io.netty.handler.codec.http.HttpRequest;
import io.netty.handler.codec.http.LastHttpContent;
import io.netty.handler.codec.http.multipart.Attribute;
import io.netty.handler.codec.http.multipart.FileUpload;
import io.netty.handler.codec.http.multipart.HttpDataFactory;
import io.netty.handler.codec.http.multipart.HttpPostMultipartRequestDecoder;
import io.netty.handler.codec.http.multipart.HttpPostRequestDecoder;
import io.netty.handler.codec.http.multipart.HttpPostStandardRequestDecoder;
import io.netty.handler.codec.http.multipart.InterfaceHttpData;
import io.netty.handler.codec.http.multipart.InterfaceHttpPostRequestDecoder;
import io.netty.util.ReferenceCountUtil;

import java.io.IOException;
import java.nio.charset.Charset;
import java.util.Collection;
import java.util.concurrent.atomic.AtomicLong;

/**
 * 

Decodes {@link MediaType#MULTIPART_FORM_DATA} in a non-blocking manner.

* *

Designed to be used by a single thread

* * @author Graeme Rocher * @since 1.0 */ @Internal public final class FormDataHttpContentProcessor { protected final NettyHttpRequest nettyHttpRequest; protected final long advertisedLength; protected final long requestMaxSize; protected final AtomicLong receivedLength = new AtomicLong(); protected final NettyHttpServerConfiguration configuration; private final InterfaceHttpPostRequestDecoder decoder; private final long partMaxSize; /** * Set to true to request a destroy by any thread. */ private volatile boolean pleaseDestroy = false; /** * {@code true} during {@link #onData}, can't destroy while that's running. */ private volatile boolean inFlight = false; /** * {@code true} if the decoder has been destroyed or will be destroyed in the near future. */ private boolean destroyed = false; /** * Whether we received a LastHttpContent. */ private boolean receivedLast = false; /** * @param nettyHttpRequest The {@link NettyHttpRequest} * @param configuration The {@link NettyHttpServerConfiguration} */ public FormDataHttpContentProcessor(NettyHttpRequest nettyHttpRequest, NettyHttpServerConfiguration configuration) { this.nettyHttpRequest = nettyHttpRequest; this.advertisedLength = nettyHttpRequest.getContentLength(); this.requestMaxSize = configuration.getMaxRequestSize(); this.configuration = configuration; Charset characterEncoding = nettyHttpRequest.getCharacterEncoding(); HttpServerConfiguration.MultipartConfiguration multipart = configuration.getMultipart(); HttpDataFactory factory = new MicronautHttpData.Factory(multipart, characterEncoding); // prevent the decoders from immediately parsing the content HttpRequest nativeRequest = nettyHttpRequest.toHttpRequestWithoutBody(); if (HttpPostRequestDecoder.isMultipart(nativeRequest)) { this.decoder = new HttpPostMultipartRequestDecoder(factory, nativeRequest, characterEncoding, configuration.getFormMaxFields(), configuration.getFormMaxBufferedBytes()); } else { this.decoder = new HttpPostStandardRequestDecoder(factory, nativeRequest, characterEncoding, configuration.getFormMaxFields(), configuration.getFormMaxBufferedBytes()); } this.partMaxSize = multipart.getMaxFileSize(); } protected void onData(ByteBufHolder message, Collection out) { boolean skip; synchronized (this) { if (destroyed) { skip = true; } else { skip = false; inFlight = true; } } if (skip) { message.release(); return; } try { if (message instanceof HttpContent httpContent) { try { InterfaceHttpPostRequestDecoder postRequestDecoder = this.decoder; postRequestDecoder.offer(httpContent); while (postRequestDecoder.hasNext()) { InterfaceHttpData data = postRequestDecoder.next(); data.touch(); switch (data.getHttpDataType()) { case Attribute -> { Attribute attribute = (Attribute) data; // bodyListHttpData keeps a copy and releases it later out.add(attribute.retain()); postRequestDecoder.removeHttpDataFromClean(attribute); } case FileUpload -> { FileUpload fileUpload = (FileUpload) data; if (fileUpload.isCompleted()) { // bodyListHttpData keeps a copy and releases it later out.add(fileUpload.retain()); postRequestDecoder.removeHttpDataFromClean(fileUpload); } } default -> { // ignore } } } InterfaceHttpData currentPartialHttpData = postRequestDecoder.currentPartialHttpData(); if (currentPartialHttpData != null) { out.add(currentPartialHttpData); postRequestDecoder.removeHttpDataFromClean(currentPartialHttpData); } } catch (HttpPostRequestDecoder.EndOfDataDecoderException e) { // ok, ignore } catch (HttpPostRequestDecoder.ErrorDataDecoderException e) { Throwable cause = e.getCause(); if (cause instanceof IOException && cause.getMessage().equals("Size exceed allowed maximum capacity")) { String partName = decoder.currentPartialHttpData().getName(); throw new ContentLengthExceededException("The part named [" + partName + "] exceeds the maximum allowed content length [" + partMaxSize + "]"); } else { throw e; } } catch (HttpPostRequestDecoder.TooManyFormFieldsException e) { throw new ContentLengthExceededException("Number of form fields exceeds configured limit of [" + configuration.getFormMaxFields() + "]"); } catch (HttpPostRequestDecoder.TooLongFormFieldException e) { throw new ContentLengthExceededException("Length of buffered form field exceeds configured limit of [" + configuration.getFormMaxBufferedBytes() + "]"); } finally { httpContent.release(); } } else { message.release(); } } finally { inFlight = false; destroyIfRequested(); } } public void add(ByteBufHolder message, Collection out) throws Throwable { try { receivedLast |= message instanceof LastHttpContent; long receivedLength1 = this.receivedLength.addAndGet(message.content().readableBytes()); ReferenceCountUtil.touch(message); if (advertisedLength > requestMaxSize) { fireExceedsLength(advertisedLength, requestMaxSize, message); } else if (receivedLength1 > requestMaxSize) { fireExceedsLength(receivedLength1, requestMaxSize, message); } else { onData(message, out); } } catch (Throwable e) { cancel(); throw e; } } public void complete(Collection out) throws Throwable { if (!receivedLast) { add(LastHttpContent.EMPTY_LAST_CONTENT, out); } cancel(); } public void cancel() { pleaseDestroy = true; destroyIfRequested(); } private void destroyIfRequested() { boolean destroy; synchronized (this) { if (pleaseDestroy && !destroyed && !inFlight) { destroy = true; destroyed = true; } else { destroy = false; } } if (destroy) { decoder.destroy(); } } /** * @param receivedLength The length of the content received * @param expected The expected length of the content * @param message The message to release */ protected void fireExceedsLength(long receivedLength, long expected, ByteBufHolder message) { message.release(); throw new ContentLengthExceededException(expected, receivedLength); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy