aboutsummaryrefslogtreecommitdiffstats
path: root/server
diff options
context:
space:
mode:
authorJohannes Dahlström <johannesd@vaadin.com>2014-05-19 16:28:09 +0300
committerArtur Signell <artur@vaadin.com>2014-05-26 13:18:00 +0000
commitdca9abccdddb5c1b39e9ae65b9cb53ad6fcab5cb (patch)
tree2ab384cecd52571388d7f2c4f113824093397f0b /server
parent17b5b7bf508586626da48e21b9c72694f03c7ff2 (diff)
downloadvaadin-framework-dca9abccdddb5c1b39e9ae65b9cb53ad6fcab5cb.tar.gz
vaadin-framework-dca9abccdddb5c1b39e9ae65b9cb53ad6fcab5cb.zip
Properly deserialize push connection (#12235)
The UI.pushConnection field is transient no longer; instead PushConnection implementations must take care of serialization internally. When a session is serialized, the client should notice that push was disconnected and try to reconnect. A deserialized PushConnection should be in a "disconnected" state so the eventual client reconnection works correctly. Change-Id: I38cfc5a5cdbd3643311f830f8d580458dcd85c56
Diffstat (limited to 'server')
-rw-r--r--server/src/com/vaadin/server/communication/AtmospherePushConnection.java29
-rw-r--r--server/src/com/vaadin/server/communication/PushConnection.java4
-rw-r--r--server/src/com/vaadin/ui/UI.java3
-rw-r--r--server/tests/src/com/vaadin/server/communication/AtmospherePushConnectionTest.java56
4 files changed, 84 insertions, 8 deletions
diff --git a/server/src/com/vaadin/server/communication/AtmospherePushConnection.java b/server/src/com/vaadin/server/communication/AtmospherePushConnection.java
index 3eae848adc..f8ef360eda 100644
--- a/server/src/com/vaadin/server/communication/AtmospherePushConnection.java
+++ b/server/src/com/vaadin/server/communication/AtmospherePushConnection.java
@@ -17,6 +17,7 @@
package com.vaadin.server.communication;
import java.io.IOException;
+import java.io.ObjectInputStream;
import java.io.Reader;
import java.io.Serializable;
import java.io.StringReader;
@@ -117,11 +118,11 @@ public class AtmospherePushConnection implements PushConnection {
CONNECTED;
}
- private State state = State.DISCONNECTED;
private UI ui;
- private AtmosphereResource resource;
- private FragmentedMessage incomingMessage;
- private Future<Object> outgoingMessage;
+ private transient State state = State.DISCONNECTED;
+ private transient AtmosphereResource resource;
+ private transient FragmentedMessage incomingMessage;
+ private transient Future<Object> outgoingMessage;
public AtmospherePushConnection(UI ui) {
this.ui = ui;
@@ -209,6 +210,7 @@ public class AtmospherePushConnection implements PushConnection {
@Override
public boolean isConnected() {
+ assert state != null;
assert (state == State.CONNECTED) ^ (resource == null);
return state == State.CONNECTED;
}
@@ -297,8 +299,25 @@ public class AtmospherePushConnection implements PushConnection {
state = State.DISCONNECTED;
}
+ /**
+ * Returns the state of this connection.
+ */
+ protected State getState() {
+ return state;
+ }
+
+ /**
+ * Reinitializes this PushConnection after deserialization. The connection
+ * is initially in disconnected state; the client will handle the
+ * reconnecting.
+ */
+ private void readObject(ObjectInputStream stream) throws IOException,
+ ClassNotFoundException {
+ stream.defaultReadObject();
+ state = State.DISCONNECTED;
+ }
+
private static Logger getLogger() {
return Logger.getLogger(AtmospherePushConnection.class.getName());
}
-
}
diff --git a/server/src/com/vaadin/server/communication/PushConnection.java b/server/src/com/vaadin/server/communication/PushConnection.java
index 52efcfcd89..d5a0d6e9bc 100644
--- a/server/src/com/vaadin/server/communication/PushConnection.java
+++ b/server/src/com/vaadin/server/communication/PushConnection.java
@@ -16,6 +16,8 @@
package com.vaadin.server.communication;
+import java.io.Serializable;
+
import com.vaadin.ui.UI;
/**
@@ -30,7 +32,7 @@ import com.vaadin.ui.UI;
* @author Vaadin Ltd
* @since 7.1
*/
-public interface PushConnection {
+public interface PushConnection extends Serializable {
/**
* Pushes pending state changes and client RPC calls to the client. Can be
diff --git a/server/src/com/vaadin/ui/UI.java b/server/src/com/vaadin/ui/UI.java
index 7a71083e85..562b30f81d 100644
--- a/server/src/com/vaadin/ui/UI.java
+++ b/server/src/com/vaadin/ui/UI.java
@@ -553,7 +553,7 @@ public abstract class UI extends AbstractSingleComponentContainer implements
private Navigator navigator;
- private transient PushConnection pushConnection = null;
+ private PushConnection pushConnection = null;
private LocaleService localeService = new LocaleService(this,
getState(false).localeServiceState);
@@ -1626,5 +1626,4 @@ public abstract class UI extends AbstractSingleComponentContainer implements
public String getEmbedId() {
return embedId;
}
-
}
diff --git a/server/tests/src/com/vaadin/server/communication/AtmospherePushConnectionTest.java b/server/tests/src/com/vaadin/server/communication/AtmospherePushConnectionTest.java
new file mode 100644
index 0000000000..f9e12ce75f
--- /dev/null
+++ b/server/tests/src/com/vaadin/server/communication/AtmospherePushConnectionTest.java
@@ -0,0 +1,56 @@
+/*
+ * Copyright 2000-2014 Vaadin Ltd.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License"); you may not
+ * use this file except in compliance with the License. You may obtain a copy of
+ * the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
+ * License for the specific language governing permissions and limitations under
+ * the License.
+ */
+package com.vaadin.server.communication;
+
+import java.io.ByteArrayInputStream;
+import java.io.ByteArrayOutputStream;
+import java.io.ObjectInputStream;
+import java.io.ObjectOutputStream;
+
+import org.atmosphere.cpr.AtmosphereResource;
+import org.easymock.EasyMock;
+import org.junit.Assert;
+import org.junit.Test;
+
+import com.vaadin.server.communication.AtmospherePushConnection.State;
+import com.vaadin.ui.UI;
+
+/**
+ * @author Vaadin Ltd
+ */
+public class AtmospherePushConnectionTest {
+ @Test
+ public void testSerialization() throws Exception {
+
+ UI ui = EasyMock.createNiceMock(UI.class);
+ AtmosphereResource resource = EasyMock
+ .createNiceMock(AtmosphereResource.class);
+
+ AtmospherePushConnection connection = new AtmospherePushConnection(ui);
+ connection.connect(resource);
+
+ Assert.assertEquals(State.CONNECTED, connection.getState());
+
+ ByteArrayOutputStream baos = new ByteArrayOutputStream();
+
+ new ObjectOutputStream(baos).writeObject(connection);
+
+ connection = (AtmospherePushConnection) new ObjectInputStream(
+ new ByteArrayInputStream(baos.toByteArray())).readObject();
+
+ Assert.assertEquals(State.DISCONNECTED, connection.getState());
+ }
+}