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

org.rapidoid.net.impl.RapidoidConnection Maven / Gradle / Ivy

There is a newer version: 5.5.5
Show newest version
package org.rapidoid.net.impl;

import org.rapidoid.RapidoidThing;
import org.rapidoid.annotation.Authors;
import org.rapidoid.annotation.Since;
import org.rapidoid.buffer.Buf;
import org.rapidoid.buffer.BufGroup;
import org.rapidoid.buffer.BufUtil;
import org.rapidoid.data.JSON;
import org.rapidoid.expire.Expiring;
import org.rapidoid.job.Jobs;
import org.rapidoid.log.Log;
import org.rapidoid.net.AsyncLogic;
import org.rapidoid.net.Protocol;
import org.rapidoid.net.abstracts.Channel;
import org.rapidoid.net.abstracts.IRequest;
import org.rapidoid.net.tls.RapidoidTLS;
import org.rapidoid.u.U;
import org.rapidoid.util.Constants;
import org.rapidoid.util.Resetable;

import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.net.InetSocketAddress;
import java.net.SocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;
import java.nio.channels.SelectionKey;
import java.nio.channels.SocketChannel;
import java.util.concurrent.atomic.AtomicLong;

/*
 * #%L
 * rapidoid-net
 * %%
 * Copyright (C) 2014 - 2017 Nikolche Mihajlovski and contributors
 * %%
 * 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.
 * #L%
 */

@Authors("Nikolche Mihajlovski")
@Since("2.0.0")
public class RapidoidConnection extends RapidoidThing implements Resetable, Channel, Expiring, Constants {

	private static final CtxListener IGNORE = new IgnorantConnectionListener();

	private static final AtomicLong ID_N = new AtomicLong();

	private static final AtomicLong SERIAL_N = new AtomicLong();

	final boolean hasTLS;

	final RapidoidTLS tls;

	final RapidoidWorker worker;

	public final Buf input;

	public final Buf output;

	public final Buf outgoing;

	private final ConnState state = new ConnState();

	private volatile boolean waitingToWrite = false;

	public volatile SelectionKey key;

	private volatile boolean closeAfterWrite = false;

	volatile boolean closed = true;

	volatile boolean closing = false;

	volatile int completedInputPos;

	private volatile CtxListener listener;

	private final long serialN = SERIAL_N.incrementAndGet();

	private volatile long id;

	private volatile boolean initial;

	volatile boolean async;

	volatile boolean done;

	private volatile boolean isClient;

	private volatile Protocol protocol;

	volatile long requestId;

	final AtomicLong readSeq = new AtomicLong();

	final AtomicLong writeSeq = new AtomicLong();

	volatile boolean resumeInProgress = false;

	volatile IRequest request;

	private volatile long expiresAt;

	public RapidoidConnection(RapidoidWorker worker, BufGroup bufs) {
		this.worker = worker;

		this.hasTLS = worker.sslContext() != null;
		this.tls = hasTLS ? new RapidoidTLS(worker.sslContext(), this) : null;

		this.input = bufs.newBuf("input#" + serialN);
		this.output = bufs.newBuf("output#" + serialN);
		this.outgoing = hasTLS ? bufs.newBuf("outgoing#" + serialN) : this.output;

		reset();
	}

	@Override
	public synchronized void reset() {
		IRequest req = request;
		if (req != null) {
			req.stop();
			request = null;
		}

		id = ID_N.incrementAndGet();
		key = null;
		closed = true;
		closing = false;
		input.clear();
		output.clear();
		outgoing.clear();
		closeAfterWrite = false;
		waitingToWrite = false;
		completedInputPos = 0;
		listener = IGNORE;
		initial = true;
		async = false;
		done = false;
		isClient = false;
		protocol = null;
		requestId = 0;
		readSeq.set(0);
		writeSeq.set(0);
		expiresAt = 0;
		state.reset();

		if (tls != null) tls.reset();
	}

	@Override
	public void log(String msg) {
		state().log(msg);
	}

	@Override
	public synchronized InetSocketAddress getAddress() {
		if (key == null) return null;

		SocketChannel socketChannel = (SocketChannel) key.channel();
		SocketAddress addr = socketChannel.socket().getRemoteSocketAddress();
		if (addr instanceof InetSocketAddress) {
			InetSocketAddress address = (InetSocketAddress) addr;
			return address;
		} else {
			throw new IllegalStateException("Cannot get remote address!");
		}
	}

	@Override
	public synchronized Channel write(String s) {
		output.append(s);
		return this;
	}

	@Override
	public synchronized Channel writeln(String s) {
		output.append(s);
		output.append(CR_LF);
		return this;
	}

	@Override
	public synchronized Channel write(byte[] bytes) {
		return write(bytes, 0, bytes.length);
	}

	@Override
	public synchronized Channel write(byte[] bytes, int offset, int length) {
		output.append(bytes, offset, length);
		return this;
	}

	@Override
	public synchronized Channel write(ByteBuffer buf) {
		output.append(buf);
		return this;
	}

	@Override
	public synchronized Channel write(File file) {
		try {
			FileInputStream stream = new FileInputStream(file);
			FileChannel fileChannel = stream.getChannel();
			output.append(fileChannel);
			stream.close();
		} catch (IOException e) {
			throw U.rte(e);
		}

		return this;
	}

	@Override
	public Channel writeJSON(Object value) {
		JSON.stringify(value, output.asOutputStream());
		return this;
	}

	public boolean closeAfterWrite() {
		return closeAfterWrite;
	}

	Channel done() {
		async = false;

		if (!done) {
			done = true;
			askToSend();
		}

		return this;
	}

	void processedSeq(long processedHandle) {

		if (processedHandle == 0) return; // a new connection

		U.must(processedHandle > 0);

		boolean increased = writeSeq.compareAndSet(processedHandle - 1, processedHandle);

		if (!increased) {
			// the current response might be already marked as processed (e.g. in non-async handlers)
			long writeSeqN = writeSeq.get();
			if (writeSeqN != processedHandle) {
				throw U.rte("Error in the response order control! Expected handle: %s, real: %s", processedHandle - 1, writeSeqN);
			}
		}
	}

	@Override
	public Channel send() {
		askToSend();
		return this;
	}

	public void error() {
		askToSend();
	}

	private synchronized void askToSend() {
		synchronized (outgoing) {
			if (hasTLS) {
				synchronized (output) {
					tls.wrapToOutgoing();
				}
			}

			if (!waitingToWrite && outgoing.size() > 0) {
				waitingToWrite = true;
				worker.wantToWrite(this);
			}
		}
	}

	public synchronized void close(boolean waitToWrite) {
		if (waitToWrite) {
			done();
		}

		if (waitToWrite && waitingToWrite) {
			closeAfterWrite = true;
		} else {
			worker.close(this);
		}
	}

	synchronized void wrote(boolean complete) {
		if (complete) {
			waitingToWrite = false;
		}

		input.deleteBefore(completedInputPos);
		completedInputPos = 0;
	}

	@Override
	public void resume(final long expectedConnId, final long handle, final AsyncLogic asyncLogic) {

		if (expectedConnId != connId()) return;

		long seq = writeSeq.get();

		if (seq < handle - 1) {
			// too early

			Jobs.execute(new Runnable() {
				@Override
				public void run() {
					resume(expectedConnId, handle, asyncLogic);
				}
			});

		} else if (seq == handle - 1) {

			synchronized (this) {

				if (expectedConnId != connId()) {
					return;
				}

//				TODO investigate options for stricter flow control:
//				U.must(!resumeInProgress, "Resume is already in progress!");

				resumeInProgress = true;

				try {
					doResume(handle, asyncLogic, seq);

				} finally {
					resumeInProgress = false;
				}
			}

		} else {
			Log.error("Tried to resume a job that already has finished!", "handle", handle, "currentHandle", seq, "job", asyncLogic);
			throw U.rte("Tried to resume a job that already has finished!");
		}
	}

	private void doResume(long handle, AsyncLogic asyncLogic, long seq) {
		U.must(seq == writeSeq.get());

		// execute the logic
		boolean finished = false;

		synchronized (output) {
			BufUtil.startWriting(output);

			try {
				finished = asyncLogic.resumeAsync();
			} catch (Throwable e) {
				Log.error("Error while resuming an asynchronous operation!", e);
			}

			BufUtil.doneWriting(output);
		}

		if (finished) {
			processedSeq(handle);
		}
	}

	@Override
	public Buf input() {
		return input;
	}

	@Override
	public Buf output() {
		return output;
	}

	@Override
	public OutputStream outputStream() {
		return output.asOutputStream();
	}

	@Override
	public boolean onSameThread() {
		return worker.onSameThread();
	}

	@Override
	public RapidoidHelper helper() {
		return worker.helper;
	}

	public CtxListener listener() {
		return listener;
	}

	public void setListener(CtxListener listener) {
		this.listener = listener;
	}

	@Override
	public String address() {
		InetSocketAddress inetSocketAddress = getAddress();
		return inetSocketAddress != null ? inetSocketAddress.getAddress().getHostAddress() : null;
	}

	@Override
	public Channel close() {
		close(true);
		return this;
	}

	@Override
	public Channel closeIf(boolean condition) {
		if (condition) {
			close();
		}
		return this;
	}

	@Override
	public String readln() {
		return input().readLn();
	}

	@Override
	public String readN(int count) {
		return input().readN(count);
	}

	@Override
	public ConnState state() {
		return state;
	}

	@Override
	public long handle() {
		return readSeq.get();
	}

	@Override
	public boolean isInitial() {
		return initial;
	}

	@Override
	public String toString() {
		return "conn#" + connId();
	}

	public void setInitial(boolean initial) {
		this.initial = initial;
	}

	@Override
	public synchronized long async() {
		U.must(onSameThread(), "The connection can be marked as 'async' only on its I/O worker thread!");

		this.async = true;
		this.done = false;
		return handle();
	}

	@Override
	public synchronized boolean isAsync() {
		return async;
	}

	public boolean isClient() {
		return isClient;
	}

	public void setClient(boolean isClient) {
		this.isClient = isClient;
	}

	public void setProtocol(Protocol protocol) {
		this.protocol = protocol;
	}

	public Protocol getProtocol() {
		return protocol;
	}

	@Override
	public boolean isClosing() {
		return closing;
	}

	@Override
	public boolean isClosed() {
		return closed;
	}

	@Override
	public void waitUntilClosing() {
		if (!isClosing()) {
			throw Buf.INCOMPLETE_READ;
		}
	}

	@Override
	public long connId() {
		return id;
	}

	@Override
	public long requestId() {
		return requestId;
	}

	@Override
	public void setRequest(IRequest request) {
		this.request = request;
	}

	@Override
	public void setExpiresAt(long expiresAt) {
		this.expiresAt = expiresAt;
	}

	@Override
	public long getExpiresAt() {
		return expiresAt;
	}

	@Override
	public void expire() {
		close(false);
	}

	public boolean finishedWriting() {
		return outgoing.size() == 0;
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy