org.asynchttpclient.shaded.request.body.multipart.MultipartBody Maven / Gradle / Ivy
/*
* 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 extends Part> 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 extends Part> 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 extends Part> 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 extends Part> 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;
}
}