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

com.firefly.net.tcp.aio.AsynchronousTcpSession Maven / Gradle / Ivy

package com.firefly.net.tcp.aio;

import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.AsynchronousCloseException;
import java.nio.channels.AsynchronousSocketChannel;
import java.nio.channels.ClosedChannelException;
import java.nio.channels.CompletionHandler;
import java.nio.channels.InterruptedByTimeoutException;
import java.util.Collection;
import java.util.Date;
import java.util.LinkedList;
import java.util.Queue;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

import com.firefly.net.BufferSizePredictor;
import com.firefly.net.ByteBufferArrayOutputEntry;
import com.firefly.net.ByteBufferOutputEntry;
import com.firefly.net.Config;
import com.firefly.net.EventManager;
import com.firefly.net.OutputEntry;
import com.firefly.net.Session;
import com.firefly.net.buffer.AdaptiveBufferSizePredictor;
import com.firefly.net.buffer.FileRegion;
import com.firefly.utils.concurrent.Callback;
import com.firefly.utils.concurrent.CountingCallback;
import com.firefly.utils.io.BufferReaderHandler;
import com.firefly.utils.io.BufferUtils;
import com.firefly.utils.log.Log;
import com.firefly.utils.log.LogFactory;
import com.firefly.utils.time.Millisecond100Clock;
import com.firefly.utils.time.SafeSimpleDateFormat;

public class AsynchronousTcpSession implements Session {

	private static Log log = LogFactory.getInstance().getLog("firefly-system");

	private final int sessionId;
	private final long openTime;
	private long closeTime;
	private long lastReadTime;
	private long lastWrittenTime;
	private long readBytes = 0;
	private long writtenBytes = 0;
	private volatile int state;
	private final AsynchronousSocketChannel socketChannel;
	private volatile InetSocketAddress localAddress;
	private volatile InetSocketAddress remoteAddress;

	private final Config config;
	private final EventManager eventManager;
	private volatile Object attachment;

	private final Lock outputLock = new ReentrantLock();
	private boolean isWriting = false;
	private final Queue> outputBuffer = new LinkedList<>();
	private final BufferSizePredictor bufferSizePredictor = new AdaptiveBufferSizePredictor();

	public AsynchronousTcpSession(int sessionId, Config config, EventManager eventManager,
			AsynchronousSocketChannel socketChannel) {
		this.sessionId = sessionId;
		this.openTime = Millisecond100Clock.currentTimeMillis();
		this.config = config;
		this.eventManager = eventManager;
		this.socketChannel = socketChannel;
		state = OPEN;
	}

	void _read() {
		if (!isOpen())
			return;

		final int bufferSize = BufferUtils.normalizeBufferSize(bufferSizePredictor.nextBufferSize());
		final ByteBuffer buf = ByteBuffer.allocate(bufferSize);

		if (log.isDebugEnabled()) {
			log.debug("the session {} buffer size is {}", getSessionId(), bufferSize);
		}
		socketChannel.read(buf, config.getTimeout(), TimeUnit.MILLISECONDS, this,
				new CompletionHandler() {

					@Override
					public void completed(Integer currentReadBytes, AsynchronousTcpSession session) {
						session.lastReadTime = Millisecond100Clock.currentTimeMillis();
						if (currentReadBytes < 0) {
							if (log.isDebugEnabled()) {
								log.debug("the session {} input is closed, {}", session.getSessionId(),
										currentReadBytes);
							}
							session.closeNow();
							return;
						}

						if (log.isDebugEnabled()) {
							log.debug("the session {} read {} bytes", session.getSessionId(), currentReadBytes);
						}
						// Update the predictor.
						session.bufferSizePredictor.previousReceivedBufferSize(currentReadBytes);
						session.readBytes += currentReadBytes;

						buf.flip();
						try {
							config.getDecoder().decode(buf, session);
						} catch (Throwable t) {
							eventManager.executeExceptionTask(session, t);
						} finally {
							_read();
						}
					}

					@Override
					public void failed(Throwable t, AsynchronousTcpSession session) {
						if (t instanceof InterruptedByTimeoutException) {
							if (log.isDebugEnabled()) {
								log.debug("the session {} reading data is timeout.", getSessionId());
							}
						} else {
							log.error("the session {} read data is failed", t, session.getSessionId());
						}

						session.closeNow();
					}
				});
	}

	private void writingFailedCallback(Callback callback, Throwable t) {
		if (t instanceof InterruptedByTimeoutException) {
			if (log.isDebugEnabled()) {
				log.debug("the session {} writing data is timeout.", getSessionId());
			}
		} else {
			log.error("the session {} writes data is failed", t, getSessionId());
		}
		closeNow();
		callback.failed(t);
	}

	private void writingCompletedCallback(Callback callback, long currentWritenBytes) {
		lastWrittenTime = Millisecond100Clock.currentTimeMillis();
		if (currentWritenBytes < 0) {
			if (log.isDebugEnabled()) {
				log.debug("the session {} output is closed, {}", getSessionId(), currentWritenBytes);
			}
			closeNow();
			return;
		}

		if (log.isDebugEnabled()) {
			log.debug("the session {} writes {} bytes", getSessionId(), currentWritenBytes);
		}

		writtenBytes += currentWritenBytes;
		callback.succeeded();

		outputLock.lock();
		try {
			OutputEntry obj = outputBuffer.poll();
			if (obj != null) {
				_write(obj);
			} else {
				isWriting = false;
			}
		} finally {
			outputLock.unlock();
		}
	}

	void _write(final OutputEntry entry) {
		if (!isOpen())
			return;

		switch (entry.getOutputEntryType()) {
		case BYTE_BUFFER:
			ByteBufferOutputEntry byteBufferOutputEntry = (ByteBufferOutputEntry) entry;
			if (log.isDebugEnabled()) {
				log.debug("the session {} will write buffer {}", getSessionId(),
						byteBufferOutputEntry.getData().remaining());
			}
			socketChannel.write(byteBufferOutputEntry.getData(), config.getTimeout(), TimeUnit.MILLISECONDS, this,
					new CompletionHandler() {

						@Override
						public void completed(Integer currentWritenBytes, AsynchronousTcpSession session) {
							writingCompletedCallback(entry.getCallback(), currentWritenBytes);
						}

						@Override
						public void failed(Throwable t, AsynchronousTcpSession session) {
							writingFailedCallback(entry.getCallback(), t);
						}
					});
			break;

		case BYTE_BUFFER_ARRAY:
			ByteBufferArrayOutputEntry byteBuffersEntry = (ByteBufferArrayOutputEntry) entry;
			socketChannel.write(byteBuffersEntry.getData(), 0, byteBuffersEntry.getData().length, config.getTimeout(),
					TimeUnit.MILLISECONDS, this, new CompletionHandler() {

						@Override
						public void completed(Long currentWritenBytes, AsynchronousTcpSession session) {
							writingCompletedCallback(entry.getCallback(), currentWritenBytes);
						}

						@Override
						public void failed(Throwable t, AsynchronousTcpSession session) {
							writingFailedCallback(entry.getCallback(), t);
						}
					});
			break;
		case DISCONNECTION:
			log.debug("the session {} will close", getSessionId());
			shutdownSocketChannel();
		default:
			break;
		}
	}

	@Override
	public void attachObject(Object attachment) {
		this.attachment = attachment;
	}

	@Override
	public Object getAttachment() {
		return attachment;
	}

	@Override
	public void fireReceiveMessage(Object message) {
		eventManager.executeReceiveTask(this, message);
	}

	@Override
	public void encode(Object message) {
		try {
			config.getEncoder().encode(message, this);
		} catch (Throwable t) {
			eventManager.executeExceptionTask(this, t);
		}
	}

	@Override
	public void write(OutputEntry entry) {
		if (entry == null)
			return;

		outputLock.lock();
		try {
			if (!isWriting) {
				isWriting = true;
				_write(entry);
			} else {
				outputBuffer.offer(entry);
			}
		} finally {
			outputLock.unlock();
		}
	}

	@Override
	public void write(ByteBuffer byteBuffer, Callback callback) {
		write(new ByteBufferOutputEntry(callback, byteBuffer));
	}

	@Override
	public void write(ByteBuffer[] buffers, Callback callback) {
		write(new ByteBufferArrayOutputEntry(callback, buffers));
	}

	@Override
	public void write(Collection buffers, Callback callback) {
		write(new ByteBufferArrayOutputEntry(callback, buffers.toArray(BufferUtils.EMPTY_BYTE_BUFFER_ARRAY)));
	}

	private class FileBufferReaderHandler implements BufferReaderHandler {

		private final long len;

		public FileBufferReaderHandler(long len) {
			this.len = len;
		}

		@Override
		public void readBuffer(ByteBuffer buf, CountingCallback countingCallback, long count) {
			log.debug("write file,  count: {} , lenth: {}", count, len);
			write(buf, countingCallback);
		}

	}

	@Override
	public void write(FileRegion file, Callback callback) {
		try (FileRegion fileRegion = file) {
			fileRegion.transferTo(callback, new FileBufferReaderHandler(fileRegion.getLength()));
		} catch (Throwable t) {
			log.error("transfer file error", t);
		}
	}

	@Override
	public void close() {
		write(DISCONNECTION_FLAG);
	}

	@Override
	public void closeNow() {
		if (!isOpen())
			return;

		closeTime = Millisecond100Clock.currentTimeMillis();
		try {
			socketChannel.close();
		} catch (AsynchronousCloseException e) {
			if (log.isDebugEnabled())
				log.debug("the session {} asynchronously closed", sessionId);
		} catch (IOException e) {
			log.error("the session {} close error", e, sessionId);
		}
		state = CLOSE;
		eventManager.executeCloseTask(this);
	}

	@Override
	public void shutdownOutput() {
		try {
			socketChannel.shutdownOutput();
		} catch (ClosedChannelException e) {
			log.debug("the session {} is closed", e, sessionId);
		} catch (IOException e) {
			log.error("the session {} shutdown output error", e, sessionId);
		}
	}

	@Override
	public void shutdownInput() {
		try {
			socketChannel.shutdownInput();
		} catch (ClosedChannelException e) {
			log.debug("the session {} is closed", e, sessionId);
		} catch (IOException e) {
			log.error("the session {} shutdown input error", e, sessionId);
		}
	}

	private void shutdownSocketChannel() {
		shutdownOutput();
		shutdownInput();
	}

	@Override
	public int getSessionId() {
		return sessionId;
	}

	@Override
	public long getOpenTime() {
		return openTime;
	}

	@Override
	public long getCloseTime() {
		return closeTime;
	}

	@Override
	public long getDuration() {
		if (closeTime > 0) {
			return closeTime - openTime;
		} else {
			return Millisecond100Clock.currentTimeMillis() - openTime;
		}
	}

	@Override
	public long getLastReadTime() {
		return lastReadTime;
	}

	@Override
	public long getLastWrittenTime() {
		return lastWrittenTime;
	}

	@Override
	public long getLastActiveTime() {
		return Math.max(lastReadTime, lastWrittenTime);
	}

	@Override
	public long getReadBytes() {
		return readBytes;
	}

	@Override
	public long getWrittenBytes() {
		return writtenBytes;
	}

	@Override
	public int getState() {
		return state;
	}

	@Override
	public boolean isOpen() {
		return state > 0;
	}

	@Override
	public InetSocketAddress getLocalAddress() {
		if (localAddress != null) {
			return localAddress;
		} else {
			try {
				localAddress = (InetSocketAddress) socketChannel.getLocalAddress();
				return localAddress;
			} catch (IOException e) {
				log.error("the session {} gets local address error", e, sessionId);
				return null;
			}
		}
	}

	@Override
	public InetSocketAddress getRemoteAddress() {
		if (remoteAddress != null) {
			return remoteAddress;
		} else {
			try {
				remoteAddress = (InetSocketAddress) socketChannel.getRemoteAddress();
				return remoteAddress;
			} catch (Throwable t) {
				log.error("the session {} gets remote address error", t, sessionId);
				return null;
			}
		}
	}

	@Override
	public String toString() {
		return "[sessionId=" + sessionId + ", openTime="
				+ SafeSimpleDateFormat.defaultDateFormat.format(new Date(openTime)) + ", closeTime="
				+ SafeSimpleDateFormat.defaultDateFormat.format(new Date(closeTime)) + ", duration=" + getDuration()
				+ ", readBytes=" + readBytes + ", writtenBytes=" + writtenBytes + "]";
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy