IDEMPIERE-175 Performance: Use atmosphere ( long pooling, NIO ) server push

This commit is contained in:
Heng Sin Low 2012-04-29 04:38:26 +08:00
parent 409453dca8
commit 8bf2eb7614
13 changed files with 2064 additions and 84 deletions

View File

@ -17,6 +17,7 @@
package org.adempiere.util; package org.adempiere.util;
import java.io.Serializable;
import java.util.Properties; import java.util.Properties;
/** /**
@ -25,7 +26,7 @@ import java.util.Properties;
* @date Feb 25, 2007 * @date Feb 25, 2007
* @version $Revision: 0.10 $ * @version $Revision: 0.10 $
*/ */
public final class ServerContext public final class ServerContext implements Serializable
{ {
/** /**
* generated serial version Id * generated serial version Id

View File

@ -4,7 +4,6 @@
<classpathentry kind="con" path="org.eclipse.pde.core.requiredPlugins"/> <classpathentry kind="con" path="org.eclipse.pde.core.requiredPlugins"/>
<classpathentry kind="src" path="WEB-INF/src"/> <classpathentry kind="src" path="WEB-INF/src"/>
<classpathentry exported="true" kind="lib" path="WEB-INF/classes/" sourcepath="WEB-INF/src"/> <classpathentry exported="true" kind="lib" path="WEB-INF/classes/" sourcepath="WEB-INF/src"/>
<classpathentry kind="lib" path="WEB-INF/lib/slf4j-api-1.6.1.jar"/>
<classpathentry kind="lib" path="WEB-INF/lib/atmosphere-compat-jbossweb-0.9.jar"/> <classpathentry kind="lib" path="WEB-INF/lib/atmosphere-compat-jbossweb-0.9.jar"/>
<classpathentry kind="lib" path="WEB-INF/lib/atmosphere-compat-tomcat-0.9.jar"/> <classpathentry kind="lib" path="WEB-INF/lib/atmosphere-compat-tomcat-0.9.jar"/>
<classpathentry kind="lib" path="WEB-INF/lib/atmosphere-compat-tomcat7-0.9.jar" sourcepath="WEB-INF/lib/src/atmosphere-compat-tomcat7-0.9-sources.jar"/> <classpathentry kind="lib" path="WEB-INF/lib/atmosphere-compat-tomcat7-0.9.jar" sourcepath="WEB-INF/lib/src/atmosphere-compat-tomcat7-0.9-sources.jar"/>

View File

@ -10,13 +10,15 @@ Import-Package: javax.servlet,
org.apache.commons.codec.binary, org.apache.commons.codec.binary,
org.apache.ecs, org.apache.ecs,
org.apache.ecs.xhtml, org.apache.ecs.xhtml,
org.osgi.framework;version="1.5.0" org.osgi.framework;version="1.5.0",
org.slf4j;version="1.6.1",
org.slf4j.helpers;version="1.6.1",
org.slf4j.spi;version="1.6.1"
Bundle-ClassPath: WEB-INF/classes/, Bundle-ClassPath: WEB-INF/classes/,
WEB-INF/lib/atmosphere-runtime-0.9.jar, WEB-INF/lib/atmosphere-runtime-0.9.jar,
WEB-INF/lib/atmosphere-compat-jbossweb-0.9.jar, WEB-INF/lib/atmosphere-compat-jbossweb-0.9.jar,
WEB-INF/lib/atmosphere-compat-tomcat-0.9.jar, WEB-INF/lib/atmosphere-compat-tomcat-0.9.jar,
WEB-INF/lib/atmosphere-compat-tomcat7-0.9.jar, WEB-INF/lib/atmosphere-compat-tomcat7-0.9.jar
WEB-INF/lib/slf4j-api-1.6.1.jar
Export-Package: metainfo.zk, Export-Package: metainfo.zk,
org.adempiere.webui, org.adempiere.webui,
org.adempiere.webui.acct, org.adempiere.webui.acct,
@ -40,24 +42,7 @@ Export-Package: metainfo.zk,
org.adempiere.webui.session, org.adempiere.webui.session,
org.adempiere.webui.theme, org.adempiere.webui.theme,
org.adempiere.webui.util, org.adempiere.webui.util,
org.adempiere.webui.window, org.adempiere.webui.window
org.atmosphere.cache,
org.atmosphere.client,
org.atmosphere.config,
org.atmosphere.container,
org.atmosphere.container.version,
org.atmosphere.cpr,
org.atmosphere.di,
org.atmosphere.handler,
org.atmosphere.util,
org.atmosphere.util.uri,
org.atmosphere.websocket,
org.atmosphere.websocket.protocol,
org.jboss.servlet.http,
org.apache.catalina,
org.apache.catalina.comet,
org.slf4j.helpers,
org.slf4j.spi
Require-Bundle: org.adempiere.report.jasper;bundle-version="1.0.0", Require-Bundle: org.adempiere.report.jasper;bundle-version="1.0.0",
org.adempiere.base;bundle-version="1.0.0", org.adempiere.base;bundle-version="1.0.0",
org.adempiere.report.jasper.library;bundle-version="1.0.0", org.adempiere.report.jasper.library;bundle-version="1.0.0",

View File

@ -19,6 +19,7 @@ the License.
*/ */
package fi.jawsy.jawwa.zk.atmosphere; package fi.jawsy.jawwa.zk.atmosphere;
import java.io.IOException;
import java.util.concurrent.atomic.AtomicReference; import java.util.concurrent.atomic.AtomicReference;
import org.atmosphere.cpr.AtmosphereResource; import org.atmosphere.cpr.AtmosphereResource;
@ -112,17 +113,35 @@ public class AtmosphereServerPush implements ServerPush {
return true; return true;
} }
public void clearResource(AtmosphereResource resource) { public synchronized void clearResource(AtmosphereResource resource) {
this.resource.compareAndSet(resource, null); this.resource.compareAndSet(resource, null);
} }
private void commitResponse() { private synchronized void commitResponse() throws IOException {
AtmosphereResource resource = this.resource.getAndSet(null); AtmosphereResource resource = this.resource.getAndSet(null);
if (resource != null) { if (resource != null) {
resource.resume(); resource.resume();
} }
} }
private synchronized void onPush() throws IOException {
AtmosphereResource resource = this.resource.get();
if (resource != null) {
switch (resource.transport()) {
case JSONP:
case LONG_POLLING:
if (resource.isSuspended())
commitResponse();
break;
case WEBSOCKET :
case STREAMING:
resource.getResponse().getWriter().write("@");
resource.getResponse().getWriter().flush();
break;
}
}
}
@Override @Override
public boolean deactivate(boolean stop) { public boolean deactivate(boolean stop) {
boolean stopped = false; boolean stopped = false;
@ -157,10 +176,17 @@ public class AtmosphereServerPush implements ServerPush {
} }
@Override @Override
public <T extends Event> void schedule(EventListener<T> task, T event, public synchronized <T extends Event> void schedule(EventListener<T> task, T event,
Scheduler<T> scheduler) { Scheduler<T> scheduler) {
boolean pendingPush = ((DesktopCtrl)desktop.get()).scheduledServerPush();
scheduler.schedule(task, event); scheduler.schedule(task, event);
commitResponse(); if (!pendingPush || (this.resource.get() != null && this.resource.get().isSuspended())) {
try {
onPush();
} catch (IOException e) {
log.warn(e.getLocalizedMessage(), e);
}
}
} }
@Override @Override
@ -187,32 +213,40 @@ public class AtmosphereServerPush implements ServerPush {
log.debug("Stopping server push for " + desktop); log.debug("Stopping server push for " + desktop);
Clients.response("jawwa.atmosphere.serverpush", new AuScript(null, "jawwa.atmosphere.stopServerPush('" + desktop.getId() + "');")); Clients.response("jawwa.atmosphere.serverpush", new AuScript(null, "jawwa.atmosphere.stopServerPush('" + desktop.getId() + "');"));
commitResponse(); try {
commitResponse();
} catch (IOException e) {
}
} }
public void updateResource(AtmosphereResource resource) { public synchronized void onRequest(AtmosphereResource resource) {
commitResponse(); if (this.resource.get() != null) {
AtmosphereResource aResource = this.resource.get();
if (aResource.isSuspended()) {
try {
commitResponse();
} catch (IOException e) {
e.printStackTrace();
}
}
}
boolean shouldSuspend = true; Desktop desktop = this.desktop.get();
Desktop desktop = this.desktop.get(); if (desktop != null && desktop instanceof DesktopCtrl)
if (desktop == null) { {
return; if (((DesktopCtrl)desktop).scheduledServerPush())
} {
return;
}
}
if (desktop instanceof DesktopCtrl) { this.resource.set(resource);
DesktopCtrl desktopCtrl = (DesktopCtrl) desktop; if (!resource.isSuspended()) {
shouldSuspend = !desktopCtrl.scheduledServerPush(); resource.suspend(-1, true);
} }
if (shouldSuspend) {
resource.suspend(timeout, false);
this.resource.set(resource);
} else {
this.resource.set(null);
}
} }
private static class ThreadInfo { private static class ThreadInfo {
private final Thread thread; private final Thread thread;
/** # of activate() was called. */ /** # of activate() was called. */
private int nActive; private int nActive;

View File

@ -118,11 +118,12 @@ public class ZkAtmosphereHandler implements AtmosphereHandler {
if (error != null && serverPushEither.getRightValue() == null) { if (error != null && serverPushEither.getRightValue() == null) {
response.setStatus(HttpServletResponse.SC_BAD_REQUEST); response.setStatus(HttpServletResponse.SC_BAD_REQUEST);
response.getWriter().write(error); response.getWriter().write(error);
response.getWriter().flush();
return; return;
} }
AtmosphereServerPush serverPush = serverPushEither.getRightValue(); AtmosphereServerPush serverPush = serverPushEither.getRightValue();
serverPush.updateResource(resource); serverPush.onRequest(resource);
} }
@Override @Override

View File

@ -24,6 +24,7 @@ import java.util.Properties;
import javax.servlet.ServletRequest; import javax.servlet.ServletRequest;
import javax.servlet.http.HttpSession; import javax.servlet.http.HttpSession;
import org.adempiere.util.ServerContext;
import org.adempiere.webui.apps.AEnv; import org.adempiere.webui.apps.AEnv;
import org.adempiere.webui.component.DrillCommand; import org.adempiere.webui.component.DrillCommand;
import org.adempiere.webui.component.TokenCommand; import org.adempiere.webui.component.TokenCommand;
@ -127,6 +128,8 @@ public class AdempiereWebUI extends Window implements EventListener<Event>, IWeb
loginCompleted(); loginCompleted();
} }
Executions.getCurrent().getDesktop().enableServerPush(true);
Executions.getCurrent().getDesktop().addListener(new DrillCommand()); Executions.getCurrent().getDesktop().addListener(new DrillCommand());
Executions.getCurrent().getDesktop().addListener(new TokenCommand()); Executions.getCurrent().getDesktop().addListener(new TokenCommand());
Executions.getCurrent().getDesktop().addListener(new ZoomCommand()); Executions.getCurrent().getDesktop().addListener(new ZoomCommand());
@ -283,6 +286,9 @@ public class AdempiereWebUI extends Window implements EventListener<Event>, IWeb
ctx.put(ZK_DESKTOP_SESSION_KEY, this.getPage().getDesktop()); ctx.put(ZK_DESKTOP_SESSION_KEY, this.getPage().getDesktop());
} }
//update session context
currSess.setAttribute(SessionContextListener.SESSION_CTX, ServerContext.getCurrentInstance());
if ("Y".equalsIgnoreCase(Env.getContext(ctx, BrowserToken.REMEMBER_ME)) && MSystem.isZKRememberUserAllowed()) if ("Y".equalsIgnoreCase(Env.getContext(ctx, BrowserToken.REMEMBER_ME)) && MSystem.isZKRememberUserAllowed())
{ {
MUser user = MUser.get(ctx); MUser user = MUser.get(ctx);

View File

@ -41,7 +41,7 @@ import org.zkoss.zul.Vbox;
* Contributors: * Contributors:
* CarlosRuiz - globalqss - Add unprocessed button to iDempiere * CarlosRuiz - globalqss - Add unprocessed button to iDempiere
*/ */
public class DPActivities extends DashboardPanel implements EventListener { public class DPActivities extends DashboardPanel implements EventListener<Event> {
/** /**
* *

View File

@ -59,7 +59,7 @@ public class SessionContextListener implements ExecutionInit,
* @param exec * @param exec
* @param createNew * @param createNew
*/ */
private void setupExecutionContextFromSession(Execution exec) { public static void setupExecutionContextFromSession(Execution exec) {
Session session = exec.getDesktop().getSession(); Session session = exec.getDesktop().getSession();
Properties ctx = (Properties)session.getAttribute(SESSION_CTX); Properties ctx = (Properties)session.getAttribute(SESSION_CTX);
HttpSession httpSession = (HttpSession)session.getNativeSession(); HttpSession httpSession = (HttpSession)session.getNativeSession();
@ -115,7 +115,7 @@ public class SessionContextListener implements ExecutionInit,
* @param errs * @param errs
* @see ExecutionCleanup#cleanup(Execution, Execution, List) * @see ExecutionCleanup#cleanup(Execution, Execution, List)
*/ */
public void cleanup(Execution exec, Execution parent, List errs) public void cleanup(Execution exec, Execution parent, List<Throwable> errs)
{ {
//in servlet thread //in servlet thread
if (parent == null) if (parent == null)
@ -260,12 +260,12 @@ public class SessionContextListener implements ExecutionInit,
* @param errs * @param errs
* @see EventThreadCleanup#cleanup(Component, Event, List) * @see EventThreadCleanup#cleanup(Component, Event, List)
*/ */
public void cleanup(Component comp, Event evt, List errs) throws Exception public void cleanup(Component comp, Event evt, List<Throwable> errs) throws Exception
{ {
//in event processing thread //in event processing thread
} }
private boolean isContextValid() { public static boolean isContextValid() {
Execution exec = Executions.getCurrent(); Execution exec = Executions.getCurrent();
Properties ctx = ServerContext.getCurrentInstance(); Properties ctx = ServerContext.getCurrentInstance();
if (ctx == null) if (ctx == null)
@ -278,6 +278,24 @@ public class SessionContextListener implements ExecutionInit,
{ {
return false; return false;
} }
Properties sessionCtx = (Properties) session.getAttribute(SESSION_CTX);
if (sessionCtx != null)
{
if (Env.getAD_Client_ID(sessionCtx) != Env.getAD_Client_ID(ctx))
{
return false;
}
if (Env.getAD_User_ID(sessionCtx) != Env.getAD_User_ID(ctx))
{
return false;
}
if (Env.getAD_Role_ID(sessionCtx) != Env.getAD_Role_ID(ctx))
{
return false;
}
}
return true; return true;
} }
} }

View File

@ -14,6 +14,7 @@
package org.adempiere.webui.util; package org.adempiere.webui.util;
import org.adempiere.exceptions.AdempiereException; import org.adempiere.exceptions.AdempiereException;
import org.adempiere.webui.session.SessionContextListener;
import org.zkoss.zk.ui.Desktop; import org.zkoss.zk.ui.Desktop;
import org.zkoss.zk.ui.DesktopUnavailableException; import org.zkoss.zk.ui.DesktopUnavailableException;
import org.zkoss.zk.ui.Executions; import org.zkoss.zk.ui.Executions;
@ -48,6 +49,9 @@ public class ServerPushTemplate {
EventListener<Event> task = new EventListener<Event>() { EventListener<Event> task = new EventListener<Event>() {
@Override @Override
public void onEvent(Event event) throws Exception { public void onEvent(Event event) throws Exception {
if (!SessionContextListener.isContextValid()) {
SessionContextListener.setupExecutionContextFromSession(desktop.getExecution());
}
callback.updateUI(); callback.updateUI();
} }
}; };

File diff suppressed because it is too large Load Diff

View File

@ -18,6 +18,7 @@
timeout: 300000, timeout: 300000,
delay: 1000, delay: 1000,
failures: 0, failures: 0,
count: 0,
$init: function(desktop, timeout) { $init: function(desktop, timeout) {
this.desktop = desktop; this.desktop = desktop;
@ -36,31 +37,41 @@
return; return;
var me = this; var me = this;
var jqxhr = $.ajax({
url: zk.ajaxURI("/comet", { var socket = $.atmosphere;
au: true var request = {
}), url: zk.ajaxURI("/comet", {
type: "GET", au: true
cache: false, }),
async: true, logLevel : 'debug',
global: false, transport : 'streaming',
data: { fallbackTransport: 'long-polling',
dtid: this.desktop.id method: "GET",
}, cache: false,
dataType: "", async: true,
timeout: me.timeout, timeout: me.timeout,
transport : 'long-polling', onError: function(response) {
error: function(jqxhr, textStatus, errorThrown) { me.failures += 1;
me.failures += 1; me.count--;
me._schedule(); if (response.transport == 'long-polling' && me.count == 0) {
}, me._schedule();
success: function(data) { } else if (me.failures >= 10) {
zAu.cmd0.echo(me.desktop); me.stop();
me.failures = 0; }
me._schedule(); },
} onMessage: function(response) {
}); zAu.cmd0.echo(me.desktop);
this._req = jqxhr; me.failures = 0;
me.count--;
if (response.transport == 'long-polling' && me.count == 0) {
me._schedule();
}
}
};
request.url = request.url+'?dtid='+me.desktop.id;
this.count++;
socket.subscribe(request);
}, },
start: function() { start: function() {
this.desktop._serverpush = this; this.desktop._serverpush = this;

View File

@ -1,4 +1,5 @@
<?xml version="1.0" encoding="UTF-8"?> <?xml version="1.0" encoding="UTF-8"?>
<package name="jawwa.atmosphere" language="xul/html"> <package name="jawwa.atmosphere" language="xul/html">
<script src="jquery.atmosphere.js" />
<script src="serverpush.js" /> <script src="serverpush.js" />
</package> </package>

View File

@ -14,8 +14,7 @@ bin.includes = META-INF/,\
WEB-INF/lib/atmosphere-runtime-0.9.jar,\ WEB-INF/lib/atmosphere-runtime-0.9.jar,\
WEB-INF/lib/atmosphere-compat-jbossweb-0.9.jar,\ WEB-INF/lib/atmosphere-compat-jbossweb-0.9.jar,\
WEB-INF/lib/atmosphere-compat-tomcat-0.9.jar,\ WEB-INF/lib/atmosphere-compat-tomcat-0.9.jar,\
WEB-INF/lib/atmosphere-compat-tomcat7-0.9.jar,\ WEB-INF/lib/atmosphere-compat-tomcat7-0.9.jar
WEB-INF/lib/slf4j-api-1.6.1.jar
src.includes = WEB-INF/classes/,\ src.includes = WEB-INF/classes/,\
WEB-INF/tld/,\ WEB-INF/tld/,\
WEB-INF/web.xml,\ WEB-INF/web.xml,\