/*
* (C) Copyright 2017 Nuxeo SA (http://nuxeo.com/) and others.
*
* 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.nuxeo.ecm.platform.web.common.requestcontroller.filter;
import java.io.IOException;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import org.nuxeo.runtime.RuntimeServiceEvent;
import org.nuxeo.runtime.RuntimeServiceListener;
import org.nuxeo.runtime.api.Framework;
/**
* Blocks incoming requests when runtime is in standby mode.
*
* @since 9.2
*/
public class NuxeoStandbyFilter implements Filter {
protected Controller controller;
@Override
public void init(FilterConfig filterConfig) throws ServletException {
controller = new Controller();
Framework.addListener(new RuntimeServiceListener() {
@Override
public void handleEvent(RuntimeServiceEvent event) {
if (event.id == RuntimeServiceEvent.RUNTIME_ABOUT_TO_STANDBY) {
controller.onStandby();
} else if (event.id == RuntimeServiceEvent.RUNTIME_RESUMED) {
controller.onResumed();
}
}
});
}
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
throws IOException, ServletException {
controller.onNewRequest();
try {
chain.doFilter(request, response);
} finally {
controller.onRequestEnd();
}
}
@Override
public void destroy() {
}
protected class Controller {
protected final Lock lock = new ReentrantLock();
protected final Condition canStandby = lock.newCondition();
protected final Condition canProceed = lock.newCondition();
protected volatile boolean isStandby = Framework.getRuntime().isStandby();
protected volatile int inprogress = 0;
public void onNewRequest() {
if (!isStandby) {
inprogress += 1;
return;
}
awaitCanProceed();
}
public void onRequestEnd() {
inprogress -= 1;
if (inprogress > 0) {
return;
}
lock.lock();
try {
canStandby.signal();
} finally {
lock.unlock();
}
}
public void onStandby() throws RuntimeException {
isStandby = true;
if (inprogress > 0) {
awaitCanStandby();
}
}
public void onResumed() {
isStandby = false;
signalBlocked();
}
protected void awaitCanProceed() throws RuntimeException {
lock.lock();
try {
canProceed.await();
} catch (InterruptedException cause) {
Thread.currentThread().interrupt();
throw new RuntimeException("Interrupted while locking incoming requests", cause);
} finally {
lock.unlock();
}
}
protected void awaitCanStandby() throws RuntimeException {
lock.lock();
try {
canStandby.await();
} catch (InterruptedException cause) {
Thread.currentThread().interrupt();
throw new RuntimeException("Interrupted while waiting for web requests being drained", cause);
} finally {
lock.unlock();
}
}
protected void signalBlocked() {
lock.lock();
try {
canProceed.signalAll();
} finally {
lock.unlock();
}
}
}
}