summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorLeif Åstrand <leif@vaadin.com>2013-07-17 15:15:21 +0300
committerLeif Åstrand <leif@vaadin.com>2013-07-17 16:21:26 +0300
commit2ea19f347b9c6aaa920c09b10747b644af4c2c45 (patch)
tree46be3598db46f7bf822f980d1f15f362251d4826
parent654d5707a98fb0e2309cc5aa5c8b6bbcf7d10491 (diff)
downloadvaadin-framework-2ea19f347b9c6aaa920c09b10747b644af4c2c45.tar.gz
vaadin-framework-2ea19f347b9c6aaa920c09b10747b644af4c2c45.zip
Make VaadinService.closeInactiveUIs set UI threadlocals (#12186)
Also update javadocs for methods called without UI threadlocals Also make sure pending access tasks are run for a UI being closed Change-Id: Ia600207b2d25fc195ee4254da371d420152bf100
-rw-r--r--server/src/com/vaadin/server/VaadinService.java17
-rw-r--r--server/src/com/vaadin/ui/UI.java36
2 files changed, 44 insertions, 9 deletions
diff --git a/server/src/com/vaadin/server/VaadinService.java b/server/src/com/vaadin/server/VaadinService.java
index 1be5bd5c99..f603fbd1e9 100644
--- a/server/src/com/vaadin/server/VaadinService.java
+++ b/server/src/com/vaadin/server/VaadinService.java
@@ -1136,13 +1136,18 @@ public abstract class VaadinService implements Serializable {
* @since 7.0.0
*/
private void closeInactiveUIs(VaadinSession session) {
- String sessionId = session.getSession().getId();
- for (UI ui : session.getUIs()) {
+ final String sessionId = session.getSession().getId();
+ for (final UI ui : session.getUIs()) {
if (!isUIActive(ui) && !ui.isClosing()) {
- getLogger().log(Level.FINE,
- "Closing inactive UI #{0} in session {1}",
- new Object[] { ui.getUIId(), sessionId });
- ui.close();
+ ui.accessSynchronously(new Runnable() {
+ @Override
+ public void run() {
+ getLogger().log(Level.FINE,
+ "Closing inactive UI #{0} in session {1}",
+ new Object[] { ui.getUIId(), sessionId });
+ ui.close();
+ }
+ });
}
}
}
diff --git a/server/src/com/vaadin/ui/UI.java b/server/src/com/vaadin/ui/UI.java
index 403bb31f63..0746431302 100644
--- a/server/src/com/vaadin/ui/UI.java
+++ b/server/src/com/vaadin/ui/UI.java
@@ -438,8 +438,12 @@ public abstract class UI extends AbstractSingleComponentContainer implements
* the server that originates from this UI.
* {@link VaadinService#findUI(VaadinRequest)} uses this id to find the
* route to which the request belongs.
+ * <p>
+ * This method is not intended to be overridden. If it is overridden, care
+ * should be taken since this method might be called in situations where
+ * {@link UI#getCurrent()} does not return this UI.
*
- * @return
+ * @return the id of this UI
*/
public int getUIId() {
return uiId;
@@ -1014,9 +1018,12 @@ public abstract class UI extends AbstractSingleComponentContainer implements
/**
* Returns the timestamp of the last received heartbeat for this UI.
+ * <p>
+ * This method is not intended to be overridden. If it is overridden, care
+ * should be taken since this method might be called in situations where
+ * {@link UI#getCurrent()} does not return this UI.
*
- * @see #heartbeat()
- * @see VaadinSession#cleanupInactiveUIs()
+ * @see VaadinService#closeInactiveUIs(VaadinSession)
*
* @return The time the last heartbeat request occurred, in milliseconds
* since the epoch.
@@ -1029,6 +1036,10 @@ public abstract class UI extends AbstractSingleComponentContainer implements
* Sets the last heartbeat request timestamp for this UI. Called by the
* framework whenever the application receives a valid heartbeat request for
* this UI.
+ * <p>
+ * This method is not intended to be overridden. If it is overridden, care
+ * should be taken since this method might be called in situations where
+ * {@link UI#getCurrent()} does not return this UI.
*
* @param lastHeartbeat
* The time the last heartbeat request occurred, in milliseconds
@@ -1072,6 +1083,11 @@ public abstract class UI extends AbstractSingleComponentContainer implements
if (getPushConnection() != null) {
// Push the Rpc to the client. The connection will be closed when
// the UI is detached and cleaned up.
+
+ // Can't use UI.push() directly since it checks for a valid session
+ if (session != null) {
+ session.getService().runPendingAccessTasks(session);
+ }
getPushConnection().push();
}
@@ -1079,6 +1095,10 @@ public abstract class UI extends AbstractSingleComponentContainer implements
/**
* Returns whether this UI is marked as closed and is to be detached.
+ * <p>
+ * This method is not intended to be overridden. If it is overridden, care
+ * should be taken since this method might be called in situations where
+ * {@link UI#getCurrent()} does not return this UI.
*
* @see #close()
*
@@ -1344,6 +1364,13 @@ public abstract class UI extends AbstractSingleComponentContainer implements
* Returns the internal push connection object used by this UI. This method
* should only be called by the framework. If the returned PushConnection is
* not null, it is guaranteed to have {@code isConnected() == true}.
+ * <p>
+ * This method is not intended to be overridden. If it is overridden, care
+ * should be taken since this method might be called in situations where
+ * {@link UI#getCurrent()} does not return this UI.
+ *
+ * @return the push connection used by this UI, <code>null</code> if there
+ * is no active push connection.
*/
public PushConnection getPushConnection() {
assert (pushConnection == null || pushConnection.isConnected());
@@ -1354,6 +1381,9 @@ public abstract class UI extends AbstractSingleComponentContainer implements
* Sets the internal push connection object used by this UI. This method
* should only be called by the framework. If {@pushConnection} is not null,
* its {@code isConnected()} must be true.
+ *
+ * @param pushConnection
+ * the push connection to use for this UI
*/
public void setPushConnection(PushConnection pushConnection) {
// If pushMode is disabled then there should never be a pushConnection