Skip to content

Commit d145cd4

Browse files
Catzillampociot
authored andcommitted
Fixed channels HTTP API endpoint (beyondcode#127)
* Fixed channels HTTP API endpoint
1 parent 03c9835 commit d145cd4

File tree

2 files changed

+84
-13
lines changed

2 files changed

+84
-13
lines changed

src/HttpApi/Controllers/FetchChannelsController.php

+19-8
Original file line numberDiff line numberDiff line change
@@ -5,15 +5,23 @@
55
use Illuminate\Support\Str;
66
use Illuminate\Http\Request;
77
use Illuminate\Support\Collection;
8-
use BeyondCode\LaravelWebSockets\WebSockets\Channels\PresenceChannel;
8+
use Symfony\Component\HttpKernel\Exception\HttpException;
99

1010
class FetchChannelsController extends Controller
1111
{
1212
public function __invoke(Request $request)
1313
{
14-
$channels = Collection::make($this->channelManager->getChannels($request->appId))->filter(function ($channel) {
15-
return $channel instanceof PresenceChannel;
16-
});
14+
$attributes = [];
15+
16+
if ($request->has('info')) {
17+
$attributes = explode(',', trim($request->info));
18+
19+
if (in_array('user_count', $attributes) && ! Str::startsWith($request->filter_by_prefix, 'presence-')) {
20+
throw new HttpException(400, 'Request must be limited to presence channels in order to fetch user_count');
21+
}
22+
}
23+
24+
$channels = Collection::make($this->channelManager->getChannels($request->appId));
1725

1826
if ($request->has('filter_by_prefix')) {
1927
$channels = $channels->filter(function ($channel, $channelName) use ($request) {
@@ -22,10 +30,13 @@ public function __invoke(Request $request)
2230
}
2331

2432
return [
25-
'channels' => $channels->map(function ($channel) {
26-
return [
27-
'user_count' => count($channel->getUsers()),
28-
];
33+
'channels' => $channels->map(function ($channel) use ($attributes) {
34+
$info = new \stdClass;
35+
if (in_array('user_count', $attributes)) {
36+
$info->user_count = count($channel->getUsers());
37+
}
38+
39+
return $info;
2940
})->toArray() ?: new \stdClass,
3041
];
3142
}

tests/HttpApi/FetchChannelsTest.php

+65-5
Original file line numberDiff line numberDiff line change
@@ -38,8 +38,6 @@ public function invalid_signatures_can_not_access_the_api()
3838
public function it_returns_the_channel_information()
3939
{
4040
$this->joinPresenceChannel('presence-channel');
41-
$this->joinPresenceChannel('presence-channel');
42-
$this->joinPresenceChannel('presence-channel');
4341

4442
$connection = new Connection();
4543

@@ -61,9 +59,7 @@ public function it_returns_the_channel_information()
6159

6260
$this->assertSame([
6361
'channels' => [
64-
'presence-channel' => [
65-
'user_count' => 3,
66-
],
62+
'presence-channel' => [],
6763
],
6864
], json_decode($response->getContent(), true));
6965
}
@@ -96,6 +92,43 @@ public function it_returns_the_channel_information_for_prefix()
9692
/** @var JsonResponse $response */
9793
$response = array_pop($connection->sentRawData);
9894

95+
$this->assertSame([
96+
'channels' => [
97+
'presence-global.1' => [],
98+
'presence-global.2' => [],
99+
],
100+
], json_decode($response->getContent(), true));
101+
}
102+
103+
/** @test */
104+
public function it_returns_the_channel_information_for_prefix_with_user_count()
105+
{
106+
$this->joinPresenceChannel('presence-global.1');
107+
$this->joinPresenceChannel('presence-global.1');
108+
$this->joinPresenceChannel('presence-global.2');
109+
$this->joinPresenceChannel('presence-notglobal.2');
110+
111+
$connection = new Connection();
112+
113+
$requestPath = '/apps/1234/channels';
114+
$routeParams = [
115+
'appId' => '1234',
116+
];
117+
118+
$queryString = Pusher::build_auth_query_string('TestKey', 'TestSecret', 'GET', $requestPath, [
119+
'filter_by_prefix' => 'presence-global',
120+
'info' => 'user_count',
121+
]);
122+
123+
$request = new Request('GET', "{$requestPath}?{$queryString}&".http_build_query($routeParams));
124+
125+
$controller = app(FetchChannelsController::class);
126+
127+
$controller->onOpen($connection, $request);
128+
129+
/** @var JsonResponse $response */
130+
$response = array_pop($connection->sentRawData);
131+
99132
$this->assertSame([
100133
'channels' => [
101134
'presence-global.1' => [
@@ -108,6 +141,33 @@ public function it_returns_the_channel_information_for_prefix()
108141
], json_decode($response->getContent(), true));
109142
}
110143

144+
/** @test */
145+
public function can_not_get_non_presence_channel_user_count()
146+
{
147+
$this->expectException(HttpException::class);
148+
$this->expectExceptionMessage('Request must be limited to presence channels in order to fetch user_count');
149+
150+
$connection = new Connection();
151+
152+
$requestPath = '/apps/1234/channels';
153+
$routeParams = [
154+
'appId' => '1234',
155+
];
156+
157+
$queryString = Pusher::build_auth_query_string('TestKey', 'TestSecret', 'GET', $requestPath, [
158+
'info' => 'user_count',
159+
]);
160+
161+
$request = new Request('GET', "{$requestPath}?{$queryString}&".http_build_query($routeParams));
162+
163+
$controller = app(FetchChannelsController::class);
164+
165+
$controller->onOpen($connection, $request);
166+
167+
/** @var JsonResponse $response */
168+
$response = array_pop($connection->sentRawData);
169+
}
170+
111171
/** @test */
112172
public function it_returns_empty_object_for_no_channels_found()
113173
{

0 commit comments

Comments
 (0)