Corrected changes introduced with Promise.

Save creation of iterators for every append() and prepend() in StandardSession.
Removed PromisingCallback, only used by SPDY and better implemented otherwise.
This commit is contained in:
Simone Bordet 2012-11-23 12:06:46 +01:00
parent ab642141d1
commit ee893d8526
8 changed files with 79 additions and 138 deletions

View File

@ -29,7 +29,6 @@ import org.eclipse.jetty.io.RuntimeIOException;
import org.eclipse.jetty.spdy.Controller; import org.eclipse.jetty.spdy.Controller;
import org.eclipse.jetty.spdy.ISession; import org.eclipse.jetty.spdy.ISession;
import org.eclipse.jetty.spdy.IdleListener; import org.eclipse.jetty.spdy.IdleListener;
import org.eclipse.jetty.spdy.StandardSession;
import org.eclipse.jetty.spdy.parser.Parser; import org.eclipse.jetty.spdy.parser.Parser;
import org.eclipse.jetty.util.Callback; import org.eclipse.jetty.util.Callback;
import org.eclipse.jetty.util.log.Log; import org.eclipse.jetty.util.log.Log;
@ -52,7 +51,11 @@ public class SPDYConnection extends AbstractConnection implements Controller, Id
public SPDYConnection(EndPoint endPoint, ByteBufferPool bufferPool, Parser parser, Executor executor,int bufferSize) public SPDYConnection(EndPoint endPoint, ByteBufferPool bufferPool, Parser parser, Executor executor,int bufferSize)
{ {
// TODO explain why we are passing false here // Since SPDY is multiplexed, onFillable() must never block
// while calling application code. In fact, onFillable()
// always dispatches to a new thread when calling application
// code, so here we can safely pass false as last parameter,
// and avoid to dispatch to onFillable().
super(endPoint, executor, false); super(endPoint, executor, false);
this.bufferPool = bufferPool; this.bufferPool = bufferPool;
this.parser = parser; this.parser = parser;

View File

@ -24,13 +24,14 @@ import org.eclipse.jetty.spdy.api.Stream;
import org.eclipse.jetty.spdy.api.StreamFrameListener; import org.eclipse.jetty.spdy.api.StreamFrameListener;
import org.eclipse.jetty.spdy.api.SynInfo; import org.eclipse.jetty.spdy.api.SynInfo;
import org.eclipse.jetty.spdy.frames.ControlFrame; import org.eclipse.jetty.spdy.frames.ControlFrame;
import org.eclipse.jetty.util.Callback;
/** /**
* <p>The internal interface that represents a stream.</p> * <p>The internal interface that represents a stream.</p>
* <p>{@link IStream} contains additional methods used by a SPDY * <p>{@link IStream} contains additional methods used by a SPDY
* implementation (and not by an application).</p> * implementation (and not by an application).</p>
*/ */
public interface IStream extends Stream public interface IStream extends Stream, Callback
{ {
/** /**
* <p>Senders of data frames need to know the current window size * <p>Senders of data frames need to know the current window size

View File

@ -24,10 +24,8 @@ import java.nio.channels.InterruptedByTimeoutException;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collections; import java.util.Collections;
import java.util.HashSet; import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedList; import java.util.LinkedList;
import java.util.List; import java.util.List;
import java.util.ListIterator;
import java.util.Map; import java.util.Map;
import java.util.Objects; import java.util.Objects;
import java.util.Set; import java.util.Set;
@ -172,7 +170,7 @@ public class StandardSession implements ISession, Parser.Listener, Dumpable
int streamId = streamIds.getAndAdd(2); int streamId = streamIds.getAndAdd(2);
// TODO: for SPDYv3 we need to support the "slot" argument // TODO: for SPDYv3 we need to support the "slot" argument
SynStreamFrame synStream = new SynStreamFrame(version, synInfo.getFlags(), streamId, associatedStreamId, synInfo.getPriority(), (short)0, synInfo.getHeaders()); SynStreamFrame synStream = new SynStreamFrame(version, synInfo.getFlags(), streamId, associatedStreamId, synInfo.getPriority(), (short)0, synInfo.getHeaders());
StandardStream stream = createStream(synStream, listener, true, promise); IStream stream = createStream(synStream, listener, true, promise);
generateAndEnqueueControlFrame(stream, synStream, timeout, unit, stream); generateAndEnqueueControlFrame(stream, synStream, timeout, unit, stream);
} }
flush(); flush();
@ -235,7 +233,7 @@ public class StandardSession implements ISession, Parser.Listener, Dumpable
public void ping(long timeout, TimeUnit unit, Promise<PingInfo> promise) public void ping(long timeout, TimeUnit unit, Promise<PingInfo> promise)
{ {
int pingId = pingIds.getAndAdd(2); int pingId = pingIds.getAndAdd(2);
PromisingPingInfoCallback pingInfo = new PromisingPingInfoCallback(pingId,promise); PingInfoCallback pingInfo = new PingInfoCallback(pingId, promise);
PingFrame frame = new PingFrame(version, pingId); PingFrame frame = new PingFrame(version, pingId);
control(null, frame, timeout, unit, pingInfo); control(null, frame, timeout, unit, pingInfo);
} }
@ -475,10 +473,10 @@ public class StandardSession implements ISession, Parser.Listener, Dumpable
removeStream(stream); removeStream(stream);
} }
private StandardStream createStream(SynStreamFrame frame, StreamFrameListener listener, boolean local,Promise<Stream> promise) private IStream createStream(SynStreamFrame frame, StreamFrameListener listener, boolean local, Promise<Stream> promise)
{ {
IStream associatedStream = streams.get(frame.getAssociatedStreamId()); IStream associatedStream = streams.get(frame.getAssociatedStreamId());
StandardStream stream = new StandardStream(frame.getStreamId(), frame.getPriority(), this, associatedStream,promise); IStream stream = new StandardStream(frame.getStreamId(), frame.getPriority(), this, associatedStream, promise);
flowControlStrategy.onNewStream(this, stream); flowControlStrategy.onNewStream(this, stream);
stream.updateCloseState(frame.isClose(), local); stream.updateCloseState(frame.isClose(), local);
@ -898,10 +896,9 @@ public class StandardSession implements ISession, Parser.Listener, Dumpable
return; return;
Set<IStream> stalledStreams = null; Set<IStream> stalledStreams = null;
Iterator<FrameBytes> iter = queue.iterator(); for (int i = 0; i < queue.size(); ++i)
while(iter.hasNext())
{ {
frameBytes=iter.next(); frameBytes = queue.get(i);
IStream stream = frameBytes.getStream(); IStream stream = frameBytes.getStream();
if (stream != null && stalledStreams != null && stalledStreams.contains(stream)) if (stream != null && stalledStreams != null && stalledStreams.contains(stream))
@ -910,7 +907,7 @@ public class StandardSession implements ISession, Parser.Listener, Dumpable
buffer = frameBytes.getByteBuffer(); buffer = frameBytes.getByteBuffer();
if (buffer != null) if (buffer != null)
{ {
iter.remove(); queue.remove(i);
if (stream != null && stream.isReset()) if (stream != null && stream.isReset())
{ {
frameBytes.fail(new StreamException(stream.getId(),StreamStatus.INVALID_STREAM, frameBytes.fail(new StreamException(stream.getId(),StreamStatus.INVALID_STREAM,
@ -945,18 +942,15 @@ public class StandardSession implements ISession, Parser.Listener, Dumpable
failure = this.failure; failure = this.failure;
if (failure == null) if (failure == null)
{ {
ListIterator<FrameBytes> iter = queue.listIterator(queue.size()); int index = queue.size();
while (index > 0)
while(iter.hasPrevious())
{ {
FrameBytes element = iter.previous(); FrameBytes element = queue.get(index - 1);
if (element.compareTo(frameBytes) >= 0) if (element.compareTo(frameBytes) >= 0)
{
iter.next();
break; break;
--index;
} }
} queue.add(index, frameBytes);
iter.add(frameBytes);
} }
} }
@ -972,18 +966,15 @@ public class StandardSession implements ISession, Parser.Listener, Dumpable
failure = this.failure; failure = this.failure;
if (failure == null) if (failure == null)
{ {
ListIterator<FrameBytes> iter = queue.listIterator(0); int index = 0;
while (index < queue.size())
while(iter.hasNext())
{ {
FrameBytes element = iter.next(); FrameBytes element = queue.get(index);
if (element.compareTo(frameBytes) <= 0) if (element.compareTo(frameBytes) <= 0)
{
iter.previous();
break; break;
++index;
} }
} queue.add(index,frameBytes);
iter.add(frameBytes);
} }
} }
@ -1154,7 +1145,6 @@ public class StandardSession implements ISession, Parser.Listener, Dumpable
fail(new InterruptedByTimeoutException()); fail(new InterruptedByTimeoutException());
} }
@Override @Override
public void succeeded() public void succeeded()
{ {
@ -1170,9 +1160,6 @@ public class StandardSession implements ISession, Parser.Listener, Dumpable
@Override @Override
public void failed(Throwable x) public void failed(Throwable x)
{ {
// TODO because this is using frameBytes here, then it is not really a Promise.
// frameBytes is not a result, but is something known before the operation is attempted!
List<FrameBytes> frameBytesToFail = new ArrayList<>(); List<FrameBytes> frameBytesToFail = new ArrayList<>();
frameBytesToFail.add(this); frameBytesToFail.add(this);
@ -1324,16 +1311,16 @@ public class StandardSession implements ISession, Parser.Listener, Dumpable
} }
} }
private static class PromisingPingInfoCallback extends PingInfo implements Callback private static class PingInfoCallback extends PingInfo implements Callback
{ {
public PromisingPingInfoCallback(int pingId,Promise<PingInfo> promise) private final Promise<PingInfo> promise;
public PingInfoCallback(int pingId, Promise<PingInfo> promise)
{ {
super(pingId); super(pingId);
this.promise=promise; this.promise=promise;
} }
private final Promise<PingInfo> promise;
@Override @Override
public void succeeded() public void succeeded()
{ {
@ -1347,6 +1334,5 @@ public class StandardSession implements ISession, Parser.Listener, Dumpable
if (promise != null) if (promise != null)
promise.failed(x); promise.failed(x);
} }
} }
} }

View File

@ -41,11 +41,10 @@ import org.eclipse.jetty.util.Callback;
import org.eclipse.jetty.util.FutureCallback; import org.eclipse.jetty.util.FutureCallback;
import org.eclipse.jetty.util.FuturePromise; import org.eclipse.jetty.util.FuturePromise;
import org.eclipse.jetty.util.Promise; import org.eclipse.jetty.util.Promise;
import org.eclipse.jetty.util.PromisingCallback;
import org.eclipse.jetty.util.log.Log; import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger; import org.eclipse.jetty.util.log.Logger;
public class StandardStream extends PromisingCallback<Stream> implements IStream public class StandardStream implements IStream
{ {
private static final Logger LOG = Log.getLogger(Stream.class); private static final Logger LOG = Log.getLogger(Stream.class);
private final Map<String, Object> attributes = new ConcurrentHashMap<>(); private final Map<String, Object> attributes = new ConcurrentHashMap<>();
@ -53,6 +52,7 @@ public class StandardStream extends PromisingCallback<Stream> implements IStream
private final byte priority; private final byte priority;
private final ISession session; private final ISession session;
private final IStream associatedStream; private final IStream associatedStream;
private final Promise<Stream> promise;
private final AtomicInteger windowSize = new AtomicInteger(); private final AtomicInteger windowSize = new AtomicInteger();
private final Set<Stream> pushedStreams = Collections.newSetFromMap(new ConcurrentHashMap<Stream, Boolean>()); private final Set<Stream> pushedStreams = Collections.newSetFromMap(new ConcurrentHashMap<Stream, Boolean>());
private volatile StreamFrameListener listener; private volatile StreamFrameListener listener;
@ -62,11 +62,11 @@ public class StandardStream extends PromisingCallback<Stream> implements IStream
public StandardStream(int id, byte priority, ISession session, IStream associatedStream, Promise<Stream> promise) public StandardStream(int id, byte priority, ISession session, IStream associatedStream, Promise<Stream> promise)
{ {
super(promise);
this.id = id; this.id = id;
this.priority = priority; this.priority = priority;
this.session = session; this.session = session;
this.associatedStream = associatedStream; this.associatedStream = associatedStream;
this.promise = promise;
} }
@Override @Override
@ -253,6 +253,20 @@ public class StandardStream extends PromisingCallback<Stream> implements IStream
session.flush(); session.flush();
} }
@Override
public void succeeded()
{
if (promise != null)
promise.succeeded(this);
}
@Override
public void failed(Throwable x)
{
if (promise != null)
promise.failed(x);
}
private void notifyOnReply(ReplyInfo replyInfo) private void notifyOnReply(ReplyInfo replyInfo)
{ {
final StreamFrameListener listener = this.listener; final StreamFrameListener listener = this.listener;
@ -329,16 +343,16 @@ public class StandardStream extends PromisingCallback<Stream> implements IStream
} }
@Override @Override
public void syn(SynInfo synInfo, long timeout, TimeUnit unit, Promise<Stream> callback) public void syn(SynInfo synInfo, long timeout, TimeUnit unit, Promise<Stream> promise)
{ {
if (isClosed() || isReset()) if (isClosed() || isReset())
{ {
callback.failed(new StreamException(getId(), StreamStatus.STREAM_ALREADY_CLOSED, promise.failed(new StreamException(getId(), StreamStatus.STREAM_ALREADY_CLOSED,
"Stream: " + this + " already closed or reset!")); "Stream: " + this + " already closed or reset!"));
return; return;
} }
PushSynInfo pushSynInfo = new PushSynInfo(getId(), synInfo); PushSynInfo pushSynInfo = new PushSynInfo(getId(), synInfo);
session.syn(pushSynInfo, null, timeout, unit, callback); session.syn(pushSynInfo, null, timeout, unit, promise);
} }
@Override @Override
@ -363,9 +377,9 @@ public class StandardStream extends PromisingCallback<Stream> implements IStream
@Override @Override
public Future<Void> data(DataInfo dataInfo) public Future<Void> data(DataInfo dataInfo)
{ {
FutureCallback fcb = new FutureCallback(); FutureCallback result = new FutureCallback();
data(dataInfo,0,TimeUnit.MILLISECONDS,fcb); data(dataInfo, 0, TimeUnit.MILLISECONDS, result);
return fcb; return result;
} }
@Override @Override
@ -390,9 +404,9 @@ public class StandardStream extends PromisingCallback<Stream> implements IStream
@Override @Override
public Future<Void> headers(HeadersInfo headersInfo) public Future<Void> headers(HeadersInfo headersInfo)
{ {
FutureCallback fcb = new FutureCallback(); FutureCallback result = new FutureCallback();
headers(headersInfo,0,TimeUnit.MILLISECONDS,fcb); headers(headersInfo, 0, TimeUnit.MILLISECONDS, result);
return fcb; return result;
} }
@Override @Override

View File

@ -22,7 +22,7 @@ import java.nio.ByteBuffer;
import java.nio.channels.ClosedChannelException; import java.nio.channels.ClosedChannelException;
import java.util.concurrent.CountDownLatch; import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutionException; import java.util.concurrent.ExecutionException;
import java.util.concurrent.Executor; import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException; import java.util.concurrent.TimeoutException;
@ -73,22 +73,19 @@ public class StandardSessionTest
{ {
@Mock @Mock
private Controller controller; private Controller controller;
private ExecutorService threadPool;
private ByteBufferPool bufferPool;
private Executor threadPool;
private StandardSession session; private StandardSession session;
private Generator generator;
private Scheduler scheduler; private Scheduler scheduler;
private Fields headers; private Fields headers;
@Before @Before
public void setUp() throws Exception public void setUp() throws Exception
{ {
bufferPool = new MappedByteBufferPool();
threadPool = Executors.newCachedThreadPool(); threadPool = Executors.newCachedThreadPool();
scheduler = new TimerScheduler(); scheduler = new TimerScheduler();
scheduler.start(); scheduler.start();
generator = new Generator(bufferPool, new StandardCompressionFactory.StandardCompressor()); ByteBufferPool bufferPool = new MappedByteBufferPool();
Generator generator = new Generator(bufferPool, new StandardCompressionFactory.StandardCompressor());
session = new StandardSession(SPDY.V2, bufferPool,threadPool,scheduler,controller,null,1,null, generator,new FlowControlStrategy.None()); session = new StandardSession(SPDY.V2, bufferPool,threadPool,scheduler,controller,null,1,null, generator,new FlowControlStrategy.None());
headers = new Fields(); headers = new Fields();
} }
@ -97,6 +94,7 @@ public class StandardSessionTest
public void after() throws Exception public void after() throws Exception
{ {
scheduler.stop(); scheduler.stop();
threadPool.shutdownNow();
} }
@SuppressWarnings("unchecked") @SuppressWarnings("unchecked")
@ -492,7 +490,7 @@ public class StandardSessionTest
private void assertThatPushStreamIsNotInSession(Stream pushStream) private void assertThatPushStreamIsNotInSession(Stream pushStream)
{ {
assertThat("pushStream is not in session",session.getStreams().contains(pushStream.getId()),not(true)); assertThat("pushStream is not in session",session.getStreams().contains(pushStream),not(true));
} }
private void assertThatPushStreamIsInSession(Stream pushStream) private void assertThatPushStreamIsInSession(Stream pushStream)

View File

@ -217,9 +217,8 @@ public class ProxyHTTPSPDYConnection extends HttpConnection implements HttpParse
@Override @Override
public void syn(SynInfo synInfo, long timeout, TimeUnit unit, Promise<Stream> handler) public void syn(SynInfo synInfo, long timeout, TimeUnit unit, Promise<Stream> handler)
{ {
// TODO is this right? comment or imple are wrong??
// HTTP does not support pushed streams // HTTP does not support pushed streams
handler.succeeded(this); handler.succeeded(new HTTPPushStream(2, getPriority(), getSession(), this));
} }
@Override @Override

View File

@ -1,60 +0,0 @@
//
// ========================================================================
// Copyright (c) 1995-2012 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// and Apache License v2.0 which accompanies this distribution.
//
// The Eclipse Public License is available at
// http://www.eclipse.org/legal/epl-v10.html
//
// The Apache License v2.0 is available at
// http://www.opensource.org/licenses/apache2.0.php
//
// You may elect to redistribute this code under either of these licenses.
// ========================================================================
//
package org.eclipse.jetty.util;
/* ------------------------------------------------------------ */
/** A Callback that can fulfil a Promise.
* <p>When this callback is successful, it will call the
* {@link Promise#succeeded(Object)} method of the wrapped Promise,
* passing the held result.
* @param <R> The type of the result for the promise.
*/
public class PromisingCallback<R> implements Callback
{
private final Promise<R> _promise;
private final R _result;
protected PromisingCallback(Promise<R> promise)
{
_promise=promise;
_result=(R)this;
}
public PromisingCallback(Promise<R> promise, R result)
{
_promise=promise;
_result=result;
}
@Override
public void succeeded()
{
if (_promise!=null)
_promise.succeeded(_result);
}
@Override
public void failed(Throwable x)
{
if (_promise!=null)
_promise.failed(x);
}
}