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

com.firefly.codec.http2.stream.HTTP2Flusher Maven / Gradle / Ivy

package com.firefly.codec.http2.stream;

import java.nio.ByteBuffer;
import java.nio.channels.ClosedChannelException;
import java.util.ArrayDeque;
import java.util.ArrayList;
import java.util.Deque;
import java.util.LinkedList;
import java.util.List;
import java.util.Queue;

import com.firefly.codec.http2.frame.Frame;
import com.firefly.codec.http2.frame.WindowUpdateFrame;
import com.firefly.net.ByteBufferArrayOutputEntry;
import com.firefly.utils.concurrent.Callback;
import com.firefly.utils.concurrent.IteratingCallback;
import com.firefly.utils.io.BufferUtils;
import com.firefly.utils.io.EofException;
import com.firefly.utils.log.Log;
import com.firefly.utils.log.LogFactory;

public class HTTP2Flusher extends IteratingCallback {
	private static Log log = LogFactory.getInstance().getLog("firefly-system");

	private final Queue windows = new ArrayDeque<>();
	private final Deque frames = new ArrayDeque<>();
	private final Queue entries = new ArrayDeque<>();
	private final List actives = new ArrayList<>();
	private final HTTP2Session session;
	private final Queue buffers = new LinkedList<>();
	private Entry stalled;
	private boolean terminated;

	public HTTP2Flusher(HTTP2Session session) {
		this.session = session;
	}

	public void window(StreamSPI stream, WindowUpdateFrame frame) {
		boolean closed;
		synchronized (this) {
			closed = terminated;
			if (!closed)
				windows.offer(new WindowEntry(stream, frame));
		}
		// Flush stalled data.
		if (!closed)
			iterate();
	}

	public boolean prepend(Entry entry) {
		boolean closed;
		synchronized (this) {
			closed = terminated;
			if (!closed) {
				frames.offerFirst(entry);
				if (log.isDebugEnabled())
					log.debug("Prepended {}, frames={}", entry, frames.size());
			}
		}
		if (closed)
			closed(entry, new ClosedChannelException());
		return !closed;
	}

	public boolean append(Entry entry) {
		boolean closed;
		synchronized (this) {
			closed = terminated;
			if (!closed) {
				frames.offer(entry);
				if (log.isDebugEnabled())
					log.debug("Appended {}, frames={}", entry, frames.size());
			}
		}
		if (closed)
			closed(entry, new ClosedChannelException());
		return !closed;
	}

	public int getQueueSize() {
		synchronized (this) {
			return frames.size();
		}
	}

	@Override
	protected Action process() throws Exception {
		if (log.isDebugEnabled())
			log.debug("Flushing {}", session);

		synchronized (this) {
			if (terminated)
				throw new ClosedChannelException();

			while (!windows.isEmpty()) {
				WindowEntry entry = windows.poll();
				entry.perform();
			}

			if (!frames.isEmpty()) {
				for (Entry entry : frames) {
					entries.offer(entry);
					actives.add(entry);
				}
				frames.clear();
			}
		}

		if (entries.isEmpty()) {
			if (log.isDebugEnabled())
				log.debug("Flushed {}", session);
			return Action.IDLE;
		}

		while (!entries.isEmpty()) {
			Entry entry = entries.poll();
			if (log.isDebugEnabled())
				log.debug("Processing {}", entry);

			// If the stream has been reset, don't send the frame.
			if (entry.reset()) {
				if (log.isDebugEnabled())
					log.debug("Resetting {}", entry);
				continue;
			}

			try {
				if (entry.generate(buffers)) {
					if (entry.dataRemaining() > 0)
						entries.offer(entry);
				} else {
					if (stalled == null)
						stalled = entry;
				}
			} catch (Throwable failure) {
				// Failure to generate the entry is catastrophic.
				if (log.isDebugEnabled())
					log.debug("Failure generating frame " + entry.frame, failure);
				failed(failure);
				return Action.SUCCEEDED;
			}
		}

		if (buffers.isEmpty()) {
			complete();
			return Action.IDLE;
		}

		if (log.isDebugEnabled())
			log.debug("Writing {} buffers ({} bytes) for {} frames {}", buffers.size(), getBufferTotalLength(),
					actives.size(), actives.toString());

		ByteBufferArrayOutputEntry outputEntry = new ByteBufferArrayOutputEntry(this,
				buffers.toArray(BufferUtils.EMPTY_BYTE_BUFFER_ARRAY));
		session.getEndPoint().encode(outputEntry);
		return Action.SCHEDULED;
	}

	private int getBufferTotalLength() {
		int length = 0;
		for (ByteBuffer buf : buffers) {
			length += buf.remaining();
		}
		return length;
	}

	@Override
	public void succeeded() {
		if (log.isDebugEnabled())
			log.debug("Written {} frames for {}", actives.size(), actives);

		complete();

		super.succeeded();
	}

	private void complete() {
		buffers.clear();

		for(Entry entry : actives) {
			entry.complete();
		}

		if (stalled != null) {
			// We have written part of the frame, but there is more to write.
			// The API will not allow to send two data frames for the same
			// stream so we append the unfinished frame at the end to allow
			// better interleaving with other streams.
			int index = actives.indexOf(stalled);
			for (int i = index; i < actives.size(); ++i) {
				Entry entry = actives.get(i);
				if (entry.dataRemaining() > 0)
					append(entry);
			}
			for (int i = 0; i < index; ++i) {
				Entry entry = actives.get(i);
				if (entry.dataRemaining() > 0)
					append(entry);
			}
			stalled = null;
		}

		actives.clear();
	}

	@Override
	protected void onCompleteSuccess() {
		throw new IllegalStateException();
	}

	@Override
	protected void onCompleteFailure(Throwable x) {
		buffers.clear();

		boolean closed;
		synchronized (this) {
			closed = terminated;
			terminated = true;
			if (log.isDebugEnabled())
				log.debug("{}, active/queued={}/{}", closed ? "Closing" : "Failing", actives.size(), frames.size());
			actives.addAll(frames);
			frames.clear();
		}

		for(Entry entry : actives) {
			entry.failed(x);
		}
		actives.clear();

		// If the failure came from within the
		// flusher, we need to close the connection.
		if (!closed)
			session.abort(x);
	}

	void terminate() {
		boolean closed;
		synchronized (this) {
			closed = terminated;
			terminated = true;
			if (log.isDebugEnabled())
				log.debug("{}", closed ? "Terminated" : "Terminating");
		}
		if (!closed)
			iterate();
	}

	private void closed(Entry entry, Throwable failure) {
		entry.failed(failure);
	}

	public static abstract class Entry extends Callback.Nested {
		protected final Frame frame;
		protected final StreamSPI stream;
		private boolean reset;

		protected Entry(Frame frame, StreamSPI stream, Callback callback) {
			super(callback);
			this.frame = frame;
			this.stream = stream;
		}

		public int dataRemaining() {
			return 0;
		}

		protected abstract boolean generate(Queue buffers);

		private void complete() {
			if (reset)
				failed(new EofException("reset"));
			else
				succeeded();
		}

		@Override
		public void failed(Throwable x) {
			if (stream != null) {
				stream.close();
				stream.getSession().removeStream(stream);
			}
			super.failed(x);
		}

		private boolean reset() {
			return this.reset = stream != null && stream.isReset() && !isProtocol();
		}

		private boolean isProtocol() {
			switch (frame.getType()) {
			case PRIORITY:
			case RST_STREAM:
			case GO_AWAY:
			case WINDOW_UPDATE:
			case DISCONNECT:
				return true;
			default:
				return false;
			}
		}

		@Override
		public String toString() {
			return frame.toString();
		}
	}

	private class WindowEntry {
		private final StreamSPI stream;
		private final WindowUpdateFrame frame;

		public WindowEntry(StreamSPI stream, WindowUpdateFrame frame) {
			this.stream = stream;
			this.frame = frame;
		}

		public void perform() {
			FlowControlStrategy flowControl = session.getFlowControlStrategy();
			flowControl.onWindowUpdate(session, stream, frame);
		}
	}

	@Override
	public boolean isNonBlocking() {
		return false;
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy