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

org.asynchttpclient.shaded.request.body.multipart.MultipartBody Maven / Gradle / Ivy

There is a newer version: 2.12.3.1
Show newest version
/*
 * Copyright (c) 2015 AsyncHttpClient Project. All rights reserved.
 *
 * This program is licensed to you under the Apache License Version 2.0,
 * and you may not use this file except in compliance with the Apache License Version 2.0.
 * You may obtain a copy of the Apache License Version 2.0 at
 *     http://www.apache.org/licenses/LICENSE-2.0.
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the Apache License Version 2.0 is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the Apache License Version 2.0 for the specific language governing permissions and limitations there under.
 */
package org.asynchttpclient.shaded.request.body.multipart;

import org.asynchttpclient.shaded.io.netty.buffer.ByteBuf;
import org.asynchttpclient.shaded.netty.request.body.BodyChunkedInput;
import org.asynchttpclient.shaded.request.body.RandomAccessBody;
import org.asynchttpclient.shaded.request.body.multipart.part.MultipartPart;
import org.asynchttpclient.shaded.request.body.multipart.part.MultipartState;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.IOException;
import java.nio.channels.WritableByteChannel;
import java.util.List;
import java.util.concurrent.atomic.AtomicBoolean;

import static org.asynchttpclient.shaded.util.Assertions.assertNotNull;
import static org.asynchttpclient.shaded.util.MiscUtils.closeSilently;

public class MultipartBody implements RandomAccessBody {

  private final static Logger LOGGER = LoggerFactory.getLogger(MultipartBody.class);

  private final List> parts;
  private final String contentType;
  private final byte[] boundary;
  private final long contentLength;
  private int currentPartIndex;
  private boolean done = false;
  private AtomicBoolean closed = new AtomicBoolean();

  public MultipartBody(List> parts, String contentType, byte[] boundary) {
    this.boundary = boundary;
    this.contentType = contentType;
    this.parts = assertNotNull(parts, "parts");
    this.contentLength = computeContentLength();
  }

  private long computeContentLength() {
    try {
      long total = 0;
      for (MultipartPart part : parts) {
        long l = part.length();
        if (l < 0) {
          return -1;
        }
        total += l;
      }
      return total;
    } catch (Exception e) {
      LOGGER.error("An exception occurred while getting the length of the parts", e);
      return 0L;
    }
  }

  public void close() {
    if (closed.compareAndSet(false, true)) {
      for (MultipartPart part : parts) {
        closeSilently(part);
      }
    }
  }

  public long getContentLength() {
    return contentLength;
  }

  public String getContentType() {
    return contentType;
  }

  public byte[] getBoundary() {
    return boundary;
  }

  // Regular Body API
  public BodyState transferTo(ByteBuf target) throws IOException {

    if (done)
      return BodyState.STOP;

    while (target.isWritable() && !done) {
      MultipartPart currentPart = parts.get(currentPartIndex);
      currentPart.transferTo(target);

      if (currentPart.getState() == MultipartState.DONE) {
        currentPartIndex++;
        if (currentPartIndex == parts.size()) {
          done = true;
        }
      }
    }

    return BodyState.CONTINUE;
  }

  // RandomAccessBody API, suited for HTTP but not for HTTPS (zero-copy)
  @Override
  public long transferTo(WritableByteChannel target) throws IOException {

    if (done)
      return -1L;

    long transferred = 0L;
    boolean slowTarget = false;

    while (transferred < BodyChunkedInput.DEFAULT_CHUNK_SIZE && !done && !slowTarget) {
      MultipartPart currentPart = parts.get(currentPartIndex);
      transferred += currentPart.transferTo(target);
      slowTarget = currentPart.isTargetSlow();

      if (currentPart.getState() == MultipartState.DONE) {
        currentPartIndex++;
        if (currentPartIndex == parts.size()) {
          done = true;
        }
      }
    }

    return transferred;
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy