IDEMPIERE-5049 Zk Session and Desktop object not destroy immediately after logout (#991)
This commit is contained in:
parent
2486ff8148
commit
07202f7b61
|
@ -24,6 +24,7 @@ import java.util.Properties;
|
||||||
import java.util.TimeZone;
|
import java.util.TimeZone;
|
||||||
import java.util.concurrent.ConcurrentHashMap;
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
import java.util.concurrent.ConcurrentMap;
|
import java.util.concurrent.ConcurrentMap;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
import javax.servlet.ServletRequest;
|
import javax.servlet.ServletRequest;
|
||||||
import javax.servlet.http.HttpServletRequest;
|
import javax.servlet.http.HttpServletRequest;
|
||||||
|
@ -44,6 +45,7 @@ import org.adempiere.webui.theme.ITheme;
|
||||||
import org.adempiere.webui.theme.ThemeManager;
|
import org.adempiere.webui.theme.ThemeManager;
|
||||||
import org.adempiere.webui.util.BrowserToken;
|
import org.adempiere.webui.util.BrowserToken;
|
||||||
import org.adempiere.webui.util.UserPreference;
|
import org.adempiere.webui.util.UserPreference;
|
||||||
|
import org.compiere.Adempiere;
|
||||||
import org.compiere.model.MRole;
|
import org.compiere.model.MRole;
|
||||||
import org.compiere.model.MSession;
|
import org.compiere.model.MSession;
|
||||||
import org.compiere.model.MSysConfig;
|
import org.compiere.model.MSysConfig;
|
||||||
|
@ -65,10 +67,14 @@ import org.zkoss.zk.ui.Desktop;
|
||||||
import org.zkoss.zk.ui.Executions;
|
import org.zkoss.zk.ui.Executions;
|
||||||
import org.zkoss.zk.ui.Page;
|
import org.zkoss.zk.ui.Page;
|
||||||
import org.zkoss.zk.ui.Session;
|
import org.zkoss.zk.ui.Session;
|
||||||
|
import org.zkoss.zk.ui.WebApp;
|
||||||
import org.zkoss.zk.ui.event.ClientInfoEvent;
|
import org.zkoss.zk.ui.event.ClientInfoEvent;
|
||||||
import org.zkoss.zk.ui.event.Event;
|
import org.zkoss.zk.ui.event.Event;
|
||||||
import org.zkoss.zk.ui.event.EventListener;
|
import org.zkoss.zk.ui.event.EventListener;
|
||||||
import org.zkoss.zk.ui.event.Events;
|
import org.zkoss.zk.ui.event.Events;
|
||||||
|
import org.zkoss.zk.ui.sys.DesktopCache;
|
||||||
|
import org.zkoss.zk.ui.sys.SessionCtrl;
|
||||||
|
import org.zkoss.zk.ui.sys.WebAppCtrl;
|
||||||
import org.zkoss.zk.ui.util.Clients;
|
import org.zkoss.zk.ui.util.Clients;
|
||||||
import org.zkoss.zul.Window;
|
import org.zkoss.zul.Window;
|
||||||
|
|
||||||
|
@ -374,21 +380,40 @@ public class AdempiereWebUI extends Window implements EventListener<Event>, IWeb
|
||||||
*/
|
*/
|
||||||
public void logout()
|
public void logout()
|
||||||
{
|
{
|
||||||
Desktop desktop = Executions.getCurrent().getDesktop();
|
final Desktop desktop = Executions.getCurrent().getDesktop();
|
||||||
if (desktop.isServerPushEnabled())
|
if (desktop.isServerPushEnabled())
|
||||||
desktop.enableServerPush(false);
|
desktop.enableServerPush(false);
|
||||||
|
|
||||||
Session session = logout0();
|
final WebApp wapp = desktop.getWebApp();
|
||||||
|
final DesktopCache desktopCache = ((WebAppCtrl) wapp).getDesktopCache(desktop.getSession());
|
||||||
|
Adempiere.getThreadPoolExecutor().schedule(() -> {
|
||||||
|
try {
|
||||||
|
desktopCache.removeDesktop(desktop);
|
||||||
|
} catch (Throwable t) {
|
||||||
|
t.printStackTrace();
|
||||||
|
}
|
||||||
|
}, 5, TimeUnit.SECONDS);
|
||||||
|
|
||||||
|
final Session session = logout0();
|
||||||
|
|
||||||
//clear context, invalidate session
|
//clear context, invalidate session
|
||||||
Env.getCtx().clear();
|
Env.getCtx().clear();
|
||||||
session.invalidate();
|
destroySession(session);
|
||||||
desktop.setAttribute(DESKTOP_SESSION_INVALIDATED_ATTR, Boolean.TRUE);
|
desktop.setAttribute(DESKTOP_SESSION_INVALIDATED_ATTR, Boolean.TRUE);
|
||||||
|
|
||||||
//redirect to login page
|
//redirect to login page
|
||||||
Executions.sendRedirect("index.zul");
|
Executions.sendRedirect("index.zul");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void destroySession(final Session session) {
|
||||||
|
try {
|
||||||
|
((SessionCtrl)session).onDestroyed();
|
||||||
|
} catch (Throwable t) {
|
||||||
|
t.printStackTrace();
|
||||||
|
}
|
||||||
|
session.invalidate();
|
||||||
|
}
|
||||||
|
|
||||||
public void logoutAfterTabDestroyed(){
|
public void logoutAfterTabDestroyed(){
|
||||||
Desktop desktop = Executions.getCurrent().getDesktop();
|
Desktop desktop = Executions.getCurrent().getDesktop();
|
||||||
if (desktop.isServerPushEnabled())
|
if (desktop.isServerPushEnabled())
|
||||||
|
@ -398,7 +423,7 @@ public class AdempiereWebUI extends Window implements EventListener<Event>, IWeb
|
||||||
|
|
||||||
//clear context, invalidate session
|
//clear context, invalidate session
|
||||||
Env.getCtx().clear();
|
Env.getCtx().clear();
|
||||||
session.invalidate();
|
destroySession(session);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,157 @@
|
||||||
|
/***********************************************************************
|
||||||
|
* This file is part of iDempiere ERP Open Source *
|
||||||
|
* http://www.idempiere.org *
|
||||||
|
* *
|
||||||
|
* Copyright (C) Contributors *
|
||||||
|
* *
|
||||||
|
* This program is free software; you can redistribute it and/or *
|
||||||
|
* modify it under the terms of the GNU General Public License *
|
||||||
|
* as published by the Free Software Foundation; either version 2 *
|
||||||
|
* of the License, or (at your option) any later version. *
|
||||||
|
* *
|
||||||
|
* This program is distributed in the hope that it will be useful, *
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
||||||
|
* GNU General Public License for more details. *
|
||||||
|
* *
|
||||||
|
* You should have received a copy of the GNU General Public License *
|
||||||
|
* along with this program; if not, write to the Free Software *
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, *
|
||||||
|
* MA 02110-1301, USA. *
|
||||||
|
* *
|
||||||
|
* Contributors: *
|
||||||
|
* - hengsin *
|
||||||
|
**********************************************************************/
|
||||||
|
package org.adempiere.webui.util;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.atomic.LongAdder;
|
||||||
|
|
||||||
|
import org.zkoss.zk.au.AuRequest;
|
||||||
|
import org.zkoss.zk.ui.Desktop;
|
||||||
|
import org.zkoss.zk.ui.Session;
|
||||||
|
import org.zkoss.zk.ui.util.Monitor;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adapted from {@link org.zkoss.zk.ui.util.Statistic}
|
||||||
|
* @author hengsin
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class Statistic implements Monitor {
|
||||||
|
private final long _startTime;
|
||||||
|
private LongAdder _nsess = new LongAdder();
|
||||||
|
private LongAdder _actsess = new LongAdder();
|
||||||
|
private LongAdder _ndt = new LongAdder();
|
||||||
|
private LongAdder _actdt = new LongAdder();
|
||||||
|
private LongAdder _nupd = new LongAdder();
|
||||||
|
private LongAdder _actupd = new LongAdder();;
|
||||||
|
|
||||||
|
public Statistic() {
|
||||||
|
_startTime = System.currentTimeMillis();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Returns when the server (actually, this monitor) started.
|
||||||
|
*/
|
||||||
|
public long getStartTime() {
|
||||||
|
return _startTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Returns the total number of sessions that have been created
|
||||||
|
* since the server started.
|
||||||
|
*/
|
||||||
|
public long getTotalSessionCount() {
|
||||||
|
return _nsess.longValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Returns the number of active sessions.
|
||||||
|
*/
|
||||||
|
public long getActiveSessionCount() {
|
||||||
|
return _actsess.longValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Returns the average number of sessions being created in an hour.
|
||||||
|
*/
|
||||||
|
public double getAverageSessionCount() {
|
||||||
|
return _nsess.longValue() / getEscapedHours();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Returns the total number of desktops that have been created
|
||||||
|
* since the server started.
|
||||||
|
*/
|
||||||
|
public long getTotalDesktopCount() {
|
||||||
|
return _ndt.longValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Returns the number of active desktops.
|
||||||
|
*/
|
||||||
|
public long getActiveDesktopCount() {
|
||||||
|
return _actdt.longValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Returns the average number of desktops being created in an hour.
|
||||||
|
*/
|
||||||
|
public double getAverageDesktopCount() {
|
||||||
|
return _ndt.longValue() / getEscapedHours();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Returns the total number of asynchronous updates that have been received
|
||||||
|
* since the server started.
|
||||||
|
*/
|
||||||
|
public long getTotalUpdateCount() {
|
||||||
|
return _nupd.longValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Returns the number of active asynchronous updates.
|
||||||
|
*/
|
||||||
|
public long getActiveUpdateCount() {
|
||||||
|
return _actupd.longValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Returns the average number of asynchronous updates being created
|
||||||
|
* in an hour.
|
||||||
|
*/
|
||||||
|
public double getAverageUpdateCount() {
|
||||||
|
return _nupd.longValue() / getEscapedHours();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Returns how many hours escaped since the server started.
|
||||||
|
*/
|
||||||
|
private double getEscapedHours() {
|
||||||
|
long v = System.currentTimeMillis() - _startTime;
|
||||||
|
return ((double) v) / 3600000;
|
||||||
|
}
|
||||||
|
|
||||||
|
//-- Monitor --//
|
||||||
|
@Override
|
||||||
|
public void sessionCreated(Session sess) {
|
||||||
|
_nsess.increment();
|
||||||
|
_actsess.increment();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void sessionDestroyed(Session sess) {
|
||||||
|
_actsess.decrement();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void desktopCreated(Desktop desktop) {
|
||||||
|
_ndt.increment();
|
||||||
|
_actdt.increment();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void desktopDestroyed(Desktop desktop) {
|
||||||
|
_actdt.decrement();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void beforeUpdate(Desktop desktop, List<AuRequest> requests) {
|
||||||
|
_nupd.increment();
|
||||||
|
_actupd.increment();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void afterUpdate(Desktop desktop) {
|
||||||
|
_actupd.decrement();
|
||||||
|
}
|
||||||
|
}
|
|
@ -44,8 +44,10 @@ import org.adempiere.webui.event.DialogEvents;
|
||||||
import org.adempiere.webui.factory.ButtonFactory;
|
import org.adempiere.webui.factory.ButtonFactory;
|
||||||
import org.adempiere.webui.theme.ThemeManager;
|
import org.adempiere.webui.theme.ThemeManager;
|
||||||
import org.adempiere.webui.util.FeedbackManager;
|
import org.adempiere.webui.util.FeedbackManager;
|
||||||
|
import org.adempiere.webui.util.Statistic;
|
||||||
import org.adempiere.webui.util.ZKUpdateUtil;
|
import org.adempiere.webui.util.ZKUpdateUtil;
|
||||||
import org.compiere.Adempiere;
|
import org.compiere.Adempiere;
|
||||||
|
import org.compiere.model.MRole;
|
||||||
import org.compiere.model.MUser;
|
import org.compiere.model.MUser;
|
||||||
import org.compiere.util.CLogErrorBuffer;
|
import org.compiere.util.CLogErrorBuffer;
|
||||||
import org.compiere.util.CLogMgt;
|
import org.compiere.util.CLogMgt;
|
||||||
|
@ -62,6 +64,7 @@ import org.zkoss.zhtml.Textarea;
|
||||||
import org.zkoss.zk.ui.event.Event;
|
import org.zkoss.zk.ui.event.Event;
|
||||||
import org.zkoss.zk.ui.event.EventListener;
|
import org.zkoss.zk.ui.event.EventListener;
|
||||||
import org.zkoss.zk.ui.event.Events;
|
import org.zkoss.zk.ui.event.Events;
|
||||||
|
import org.zkoss.zk.ui.util.Monitor;
|
||||||
import org.zkoss.zul.Borderlayout;
|
import org.zkoss.zul.Borderlayout;
|
||||||
import org.zkoss.zul.Center;
|
import org.zkoss.zul.Center;
|
||||||
import org.zkoss.zul.Div;
|
import org.zkoss.zul.Div;
|
||||||
|
@ -338,6 +341,21 @@ public class AboutWindow extends Window implements EventListener<Event> {
|
||||||
Text text = new Text(CLogMgt.getInfo(null).toString());
|
Text text = new Text(CLogMgt.getInfo(null).toString());
|
||||||
text.setParent(pre);
|
text.setParent(pre);
|
||||||
|
|
||||||
|
if (Env.getAD_Client_ID(Env.getCtx())==0 && MRole.getDefault().isAccessAdvanced()) {
|
||||||
|
addCallback(AFTER_PAGE_ATTACHED, t-> {
|
||||||
|
Monitor monitor = getDesktop().getWebApp().getConfiguration().getMonitor();
|
||||||
|
if (monitor != null && monitor instanceof Statistic) {
|
||||||
|
Statistic stat = (Statistic) monitor;
|
||||||
|
StringBuilder info = new StringBuilder(text.getValue());
|
||||||
|
info.append("\n");
|
||||||
|
info.append("Desktop: ").append("#Created=").append(stat.getTotalDesktopCount()).append(" #Active=").append(stat.getActiveDesktopCount());
|
||||||
|
info.append("\n");
|
||||||
|
info.append("Session: ").append("#Created=").append(stat.getTotalSessionCount()).append(" #Active=").append(stat.getActiveSessionCount());
|
||||||
|
text.setValue(info.toString());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
return tabPanel;
|
return tabPanel;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -23,6 +23,11 @@
|
||||||
<listener-class>org.adempiere.webui.util.LogEventInterceptor</listener-class>
|
<listener-class>org.adempiere.webui.util.LogEventInterceptor</listener-class>
|
||||||
</listener -->
|
</listener -->
|
||||||
|
|
||||||
|
<listener>
|
||||||
|
<description>[Optional] Mointor the statistic</description>
|
||||||
|
<listener-class>org.adempiere.webui.util.Statistic</listener-class>
|
||||||
|
</listener>
|
||||||
|
|
||||||
<!-- false to use compress js which is much smaller. change to true if you need to debug -->
|
<!-- false to use compress js which is much smaller. change to true if you need to debug -->
|
||||||
<client-config>
|
<client-config>
|
||||||
<processing-prompt-delay>500</processing-prompt-delay>
|
<processing-prompt-delay>500</processing-prompt-delay>
|
||||||
|
|
Loading…
Reference in New Issue