2020-09-10 19:59:26 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
namespace BeyondCode\LaravelWebSockets\Test;
|
|
|
|
|
|
|
|
|
|
use BeyondCode\LaravelWebSockets\Server\Exceptions\InvalidSignature;
|
2020-09-12 14:45:07 +00:00
|
|
|
use Ratchet\ConnectionInterface;
|
2020-09-10 19:59:26 +00:00
|
|
|
|
|
|
|
|
class PresenceChannelTest extends TestCase
|
|
|
|
|
{
|
|
|
|
|
public function test_connect_to_presence_channel_with_invalid_signature()
|
|
|
|
|
{
|
|
|
|
|
$this->expectException(InvalidSignature::class);
|
|
|
|
|
|
|
|
|
|
$connection = $this->newConnection();
|
|
|
|
|
|
|
|
|
|
$message = new Mocks\Message([
|
|
|
|
|
'event' => 'pusher:subscribe',
|
|
|
|
|
'data' => [
|
|
|
|
|
'auth' => 'invalid',
|
|
|
|
|
'channel' => 'presence-channel',
|
|
|
|
|
],
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
$this->pusherServer->onOpen($connection);
|
|
|
|
|
$this->pusherServer->onMessage($connection, $message);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function test_connect_to_presence_channel_with_valid_signature()
|
|
|
|
|
{
|
|
|
|
|
$connection = $this->newConnection();
|
|
|
|
|
|
|
|
|
|
$this->pusherServer->onOpen($connection);
|
|
|
|
|
|
|
|
|
|
$user = [
|
|
|
|
|
'user_id' => 1,
|
|
|
|
|
'user_info' => [
|
|
|
|
|
'name' => 'Rick',
|
|
|
|
|
],
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
$encodedUser = json_encode($user);
|
|
|
|
|
|
2020-09-15 17:46:19 +00:00
|
|
|
$message = new Mocks\SignedMessage([
|
2020-09-10 19:59:26 +00:00
|
|
|
'event' => 'pusher:subscribe',
|
|
|
|
|
'data' => [
|
|
|
|
|
'channel' => 'presence-channel',
|
2020-09-15 17:46:19 +00:00
|
|
|
'channel_data' => $encodedUser,
|
2020-09-10 19:59:26 +00:00
|
|
|
],
|
2020-09-15 17:46:19 +00:00
|
|
|
], $connection, 'presence-channel', $encodedUser);
|
2020-09-10 19:59:26 +00:00
|
|
|
|
|
|
|
|
$this->pusherServer->onMessage($connection, $message);
|
|
|
|
|
|
|
|
|
|
$connection->assertSentEvent('pusher_internal:subscription_succeeded', [
|
|
|
|
|
'channel' => 'presence-channel',
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
$this->channelManager
|
|
|
|
|
->getGlobalConnectionsCount('1234', 'presence-channel')
|
|
|
|
|
->then(function ($total) {
|
|
|
|
|
$this->assertEquals(1, $total);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-16 08:02:58 +00:00
|
|
|
public function test_connect_to_presence_channel_when_user_with_same_ids_is_already_joined()
|
|
|
|
|
{
|
|
|
|
|
$rick = $this->newPresenceConnection('presence-channel', ['user_id' => 1]);
|
|
|
|
|
$morty = $this->newPresenceConnection('presence-channel', ['user_id' => 2]);
|
|
|
|
|
$pickleRick = $this->newPresenceConnection('presence-channel', ['user_id' => 1]);
|
|
|
|
|
|
|
|
|
|
foreach ([$rick, $morty, $pickleRick] as $connection) {
|
|
|
|
|
$connection->assertSentEvent('pusher_internal:subscription_succeeded', [
|
|
|
|
|
'channel' => 'presence-channel',
|
|
|
|
|
]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$this->channelManager
|
|
|
|
|
->getGlobalConnectionsCount('1234', 'presence-channel')
|
|
|
|
|
->then(function ($total) {
|
|
|
|
|
$this->assertEquals(3, $total);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
$this->channelManager
|
|
|
|
|
->getChannelMembers('1234', 'presence-channel')
|
|
|
|
|
->then(function ($members) {
|
|
|
|
|
$this->assertCount(2, $members);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-10 19:59:26 +00:00
|
|
|
public function test_presence_channel_broadcast_member_events()
|
|
|
|
|
{
|
|
|
|
|
$rick = $this->newPresenceConnection('presence-channel', ['user_id' => 1]);
|
|
|
|
|
$morty = $this->newPresenceConnection('presence-channel', ['user_id' => 2]);
|
|
|
|
|
|
|
|
|
|
$rick->assertSentEvent('pusher_internal:member_added', [
|
|
|
|
|
'channel' => 'presence-channel',
|
|
|
|
|
'data' => json_encode(['user_id' => 2]),
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
$this->channelManager
|
|
|
|
|
->getChannelMembers('1234', 'presence-channel')
|
|
|
|
|
->then(function ($members) {
|
|
|
|
|
$this->assertCount(2, $members);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
$this->pusherServer->onClose($morty);
|
|
|
|
|
|
|
|
|
|
$rick->assertSentEvent('pusher_internal:member_removed', [
|
|
|
|
|
'channel' => 'presence-channel',
|
|
|
|
|
'data' => json_encode(['user_id' => 2]),
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
$this->channelManager
|
|
|
|
|
->getGlobalConnectionsCount('1234', 'presence-channel')
|
|
|
|
|
->then(function ($total) {
|
|
|
|
|
$this->assertEquals(1, $total);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
$this->channelManager
|
|
|
|
|
->getChannelMembers('1234', 'presence-channel')
|
|
|
|
|
->then(function ($members) {
|
|
|
|
|
$this->assertCount(1, $members);
|
|
|
|
|
$this->assertEquals(1, $members[0]->user_id);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function test_unsubscribe_from_presence_channel()
|
|
|
|
|
{
|
|
|
|
|
$connection = $this->newPresenceConnection('presence-channel', ['user_id' => 1]);
|
|
|
|
|
|
|
|
|
|
$this->channelManager
|
|
|
|
|
->getGlobalConnectionsCount('1234', 'presence-channel')
|
|
|
|
|
->then(function ($total) {
|
|
|
|
|
$this->assertEquals(1, $total);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
$message = new Mocks\Message([
|
|
|
|
|
'event' => 'pusher:unsubscribe',
|
|
|
|
|
'data' => [
|
|
|
|
|
'channel' => 'presence-channel',
|
|
|
|
|
],
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
$this->pusherServer->onMessage($connection, $message);
|
|
|
|
|
|
|
|
|
|
$this->channelManager
|
|
|
|
|
->getGlobalConnectionsCount('1234', 'presence-channel')
|
|
|
|
|
->then(function ($total) {
|
|
|
|
|
$this->assertEquals(0, $total);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function test_can_whisper_to_private_channel()
|
|
|
|
|
{
|
|
|
|
|
$this->app['config']->set('websockets.apps.0.enable_client_messages', true);
|
|
|
|
|
|
|
|
|
|
$rick = $this->newPresenceConnection('presence-channel', ['user_id' => 1]);
|
|
|
|
|
$morty = $this->newPresenceConnection('presence-channel', ['user_id' => 2]);
|
|
|
|
|
|
|
|
|
|
$message = new Mocks\Message([
|
|
|
|
|
'event' => 'client-test-whisper',
|
|
|
|
|
'data' => [],
|
|
|
|
|
'channel' => 'presence-channel',
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
$this->pusherServer->onMessage($rick, $message);
|
|
|
|
|
|
|
|
|
|
$rick->assertNotSentEvent('client-test-whisper');
|
|
|
|
|
$morty->assertSentEvent('client-test-whisper', ['data' => [], 'channel' => 'presence-channel']);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function test_cannot_whisper_to_public_channel_if_having_whispering_disabled()
|
|
|
|
|
{
|
|
|
|
|
$rick = $this->newPresenceConnection('presence-channel', ['user_id' => 1]);
|
|
|
|
|
$morty = $this->newPresenceConnection('presence-channel', ['user_id' => 2]);
|
|
|
|
|
|
|
|
|
|
$message = new Mocks\Message([
|
|
|
|
|
'event' => 'client-test-whisper',
|
|
|
|
|
'data' => [],
|
|
|
|
|
'channel' => 'presence-channel',
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
$this->pusherServer->onMessage($rick, $message);
|
|
|
|
|
|
|
|
|
|
$rick->assertNotSentEvent('client-test-whisper');
|
|
|
|
|
$morty->assertNotSentEvent('client-test-whisper');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function test_statistics_get_collected_for_presenece_channels()
|
|
|
|
|
{
|
|
|
|
|
$rick = $this->newPresenceConnection('presence-channel', ['user_id' => 1]);
|
|
|
|
|
$morty = $this->newPresenceConnection('presence-channel', ['user_id' => 2]);
|
|
|
|
|
|
|
|
|
|
$this->statisticsCollector
|
|
|
|
|
->getStatistics()
|
|
|
|
|
->then(function ($statistics) {
|
|
|
|
|
$this->assertCount(1, $statistics);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
$this->statisticsCollector
|
|
|
|
|
->getAppStatistics('1234')
|
|
|
|
|
->then(function ($statistic) {
|
|
|
|
|
$this->assertEquals([
|
|
|
|
|
'peak_connections_count' => 2,
|
|
|
|
|
'websocket_messages_count' => 2,
|
|
|
|
|
'api_messages_count' => 0,
|
|
|
|
|
'app_id' => '1234',
|
|
|
|
|
], $statistic->toArray());
|
|
|
|
|
});
|
|
|
|
|
}
|
2020-09-12 14:45:07 +00:00
|
|
|
|
2020-09-15 17:46:19 +00:00
|
|
|
public function test_local_connections_for_presence_channels()
|
2020-09-12 14:45:07 +00:00
|
|
|
{
|
|
|
|
|
$this->newPresenceConnection('presence-channel', ['user_id' => 1]);
|
|
|
|
|
$this->newPresenceConnection('presence-channel-2', ['user_id' => 2]);
|
|
|
|
|
|
|
|
|
|
$this->channelManager
|
|
|
|
|
->getLocalConnections()
|
|
|
|
|
->then(function ($connections) {
|
|
|
|
|
$this->assertCount(2, $connections);
|
|
|
|
|
|
|
|
|
|
foreach ($connections as $connection) {
|
|
|
|
|
$this->assertInstanceOf(
|
|
|
|
|
ConnectionInterface::class, $connection
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
2020-09-16 08:02:58 +00:00
|
|
|
|
|
|
|
|
public function test_multiple_clients_with_same_user_id_trigger_member_added_and_removed_event_only_on_first_and_last_socket_connection()
|
|
|
|
|
{
|
|
|
|
|
// Connect the `observer` user to the server
|
|
|
|
|
$observerConnection = $this->newPresenceConnection('presence-channel', ['user_id' => 'observer']);
|
|
|
|
|
|
|
|
|
|
// Connect the first socket for user `1` to the server
|
|
|
|
|
$firstConnection = $this->newPresenceConnection('presence-channel', ['user_id' => '1']);
|
|
|
|
|
|
|
|
|
|
// Make sure the observer sees a `member_added` event for `user:1`
|
|
|
|
|
$observerConnection->assertSentEvent('pusher_internal:member_added', [
|
|
|
|
|
'event' => 'pusher_internal:member_added',
|
|
|
|
|
'channel' => 'presence-channel',
|
|
|
|
|
'data' => json_encode(['user_id' => '1']),
|
|
|
|
|
])->resetEvents();
|
|
|
|
|
|
|
|
|
|
// Connect the second socket for user `1` to the server
|
|
|
|
|
$secondConnection = $this->newPresenceConnection('presence-channel', ['user_id' => '1']);
|
|
|
|
|
|
|
|
|
|
// Make sure the observer was not notified of a `member_added` event (user was already connected)
|
|
|
|
|
$observerConnection->assertNotSentEvent('pusher_internal:member_added');
|
|
|
|
|
|
|
|
|
|
// Disconnect the first socket for user `1` on the server
|
|
|
|
|
$this->pusherServer->onClose($firstConnection);
|
|
|
|
|
|
|
|
|
|
// Make sure the observer was not notified of a `member_removed` event (user still connected on another socket)
|
|
|
|
|
$observerConnection->assertNotSentEvent('pusher_internal:member_removed');
|
|
|
|
|
|
|
|
|
|
// Disconnect the second (and last) socket for user `1` on the server
|
|
|
|
|
$this->pusherServer->onClose($secondConnection);
|
|
|
|
|
|
|
|
|
|
// Make sure the observer was notified of a `member_removed` event (last socket for user was disconnected)
|
|
|
|
|
$observerConnection->assertSentEvent('pusher_internal:member_removed');
|
|
|
|
|
|
|
|
|
|
$this->channelManager
|
|
|
|
|
->getMemberSockets('1', '1234', 'presence-channel')
|
|
|
|
|
->then(function ($sockets) {
|
|
|
|
|
$this->assertCount(0, $sockets);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
$this->channelManager
|
|
|
|
|
->getMemberSockets('2', '1234', 'presence-channel')
|
|
|
|
|
->then(function ($sockets) {
|
|
|
|
|
$this->assertCount(0, $sockets);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
$this->channelManager
|
|
|
|
|
->getMemberSockets('observer', '1234', 'presence-channel')
|
|
|
|
|
->then(function ($sockets) {
|
|
|
|
|
$this->assertCount(1, $sockets);
|
|
|
|
|
});
|
|
|
|
|
}
|
2020-09-10 19:59:26 +00:00
|
|
|
}
|