org.apache.mina.filter.executor.WriteRequestFilter Maven / Gradle / Ivy
/**
* Copyright 2007-2015, Kaazing Corporation. All rights reserved.
*
* 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.
*/
package org.apache.mina.filter.executor;
import org.apache.mina.core.filterchain.IoFilterAdapter;
import org.apache.mina.core.future.IoFutureListener;
import org.apache.mina.core.future.WriteFuture;
import org.apache.mina.core.service.IoProcessor;
import org.apache.mina.core.session.IoEvent;
import org.apache.mina.core.session.IoEventType;
import org.apache.mina.core.session.IoSession;
import org.apache.mina.core.write.WriteRequest;
/**
* Attaches an {@link IoEventQueueHandler} to an {@link IoSession}'s
* {@link WriteRequest} queue to provide accurate write queue status tracking.
*
* The biggest difference from {@link OrderedThreadPoolExecutor} and
* {@link UnorderedThreadPoolExecutor} is that {@link IoEventQueueHandler#polled(Object, IoEvent)}
* is invoked when the write operation is completed by an {@link IoProcessor},
* consequently providing the accurate tracking of the write request queue
* status to the {@link IoEventQueueHandler}.
*
* Most common usage of this filter could be detecting an {@link IoSession}
* which writes too fast which will cause {@link OutOfMemoryError} soon:
*
* session.getFilterChain().addLast(
* "writeThrottle",
* new WriteRequestFilter(new IoEventQueueThrottle()));
*
*
* Known issues
*
* You can run into a dead lock if you run this filter with the blocking
* {@link IoEventQueueHandler} implementation such as {@link IoEventQueueThrottle}
* in the {@link IoProcessor} thread. It's because an {@link IoProcessor}
* thread is what processes the {@link WriteRequest}s and notifies related
* {@link WriteFuture}s; the {@link IoEventQueueHandler} implementation that
* waits for the size of the write request queue to decrease will never wake
* up. To use such an handler, you have to insert an {@link ExecutorFilter}
* before this filter or call {@link IoSession#write(Object)} method always
* from a different thread.
*
* @author Apache MINA Project
*/
public class WriteRequestFilter extends IoFilterAdapter {
private final IoEventQueueHandler queueHandler;
/**
* Creates a new instance with a new default {@link IoEventQueueThrottle}.
*/
public WriteRequestFilter() {
this(new IoEventQueueThrottle());
}
/**
* Creates a new instance with the specified {@link IoEventQueueHandler}.
*/
public WriteRequestFilter(IoEventQueueHandler queueHandler) {
if (queueHandler == null) {
throw new NullPointerException("queueHandler");
}
this.queueHandler = queueHandler;
}
/**
* Returns the {@link IoEventQueueHandler} which is attached to this
* filter.
*/
public IoEventQueueHandler getQueueHandler() {
return queueHandler;
}
@Override
public void filterWrite(
NextFilter nextFilter,
IoSession session, WriteRequest writeRequest) throws Exception {
final IoEvent e = new IoEvent(IoEventType.WRITE, session, writeRequest);
if (queueHandler.accept(this, e)) {
nextFilter.filterWrite(session, writeRequest);
WriteFuture writeFuture = writeRequest.getFuture();
if (writeFuture == null) {
return;
}
// We can track the write request only when it has a future.
queueHandler.offered(this, e);
writeFuture.addListener(new IoFutureListener() {
public void operationComplete(WriteFuture future) {
queueHandler.polled(WriteRequestFilter.this, e);
}
});
}
}
}