summaryrefslogtreecommitdiffstats
path: root/server/src
diff options
context:
space:
mode:
authorLeif Åstrand <leif@vaadin.com>2015-06-19 15:34:29 +0300
committerVaadin Code Review <review@vaadin.com>2015-12-02 11:37:37 +0000
commit3cf57002b11ad400b8d4a33de1104b68a37e681a (patch)
treef01e0d9c88b7baaaacb31a1038a99840b805b627 /server/src
parent907e689a418f04013a58428e687ebc5132b0f45d (diff)
downloadvaadin-framework-3cf57002b11ad400b8d4a33de1104b68a37e681a.tar.gz
vaadin-framework-3cf57002b11ad400b8d4a33de1104b68a37e681a.zip
Detect hierarchy changes not sent to the client (#18317)
Change-Id: I77b420738738a42ff50e2a509e4ac4072b1b6e1f
Diffstat (limited to 'server/src')
-rw-r--r--server/src/com/vaadin/server/communication/ConnectorHierarchyWriter.java18
-rw-r--r--server/src/com/vaadin/ui/ConnectorTracker.java52
2 files changed, 70 insertions, 0 deletions
diff --git a/server/src/com/vaadin/server/communication/ConnectorHierarchyWriter.java b/server/src/com/vaadin/server/communication/ConnectorHierarchyWriter.java
index 503bf8c0ae..fe1cc0770c 100644
--- a/server/src/com/vaadin/server/communication/ConnectorHierarchyWriter.java
+++ b/server/src/com/vaadin/server/communication/ConnectorHierarchyWriter.java
@@ -26,6 +26,8 @@ import com.vaadin.server.AbstractClientConnector;
import com.vaadin.server.ClientConnector;
import com.vaadin.server.LegacyCommunicationManager;
import com.vaadin.server.PaintException;
+import com.vaadin.server.VaadinRequest;
+import com.vaadin.server.VaadinService;
import com.vaadin.ui.UI;
import elemental.json.Json;
@@ -87,6 +89,22 @@ public class ConnectorHierarchyWriter implements Serializable {
}
}
}
+ // Dummy assert just for conditionally storing away data that will be
+ // used by the real assert later on
+ assert storeSentHierarchy(hierarchyInfo);
+
writer.write(JsonUtil.stringify(hierarchyInfo));
}
+
+ private boolean storeSentHierarchy(JsonObject hierarchyInfo) {
+ VaadinRequest request = VaadinService.getCurrentRequest();
+ if (request != null) {
+ request.setAttribute(ConnectorHierarchyWriter.class.getName()
+ + ".hierarchyInfo", hierarchyInfo);
+ }
+
+ // Always true, we're just setting up for another assert
+ return true;
+ }
+
}
diff --git a/server/src/com/vaadin/ui/ConnectorTracker.java b/server/src/com/vaadin/ui/ConnectorTracker.java
index 84454f9126..a060702319 100644
--- a/server/src/com/vaadin/ui/ConnectorTracker.java
+++ b/server/src/com/vaadin/ui/ConnectorTracker.java
@@ -37,6 +37,9 @@ import com.vaadin.server.DragAndDropService;
import com.vaadin.server.GlobalResourceHandler;
import com.vaadin.server.LegacyCommunicationManager;
import com.vaadin.server.StreamVariable;
+import com.vaadin.server.VaadinRequest;
+import com.vaadin.server.VaadinService;
+import com.vaadin.server.communication.ConnectorHierarchyWriter;
import elemental.json.Json;
import elemental.json.JsonException;
@@ -326,6 +329,13 @@ public class ConnectorTracker implements Serializable {
.isConnectorVisibleToClient(connector)) {
uninitializedConnectors.add(connector);
diffStates.remove(connector);
+
+ assert isRemovalSentToClient(connector) : "Connector "
+ + connector
+ + " (id = "
+ + connector.getConnectorId()
+ + ") is no longer visible to the client, but no corresponding hierarchy change is being sent.";
+
if (getLogger().isLoggable(Level.FINE)) {
getLogger()
.log(Level.FINE,
@@ -338,6 +348,48 @@ public class ConnectorTracker implements Serializable {
cleanStreamVariables();
}
+ private boolean isRemovalSentToClient(ClientConnector connector) {
+ VaadinRequest request = VaadinService.getCurrentRequest();
+ if (request == null) {
+ // Probably run from a unit test without normal request handling
+ return true;
+ }
+
+ String attributeName = ConnectorHierarchyWriter.class.getName()
+ + ".hierarchyInfo";
+ Object hierarchyInfoObj = request.getAttribute(attributeName);
+ if (hierarchyInfoObj instanceof JsonObject) {
+ JsonObject hierachyInfo = (JsonObject) hierarchyInfoObj;
+
+ ClientConnector firstVisibleParent = findFirstVisibleParent(connector);
+ if (firstVisibleParent == null) {
+ // Connector is detached, not our business
+ return true;
+ }
+
+ if (!hierachyInfo.hasKey(firstVisibleParent.getConnectorId())) {
+ return false;
+ }
+ } else {
+ getLogger().warning(
+ "Request attribute " + attributeName
+ + " is not a JsonObject");
+ }
+
+ return true;
+ }
+
+ private ClientConnector findFirstVisibleParent(ClientConnector connector) {
+ while (connector != null) {
+ connector = connector.getParent();
+ if (LegacyCommunicationManager
+ .isConnectorVisibleToClient(connector)) {
+ return connector;
+ }
+ }
+ return null;
+ }
+
private void removeUnregisteredConnectors() {
GlobalResourceHandler globalResourceHandler = uI.getSession()
.getGlobalResourceHandler(false);