com.ning.http.client.generators.InputStreamBodyGenerator Maven / Gradle / Ivy
/*
* Copyright (c) 2010-2012 Sonatype, Inc. 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 com.ning.http.client.generators;
import com.ning.http.client.Body;
import com.ning.http.client.BodyGenerator;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.IOException;
import java.io.InputStream;
import java.nio.ByteBuffer;
/**
* A {@link BodyGenerator} which use an {@link InputStream} for reading bytes, without having to read the entire
* stream in memory.
*
* NOTE: The {@link InputStream} must support the {@link InputStream#mark} and {@link java.io.InputStream#reset()} operation.
* If not, mechanisms like authentication, redirect, or resumable download will not works.
*/
public class InputStreamBodyGenerator implements BodyGenerator {
private final static byte[] END_PADDING = "\r\n".getBytes();
private final static byte[] ZERO = "0".getBytes();
private final InputStream inputStream;
private final static Logger logger = LoggerFactory.getLogger(InputStreamBodyGenerator.class);
private boolean patchNettyChunkingIssue = false;
public InputStreamBodyGenerator(InputStream inputStream) {
this.inputStream = inputStream;
}
public InputStream getInputStream() {
return inputStream;
}
@Override
public Body createBody() throws IOException {
return new ISBody();
}
protected class ISBody implements Body {
private boolean eof = false;
private int endDataCount = 0;
private byte[] chunk;
@Override
public long getContentLength() {
return -1;
}
@Override
public long read(ByteBuffer buffer) throws IOException {
// To be safe.
chunk = new byte[buffer.remaining() - 10];
int read = -1;
try {
read = inputStream.read(chunk);
} catch (IOException ex) {
logger.warn("Unable to read", ex);
}
if (patchNettyChunkingIssue) {
if (read == -1) {
// Since we are chunked, we must output extra bytes before considering the input stream closed.
// chunking requires to end the chunking:
// - A Terminating chunk of "0\r\n".getBytes(),
// - Then a separate packet of "\r\n".getBytes()
if (!eof) {
endDataCount++;
if (endDataCount == 2)
eof = true;
if (endDataCount == 1)
buffer.put(ZERO);
buffer.put(END_PADDING);
return buffer.position();
} else {
eof = false;
}
return -1;
}
/**
* Netty 3.2.3 doesn't support chunking encoding properly, so we chunk encoding ourself.
*/
buffer.put(Integer.toHexString(read).getBytes());
// Chunking is separated by "\r\n"
buffer.put(END_PADDING);
buffer.put(chunk, 0, read);
// Was missing the final chunk \r\n.
buffer.put(END_PADDING);
} else if (read > 0) {
buffer.put(chunk, 0, read);
}
return read;
}
@Override
public void close() throws IOException {
inputStream.close();
}
}
/**
* HACK: This is required because Netty has issues with chunking.
*
* @param patchNettyChunkingIssue
*/
public void patchNettyChunkingIssue(boolean patchNettyChunkingIssue) {
this.patchNettyChunkingIssue = patchNettyChunkingIssue;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy