aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorTeemu Suo-Anttila <tsuoanttila@users.noreply.github.com>2017-11-27 13:15:56 +0200
committerAleksi Hietanen <aleksi@vaadin.com>2017-11-27 13:15:56 +0200
commitd936003c92a8e3ac2e1c0637ac240f3d7624bc0d (patch)
tree2c0e2f93caf60ed4590d0dee3fd0a4226088fc5e
parent780cab157d03a8ca81dec3667fe235ec10207efe (diff)
downloadvaadin-framework-d936003c92a8e3ac2e1c0637ac240f3d7624bc0d.tar.gz
vaadin-framework-d936003c92a8e3ac2e1c0637ac240f3d7624bc0d.zip
Prevent killing UI if heartbeats are pending (#10371)
Fixes #9663
-rw-r--r--server/src/main/java/com/vaadin/server/VaadinService.java23
-rw-r--r--uitest/src/main/java/com/vaadin/tests/core/LockingUI.java47
-rw-r--r--uitest/src/test/java/com/vaadin/tests/core/LockingUITest.java47
3 files changed, 112 insertions, 5 deletions
diff --git a/server/src/main/java/com/vaadin/server/VaadinService.java b/server/src/main/java/com/vaadin/server/VaadinService.java
index d086b68a12..9b9c7fd913 100644
--- a/server/src/main/java/com/vaadin/server/VaadinService.java
+++ b/server/src/main/java/com/vaadin/server/VaadinService.java
@@ -1363,12 +1363,25 @@ public abstract class VaadinService implements Serializable {
public boolean isUIActive(UI ui) {
if (ui.isClosing()) {
return false;
- } else {
- long now = System.currentTimeMillis();
- int timeout = 1000 * getHeartbeatTimeout();
- return timeout < 0
- || now - ui.getLastHeartbeatTimestamp() < timeout;
}
+
+ // Check for long running tasks
+ Lock lockInstance = ui.getSession().getLockInstance();
+ if (lockInstance instanceof ReentrantLock) {
+ if (((ReentrantLock) lockInstance).hasQueuedThreads()) {
+ /*
+ * Someone is trying to access the session. Leaving all UIs
+ * alive for now. A possible kill decision will be made at a
+ * later time when the session access has ended.
+ */
+ return true;
+ }
+ }
+
+ // Check timeout
+ long now = System.currentTimeMillis();
+ int timeout = 1000 * getHeartbeatTimeout();
+ return timeout < 0 || now - ui.getLastHeartbeatTimestamp() < timeout;
}
/**
diff --git a/uitest/src/main/java/com/vaadin/tests/core/LockingUI.java b/uitest/src/main/java/com/vaadin/tests/core/LockingUI.java
new file mode 100644
index 0000000000..ba0cf9e008
--- /dev/null
+++ b/uitest/src/main/java/com/vaadin/tests/core/LockingUI.java
@@ -0,0 +1,47 @@
+package com.vaadin.tests.core;
+
+import com.vaadin.launcher.CustomDeploymentConfiguration;
+import com.vaadin.launcher.CustomDeploymentConfiguration.Conf;
+import com.vaadin.server.VaadinRequest;
+import com.vaadin.server.VaadinService;
+import com.vaadin.ui.Button;
+import com.vaadin.ui.Notification;
+import com.vaadin.ui.Notification.Type;
+import com.vaadin.ui.UI;
+import com.vaadin.ui.VerticalLayout;
+
+@CustomDeploymentConfiguration({
+ @Conf(name = "heartbeatInterval", value = "2") })
+public class LockingUI extends UI {
+
+ public static final String LOCKING_ENDED = "Locking has ended";
+ public static final String ALL_OK = "All is fine";
+
+ @Override
+ protected void init(VaadinRequest request) {
+ Button lockButton = new Button("Lock UI for too long", e -> {
+ int heartbeatInterval = VaadinService.getCurrent()
+ .getDeploymentConfiguration().getHeartbeatInterval();
+ try {
+ // Wait for 4 heartbeats
+ long timeout = heartbeatInterval * 1000;
+ for (int i = 0; i < 4; ++i) {
+ Thread.sleep(timeout);
+ }
+
+ } catch (InterruptedException e1) {
+ throw new RuntimeException(
+ "Timeout should not get interrupted.");
+ }
+ Notification.show(LOCKING_ENDED, Type.TRAY_NOTIFICATION);
+ });
+ Button checkButton = new Button("Test communication",
+ e -> Notification.show(ALL_OK, Type.TRAY_NOTIFICATION));
+
+ lockButton.setId("lock");
+ checkButton.setId("check");
+
+ setContent(new VerticalLayout(lockButton, checkButton));
+ }
+
+}
diff --git a/uitest/src/test/java/com/vaadin/tests/core/LockingUITest.java b/uitest/src/test/java/com/vaadin/tests/core/LockingUITest.java
new file mode 100644
index 0000000000..214adfc94d
--- /dev/null
+++ b/uitest/src/test/java/com/vaadin/tests/core/LockingUITest.java
@@ -0,0 +1,47 @@
+package com.vaadin.tests.core;
+
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertFalse;
+import static org.junit.Assert.assertTrue;
+
+import org.junit.Test;
+
+import com.vaadin.testbench.elements.ButtonElement;
+import com.vaadin.testbench.elements.NotificationElement;
+import com.vaadin.tests.tb3.SingleBrowserTest;
+
+public class LockingUITest extends SingleBrowserTest {
+
+ @Test
+ public void testLockingTheUIFor4HeartBeats() {
+ openTestURL();
+
+ clickButtonAndCheckNotification("check", LockingUI.ALL_OK);
+ clickButtonAndCheckNotification("lock", LockingUI.LOCKING_ENDED);
+ clickButtonAndCheckNotification("check", LockingUI.ALL_OK);
+ }
+
+ private void clickButtonAndCheckNotification(String buttonId, String text) {
+ checkNoInitialNotification();
+
+ $(ButtonElement.class).id(buttonId).click();
+ testBench().waitForVaadin();
+
+ checkNotification(text);
+ }
+
+ private void checkNotification(String text) {
+ assertTrue("Notification should be displayed",
+ $(NotificationElement.class).exists());
+
+ NotificationElement notification = $(NotificationElement.class).first();
+ assertEquals("Unexpected text content in Notification", text,
+ notification.getText());
+ notification.close();
+ }
+
+ private void checkNoInitialNotification() {
+ assertFalse("Extra notification displayed",
+ $(NotificationElement.class).exists());
+ }
+}