/*
* Copyright (c) 2008-2017 the original author or authors.
*
* 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 org.cometd.javascript.extension;
import org.cometd.javascript.AbstractCometDTest;
import org.cometd.javascript.Latch;
import org.cometd.server.AbstractService;
import org.cometd.server.BayeuxServerImpl;
import org.cometd.server.ext.AcknowledgedMessagesExtension;
import org.junit.Assert;
import org.junit.Before;
import org.junit.Test;
public class CometDAckExtensionTest extends AbstractCometDTest {
private AckService ackService;
@Before
public void initExtension() throws Exception {
bayeuxServer.addExtension(new AcknowledgedMessagesExtension());
ackService = new AckService(bayeuxServer);
}
@Test
public void testClientSupportsAckExtension() throws Exception {
defineClass(Latch.class);
evaluateScript("var cometd = cometd;");
evaluateScript("cometd.configure({url: '" + cometdURL + "', logLevel: '" + getLogLevel() + "'});");
// Check that during handshake the ack extension capability is sent to server
evaluateScript("var clientSupportsAck = false;");
evaluateScript("cometd.registerExtension('test', {" +
"outgoing: function(message)" +
"{" +
" if (message.channel == '/meta/handshake')" +
" {" +
" clientSupportsAck = message.ext && message.ext.ack;" +
" }" +
" return message;" +
"}" +
"});");
provideMessageAcknowledgeExtension();
evaluateScript("var readyLatch = new Latch(1);");
Latch readyLatch = get("readyLatch");
evaluateScript("cometd.addListener('/meta/handshake', function(message) { readyLatch.countDown(); });");
evaluateScript("cometd.handshake();");
Assert.assertTrue(readyLatch.await(5000));
Boolean clientSupportsAck = get("clientSupportsAck");
Assert.assertTrue(clientSupportsAck);
evaluateScript("cometd.unregisterExtension('test');");
evaluateScript("cometd.disconnect(true);");
}
@Test
public void testAcknowledgement() throws Exception {
defineClass(Latch.class);
evaluateScript("cometd.configure({url: '" + cometdURL + "', logLevel: '" + getLogLevel() + "'});");
evaluateScript("var inAckId = undefined;");
evaluateScript("var outAckId = undefined;");
evaluateScript("cometd.registerExtension('test', {" +
"incoming: function(message)" +
"{" +
" if (message.channel == '/meta/connect')" +
" {" +
" inAckId = message.ext && message.ext.ack;" +
" }" +
" return message;" +
"}," +
"outgoing: function(message)" +
"{" +
" if (message.channel == '/meta/connect')" +
" {" +
" outAckId = message.ext && message.ext.ack;" +
" }" +
" return message;" +
"}" +
"});");
provideMessageAcknowledgeExtension();
evaluateScript("var readyLatch = new Latch(1);");
Latch readyLatch = get("readyLatch");
evaluateScript("cometd.addListener('/meta/connect', function(message) { readyLatch.countDown(); });");
evaluateScript("cometd.handshake();");
Assert.assertTrue(readyLatch.await(5000));
Number inAckId = get("inAckId");
// The server should have returned a non-negative value during the first connect call
Assert.assertTrue(inAckId.intValue() >= 0);
// Subscribe to receive server events
evaluateScript("var msgCount = 0;");
evaluateScript("var subscribeLatch = new Latch(1);");
Latch subscribeLatch = get("subscribeLatch");
evaluateScript("var publishLatch = new Latch(1);");
Latch publishLatch = get("publishLatch");
evaluateScript("cometd.addListener('/meta/subscribe', subscribeLatch, 'countDown');");
evaluateScript("cometd.subscribe('/echo', function(message) { ++msgCount; publishLatch.countDown(); });");
Assert.assertTrue(subscribeLatch.await(5000));
// The server receives an event and sends it to the client via the long poll
ackService.emit("test acknowledgement");
Assert.assertTrue(publishLatch.await(5000));
inAckId = get("inAckId");
Number outAckId = get("outAckId");
Assert.assertTrue(inAckId.intValue() >= outAckId.intValue());
Number msgCount = get("msgCount");
Assert.assertEquals(1, msgCount.intValue());
evaluateScript("cometd.unregisterExtension('test');");
evaluateScript("cometd.disconnect(true);");
}
public static class AckService extends AbstractService {
private AckService(BayeuxServerImpl bayeux) {
super(bayeux, "ack-test");
}
public void emit(String content) {
getBayeux().getChannel("/echo").publish(getServerSession(), content);
}
}
}