Skip to content

Commit

Permalink
prevent duplicate events
Browse files Browse the repository at this point in the history
  • Loading branch information
joedixon committed Jun 26, 2024
1 parent 307ebfa commit d7acda0
Show file tree
Hide file tree
Showing 2 changed files with 69 additions and 15 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -15,12 +15,20 @@ public function subscribe(Connection $connection, ?string $auth = null, ?string
{
$this->verify($connection, $auth, $data);

$userData = $data ? json_decode($data, associative: true, flags: JSON_THROW_ON_ERROR) : [];

if ($this->userIsSubscribed($userData['user_id'] ?? null)) {
parent::subscribe($connection, $auth, $data);

return;
}

parent::subscribe($connection, $auth, $data);

parent::broadcastInternally(
[
'event' => 'pusher_internal:member_added',
'data' => $data ? json_decode($data, associative: true, flags: JSON_THROW_ON_ERROR) : [],
'data' => $userData,
'channel' => $this->name(),
],
$connection
Expand All @@ -32,24 +40,26 @@ public function subscribe(Connection $connection, ?string $auth = null, ?string
*/
public function unsubscribe(Connection $connection): void
{
if (! $subscription = $this->connections->find($connection)) {
parent::unsubscribe($connection);
$subscription = $this->connections->find($connection);

return;
}
parent::unsubscribe($connection);

if ($userId = $subscription->data('user_id')) {
parent::broadcast(
[
'event' => 'pusher_internal:member_removed',
'data' => ['user_id' => $userId],
'channel' => $this->name(),
],
$connection
);
if (
! $subscription ||
! $subscription->data('user_id') ||
$this->userIsSubscribed($subscription->data('user_id'))
) {
return;
}

parent::unsubscribe($connection);
parent::broadcast(
[
'event' => 'pusher_internal:member_removed',
'data' => ['user_id' => $subscription->data('user_id')],
'channel' => $this->name(),
],
$connection
);
}

/**
Expand Down Expand Up @@ -79,4 +89,16 @@ public function data(): array
],
];
}

/**
* Determine if the given user is subscribed to the channel.
*/
protected function userIsSubscribed(?string $userId): bool
{
if (! $userId) {
return false;
}

return collect($this->connections->all())->map(fn ($connection) => (string) $connection->data('user_id'))->contains($userId);
}
}
32 changes: 32 additions & 0 deletions tests/Unit/Protocols/Pusher/Channels/PresenceChannelTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -161,3 +161,35 @@
'channel' => 'presence-test-channel',
]));
});

it('ensures the "member_added" event is only fired once', function () {
$channel = new PresenceChannel('presence-test-channel');

$connectionOne = collect(factory(data: ['user_info' => ['name' => 'Joe'], 'user_id' => 1]))->first();
$connectionTwo = collect(factory(data: ['user_info' => ['name' => 'Joe'], 'user_id' => 1]))->first();

$this->channelConnectionManager->shouldReceive('all')
->andReturn([$connectionOne, $connectionTwo]);

$channel->subscribe($connectionOne->connection(), validAuth($connectionOne->id(), 'presence-test-channel', $data = json_encode($connectionOne->data())), $data);
$channel->subscribe($connectionTwo->connection(), validAuth($connectionTwo->id(), 'presence-test-channel', $data = json_encode($connectionTwo->data())), $data);

$connectionOne->connection()->assertNothingReceived();
});

it('ensures the "member_removed" event is only fired once', function () {
$channel = new PresenceChannel('presence-test-channel');

$connectionOne = collect(factory(data: ['user_info' => ['name' => 'Joe'], 'user_id' => 1]))->first();
$connectionTwo = collect(factory(data: ['user_info' => ['name' => 'Joe'], 'user_id' => 1]))->first();

$this->channelConnectionManager->shouldReceive('find')
->andReturn($connectionOne);

$this->channelConnectionManager->shouldReceive('all')
->andReturn([$connectionOne, $connectionTwo]);

$channel->unsubscribe($connectionTwo->connection(), validAuth($connectionTwo->id(), 'presence-test-channel', $data = json_encode($connectionTwo->data())), $data);

$connectionOne->connection()->assertNothingReceived();
});

0 comments on commit d7acda0

Please sign in to comment.