package io.undertow.websockets.core.function;
import org.xnio.ChannelListener;
import org.xnio.Option;
import org.xnio.XnioExecutor;
import org.xnio.XnioIoThread;
import org.xnio.XnioWorker;
import org.xnio.channels.StreamSinkChannel;
import org.xnio.channels.StreamSourceChannel;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;
import java.util.concurrent.TimeUnit;
public class ChannelFunctionStreamSourceChannel implements StreamSourceChannel {
private final StreamSourceChannel channel;
private final ChannelFunction[] functions;
public ChannelFunctionStreamSourceChannel(StreamSourceChannel channel, ChannelFunction... functions) {
this.channel = channel;
this.functions = functions;
}
@Override
public long transferTo(long position, long count, FileChannel target) throws IOException {
return channel.transferTo(position, count, new ChannelFunctionFileChannel(target, functions));
}
@Override
public long transferTo(long count, ByteBuffer throughBuffer, StreamSinkChannel target) throws IOException {
return target.transferFrom(this, count, throughBuffer);
}
@Override
public ChannelListener.Setter<? extends StreamSourceChannel> getReadSetter() {
return channel.getReadSetter();
}
@Override
public ChannelListener.Setter<? extends StreamSourceChannel> getCloseSetter() {
return channel.getCloseSetter();
}
@Override
public long read(ByteBuffer[] dsts, int offset, int length) throws IOException {
long r = 0;
for (int a = offset; a < length; a++) {
int i = read(dsts[a]);
if (i < 1) {
break;
}
r += i;
}
return r;
}
@Override
public long read(ByteBuffer[] dsts) throws IOException {
long r = 0;
for (ByteBuffer buf: dsts) {
int i = read(buf);
if (i < 1) {
break;
}
r += i;
}
return r;
}
@Override
public void suspendReads() {
channel.suspendReads();
}
@Override
public void resumeReads() {
channel.resumeReads();
}
@Override
public boolean isReadResumed() {
return channel.isReadResumed();
}
@Override
public void wakeupReads() {
channel.wakeupReads();
}
@Override
public void shutdownReads() throws IOException {
channel.shutdownReads();
}
@Override
public void awaitReadable() throws IOException {
channel.awaitReadable();
}
@Override
public void awaitReadable(long time, TimeUnit timeUnit) throws IOException {
channel.awaitReadable(time, timeUnit);
}
@Override
public XnioExecutor getReadThread() {
return channel.getReadThread();
}
@Override
public int read(ByteBuffer dst) throws IOException {
int position = dst.position();
int r = channel.read(dst);
if (r > 0) {
afterReading(dst, position, r);
}
return r;
}
@Override
public XnioWorker getWorker() {
return channel.getWorker();
}
@Override
public XnioIoThread getIoThread() {
return channel.getIoThread();
}
@Override
public boolean supportsOption(Option<?> option) {
return channel.supportsOption(option);
}
@Override
public <T> T getOption(Option<T> option) throws IOException {
return channel.getOption(option);
}
@Override
public <T> T setOption(Option<T> option, T value) throws IOException {
return channel.setOption(option, value);
}
private void afterReading(ByteBuffer buffer, int position, int length) throws IOException {
for (ChannelFunction func: functions) {
func.afterRead(buffer, position, length);
}
}
@Override
public boolean isOpen() {
return channel.isOpen();
}
@Override
public void close() throws IOException {
channel.close();
}
}