com.mongodb.async.client.gridfs.helpers.AsyncStreamHelper Maven / Gradle / Ivy
/*
* Copyright 2015 MongoDB, Inc.
*
* 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.mongodb.async.client.gridfs.helpers;
import com.mongodb.MongoGridFSException;
import com.mongodb.async.SingleResultCallback;
import com.mongodb.async.client.gridfs.AsyncInputStream;
import com.mongodb.async.client.gridfs.AsyncOutputStream;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.ByteBuffer;
import static org.bson.assertions.Assertions.notNull;
/**
* A general helper class that creates {@link AsyncInputStream} or {@link AsyncOutputStream} instances.
*
* Provides support for:
*
* - {@code byte[]} - Converts byte arrays into Async Streams
* - {@link ByteBuffer} - Converts ByteBuffers into Async Streams
* - {@link InputStream} - Converts InputStreams into Async Streams (Note: InputStream implementations are blocking)
* - {@link OutputStream} - Converts OutputStreams into Async Streams (Note: OutputStream implementations are blocking)
*
*
* @since 3.3
*/
public final class AsyncStreamHelper {
/**
* Converts a {@code byte[]} into a {@link AsyncInputStream}
*
* @param srcBytes the data source
* @return the AsyncInputStream
*/
public static AsyncInputStream toAsyncInputStream(final byte[] srcBytes) {
return toAsyncInputStream(ByteBuffer.wrap(srcBytes));
}
/**
* Converts a {@code byte[]} into a {@link AsyncOutputStream}
*
* @param dstBytes the data destination
* @return the AsyncOutputStream
*/
public static AsyncOutputStream toAsyncOutputStream(final byte[] dstBytes) {
return toAsyncOutputStream(ByteBuffer.wrap(dstBytes));
}
/**
* Converts a {@link ByteBuffer} into a {@link AsyncInputStream}
*
* @param srcByteBuffer the data source
* @return the AsyncInputStream
*/
public static AsyncInputStream toAsyncInputStream(final ByteBuffer srcByteBuffer) {
notNull("srcByteBuffer", srcByteBuffer);
return new AsyncInputStream() {
@Override
public void read(final ByteBuffer dstByteBuffer, final SingleResultCallback callback) {
transferDataFromByteBuffers(srcByteBuffer, dstByteBuffer, callback);
}
@Override
public void close(final SingleResultCallback callback) {
callback.onResult(null, null);
}
};
}
/**
* Converts a {@link ByteBuffer} into a {@link AsyncOutputStream}
*
* @param dstByteBuffer the data destination
* @return the AsyncOutputStream
*/
public static AsyncOutputStream toAsyncOutputStream(final ByteBuffer dstByteBuffer) {
notNull("dstByteBuffer", dstByteBuffer);
return new AsyncOutputStream() {
@Override
public void write(final ByteBuffer srcByteBuffer, final SingleResultCallback callback) {
transferDataFromByteBuffers(srcByteBuffer, dstByteBuffer, callback);
}
@Override
public void close(final SingleResultCallback callback) {
callback.onResult(null, null);
}
};
}
/**
* Converts a {@link InputStream} into a {@link AsyncInputStream}
*
* @param inputStream the InputStream
* @return the AsyncInputStream
*/
public static AsyncInputStream toAsyncInputStream(final InputStream inputStream) {
notNull("inputStream", inputStream);
return new AsyncInputStream() {
@Override
public void read(final ByteBuffer dstByteBuffer, final SingleResultCallback callback) {
notNull("dst", dstByteBuffer);
notNull("callback", callback);
if (!dstByteBuffer.hasRemaining()) {
callback.onResult(-1, null);
return;
}
int maxAmount = dstByteBuffer.remaining();
byte[] bytes = new byte[maxAmount];
int amountRead;
try {
amountRead = inputStream.read(bytes);
} catch (Throwable t) {
callback.onResult(null, new MongoGridFSException("Error reading from input stream", t));
return;
}
if (amountRead > 0) {
if (amountRead < maxAmount) {
byte[] dataRead = new byte[amountRead];
System.arraycopy(bytes, 0, dataRead, 0, amountRead);
dstByteBuffer.put(dataRead);
} else {
dstByteBuffer.put(bytes);
}
}
callback.onResult(amountRead, null);
}
@Override
public void close(final SingleResultCallback callback) {
try {
inputStream.close();
} catch (Throwable t) {
callback.onResult(null, new MongoGridFSException("Error closing input stream", t));
return;
}
callback.onResult(null, null);
}
};
}
/**
* Converts a {@link OutputStream} into a {@link AsyncOutputStream}
*
* @param outputStream the OutputStream
* @return the AsyncOutputStream
*/
public static AsyncOutputStream toAsyncOutputStream(final OutputStream outputStream) {
notNull("outputStream", outputStream);
return new AsyncOutputStream() {
@Override
public void write(final ByteBuffer srcByteBuffer, final SingleResultCallback callback) {
notNull("src", srcByteBuffer);
notNull("callback", callback);
if (!srcByteBuffer.hasRemaining()) {
callback.onResult(-1, null);
return;
}
int amount = srcByteBuffer.remaining();
byte[] bytes = new byte[amount];
try {
srcByteBuffer.get(bytes);
outputStream.write(bytes);
} catch (Throwable t) {
callback.onResult(null, new MongoGridFSException("Error reading from input stream", t));
return;
}
callback.onResult(amount, null);
}
@Override
public void close(final SingleResultCallback callback) {
try {
outputStream.close();
} catch (Throwable t) {
callback.onResult(null, new MongoGridFSException("Error closing from output stream", t));
return;
}
callback.onResult(null, null);
}
};
}
private static void transferDataFromByteBuffers(final ByteBuffer srcByteBuffer, final ByteBuffer dstByteBuffer,
final SingleResultCallback callback) {
if (!srcByteBuffer.hasRemaining()) {
callback.onResult(-1, null);
return;
}
int transferAmount = Math.min(dstByteBuffer.remaining(), srcByteBuffer.remaining());
if (transferAmount > 0) {
ByteBuffer temp = srcByteBuffer.duplicate();
temp.limit(temp.position() + transferAmount);
dstByteBuffer.put(temp);
srcByteBuffer.position(srcByteBuffer.position() + transferAmount);
}
callback.onResult(transferAmount, null);
}
private AsyncStreamHelper() {
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy