From 4ff179ad4d391c72f5fd2d3767a3c7e24e9cb441 Mon Sep 17 00:00:00 2001 From: Daniel Supernault Date: Mon, 13 Jun 2022 03:08:43 -0600 Subject: [PATCH 1/6] Update ApiV1Controller, improve local/remote logic in public timeline endpoint --- app/Http/Controllers/Api/ApiV1Controller.php | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/app/Http/Controllers/Api/ApiV1Controller.php b/app/Http/Controllers/Api/ApiV1Controller.php index 18890483b..3b1edace0 100644 --- a/app/Http/Controllers/Api/ApiV1Controller.php +++ b/app/Http/Controllers/Api/ApiV1Controller.php @@ -1973,14 +1973,15 @@ class ApiV1Controller extends Controller 'min_id' => 'nullable|integer|min:0|max:' . PHP_INT_MAX, 'max_id' => 'nullable|integer|min:0|max:' . PHP_INT_MAX, 'limit' => 'nullable|integer|max:100', - 'remote' => 'sometimes' + 'remote' => 'sometimes', + 'local' => 'sometimes' ]); $min = $request->input('min_id'); $max = $request->input('max_id'); $limit = $request->input('limit') ?? 20; $user = $request->user(); - $remote = $request->has('remote'); + $remote = ($request->has('remote') && $request->input('remote') == true) || ($request->filled('local') && $request->input('local') != true); $filtered = $user ? UserFilterService::filters($user->profile_id) : []; if($remote && config('instance.timeline.network.cached')) { From 494f11202f58066699762c88b56db776d18d95f9 Mon Sep 17 00:00:00 2001 From: Daniel Supernault Date: Wed, 15 Jun 2022 00:46:20 -0600 Subject: [PATCH 2/6] Update LiveStream model --- app/Http/Controllers/LiveStreamController.php | 429 +++++++++++------- app/Models/LiveStream.php | 58 ++- 2 files changed, 280 insertions(+), 207 deletions(-) diff --git a/app/Http/Controllers/LiveStreamController.php b/app/Http/Controllers/LiveStreamController.php index 51cc160f9..ecc999770 100644 --- a/app/Http/Controllers/LiveStreamController.php +++ b/app/Http/Controllers/LiveStreamController.php @@ -12,225 +12,304 @@ use App\Services\LiveStreamService; class LiveStreamController extends Controller { - public function createStream(Request $request) - { - abort_if(!config('livestreaming.enabled'), 400); - abort_if(!$request->user(), 403); + public function createStream(Request $request) + { + abort_if(!config('livestreaming.enabled'), 400); + abort_if(!$request->user(), 403); - if(config('livestreaming.broadcast.limits.enabled')) { - if($request->user()->is_admin) { + if(config('livestreaming.broadcast.limits.enabled')) { + if($request->user()->is_admin) { - } else { - $limits = config('livestreaming.broadcast.limits'); - $user = $request->user(); - abort_if($limits['admins_only'] && $user->is_admin == false, 401, 'LSE:003'); - if($limits['min_account_age']) { - abort_if($user->created_at->gt(now()->subDays($limits['min_account_age'])), 403, 'LSE:005'); - } + } else { + $limits = config('livestreaming.broadcast.limits'); + $user = $request->user(); + abort_if($limits['admins_only'] && $user->is_admin == false, 401, 'LSE:003'); + if($limits['min_account_age']) { + abort_if($user->created_at->gt(now()->subDays($limits['min_account_age'])), 403, 'LSE:005'); + } - if($limits['min_follower_count']) { - $account = AccountService::get($user->profile_id); - abort_if($account['followers_count'] < $limits['min_follower_count'], 403, 'LSE:008'); - } - } - } + if($limits['min_follower_count']) { + $account = AccountService::get($user->profile_id); + abort_if($account['followers_count'] < $limits['min_follower_count'], 403, 'LSE:008'); + } + } + } - $this->validate($request, [ - 'name' => 'nullable|string|max:80', - 'description' => 'nullable|string|max:240', - 'visibility' => 'required|in:public,private' - ]); + $this->validate($request, [ + 'name' => 'nullable|string|max:80', + 'description' => 'nullable|string|max:240', + 'visibility' => 'required|in:public,private' + ]); - $stream = new LiveStream; - $stream->name = $request->input('name'); - $stream->description = $request->input('description'); - $stream->visibility = $request->input('visibility'); - $stream->profile_id = $request->user()->profile_id; - $stream->stream_id = Str::random(40); - $stream->stream_key = Str::random(64); - $stream->save(); + $stream = new LiveStream; + $stream->name = $request->input('name'); + $stream->description = $request->input('description'); + $stream->visibility = $request->input('visibility'); + $stream->profile_id = $request->user()->profile_id; + $stream->stream_id = Str::random(40) . '_' . $stream->profile_id; + $stream->stream_key = 'streamkey-' . Str::random(64); + $stream->save(); - return [ - 'url' => $stream->getStreamKeyUrl(), - 'id' => $stream->stream_id - ]; - } + return [ + 'host' => $stream->getStreamServer(), + 'key' => $stream->stream_key, + 'url' => $stream->getStreamKeyUrl(), + 'id' => $stream->stream_id + ]; + } - public function getUserStream(Request $request) - { - abort_if(!config('livestreaming.enabled'), 400); - abort_if(!$request->user(), 403); + public function getUserStream(Request $request) + { + abort_if(!config('livestreaming.enabled'), 400); + abort_if(!$request->user(), 403); - $stream = LiveStream::whereProfileId($request->input('profile_id'))->first(); + $stream = LiveStream::whereProfileId($request->input('profile_id'))->first(); - if(!$stream) { - return []; - } + if(!$stream) { + return []; + } - $res = []; - $owner = $stream->profile_id == $request->user()->profile_id; + $res = []; + $owner = $stream->profile_id == $request->user()->profile_id; - if($stream->visibility === 'private') { - abort_if(!$owner && !FollowerService::follows($request->user()->profile_id, $stream->profile_id), 403, 'LSE:011'); - } + if($stream->visibility === 'private') { + abort_if(!$owner && !FollowerService::follows($request->user()->profile_id, $stream->profile_id), 403, 'LSE:011'); + } - if($owner) { - $res['stream_key'] = $stream->stream_key; - $res['stream_id'] = $stream->stream_id; - $res['stream_url'] = $stream->getStreamKeyUrl(); - } + if($owner) { + $res['stream_key'] = $stream->stream_key; + $res['stream_id'] = $stream->stream_id; + $res['stream_url'] = $stream->getStreamKeyUrl(); + } - if($stream->live_at == null) { - $res['hls_url'] = null; - $res['name'] = $stream->name; - $res['description'] = $stream->description; - return $res; - } + if($stream->live_at == null) { + $res['hls_url'] = null; + $res['name'] = $stream->name; + $res['description'] = $stream->description; + return $res; + } - $res = [ - 'hls_url' => $stream->getHlsUrl(), - 'name' => $stream->name, - 'description' => $stream->description - ]; + $res = [ + 'hls_url' => $stream->getHlsUrl(), + 'name' => $stream->name, + 'description' => $stream->description + ]; - return response()->json($res, 200, [], JSON_UNESCAPED_SLASHES); - } + return response()->json($res, 200, [], JSON_UNESCAPED_SLASHES); + } - public function deleteStream(Request $request) - { - abort_if(!config('livestreaming.enabled'), 400); - abort_if(!$request->user(), 403); + public function deleteStream(Request $request) + { + abort_if(!config('livestreaming.enabled'), 400); + abort_if(!$request->user(), 403); - LiveStream::whereProfileId($request->user()->profile_id) - ->get() - ->each(function($stream) { - Storage::deleteDirectory("public/live-hls/{$stream->stream_id}"); - $stream->delete(); - }); + LiveStream::whereProfileId($request->user()->profile_id) + ->get() + ->each(function($stream) { + Storage::deleteDirectory("public/live-hls/{$stream->stream_id}"); + $stream->delete(); + }); - return [200]; - } + return [200]; + } - public function getActiveStreams(Request $request) - { - abort_if(!config('livestreaming.enabled'), 400); - abort_if(!$request->user(), 403); + public function getActiveStreams(Request $request) + { + abort_if(!config('livestreaming.enabled'), 400); + abort_if(!$request->user(), 403); - return LiveStream::whereVisibility('local')->whereNotNull('live_at')->get()->map(function($stream) { - return [ - 'account' => AccountService::get($stream->profile_id), - 'stream_id' => $stream->stream_id - ]; - }); - } + return LiveStream::whereVisibility('local')->whereNotNull('live_at')->get()->map(function($stream) { + return [ + 'account' => AccountService::get($stream->profile_id), + 'stream_id' => $stream->stream_id + ]; + }); + } - public function getLatestChat(Request $request) - { - abort_if(!config('livestreaming.enabled'), 400); - abort_if(!$request->user(), 403); + public function getLatestChat(Request $request) + { + abort_if(!config('livestreaming.enabled'), 400); + abort_if(!$request->user(), 403); - $stream = LiveStream::whereProfileId($request->input('profile_id')) - ->whereNotNull('live_at') - ->first(); + $stream = LiveStream::whereProfileId($request->input('profile_id')) + ->whereNotNull('live_at') + ->first(); - if(!$stream) { - return []; - } + if(!$stream) { + return []; + } - $owner = $stream->profile_id == $request->user()->profile_id; - if($stream->visibility === 'private') { - abort_if(!$owner && !FollowerService::follows($request->user()->profile_id, $stream->profile_id), 403, 'LSE:021'); - } + $owner = $stream->profile_id == $request->user()->profile_id; + if($stream->visibility === 'private') { + abort_if(!$owner && !FollowerService::follows($request->user()->profile_id, $stream->profile_id), 403, 'LSE:021'); + } - $res = collect(LiveStreamService::getComments($stream->profile_id)) - ->map(function($r) { - return json_decode($r); - }); + $res = collect(LiveStreamService::getComments($stream->profile_id)) + ->map(function($res) { + return json_decode($res); + }); - return $res; - } + return $res; + } - public function addChatComment(Request $request) - { - abort_if(!config('livestreaming.enabled'), 400); - abort_if(!$request->user(), 403); + public function addChatComment(Request $request) + { + abort_if(!config('livestreaming.enabled'), 400); + abort_if(!$request->user(), 403); - $this->validate($request, [ - 'profile_id' => 'required|exists:profiles,id', - 'message' => 'required|max:140' - ]); + $this->validate($request, [ + 'profile_id' => 'required|exists:profiles,id', + 'message' => 'required|max:140' + ]); - $stream = LiveStream::whereProfileId($request->input('profile_id'))->firstOrFail(); + $stream = LiveStream::whereProfileId($request->input('profile_id'))->firstOrFail(); - $owner = $stream->profile_id == $request->user()->profile_id; - if($stream->visibility === 'private') { - abort_if(!$owner && !FollowerService::follows($request->user()->profile_id, $stream->profile_id), 403, 'LSE:022'); - } + $owner = $stream->profile_id == $request->user()->profile_id; + if($stream->visibility === 'private') { + abort_if(!$owner && !FollowerService::follows($request->user()->profile_id, $stream->profile_id), 403, 'LSE:022'); + } - $res = [ - 'pid' => (string) $request->user()->profile_id, - 'username' => $request->user()->username, - 'text' => $request->input('message'), - 'ts' => now()->timestamp - ]; + $res = [ + 'pid' => (string) $request->user()->profile_id, + 'username' => $request->user()->username, + 'text' => $request->input('message'), + 'ts' => now()->timestamp + ]; - LiveStreamService::addComment($stream->profile_id, json_encode($res, JSON_UNESCAPED_SLASHES)); + LiveStreamService::addComment($stream->profile_id, json_encode($res, JSON_UNESCAPED_SLASHES)); - return $res; - } + return $res; + } - public function editStream(Request $request) - { - abort_if(!config('livestreaming.enabled'), 400); - abort_if(!$request->user(), 403); + public function editStream(Request $request) + { + abort_if(!config('livestreaming.enabled'), 400); + abort_if(!$request->user(), 403); - $this->validate($request, [ - 'name' => 'nullable|string|max:80', - 'description' => 'nullable|string|max:240' - ]); + $this->validate($request, [ + 'name' => 'nullable|string|max:80', + 'description' => 'nullable|string|max:240' + ]); - $stream = LiveStream::whereProfileId($request->user()->profile_id)->firstOrFail(); - $stream->name = $request->input('name'); - $stream->description = $request->input('description'); - $stream->save(); + $stream = LiveStream::whereProfileId($request->user()->profile_id)->firstOrFail(); + $stream->name = $request->input('name'); + $stream->description = $request->input('description'); + $stream->save(); - return; - } + return; + } - public function deleteChatComment(Request $request) - { - abort_if(!config('livestreaming.enabled'), 400); - abort_if(!$request->user(), 403); + public function deleteChatComment(Request $request) + { + abort_if(!config('livestreaming.enabled'), 400); + abort_if(!$request->user(), 403); - $this->validate($request, [ - 'profile_id' => 'required|exists:profiles,id', - 'message' => 'required' - ]); + $this->validate($request, [ + 'profile_id' => 'required|exists:profiles,id', + 'message' => 'required' + ]); - abort_if($request->user()->profile_id != $request->input('profile_id'), 403); + abort_if($request->user()->profile_id != $request->input('profile_id'), 403); - $stream = LiveStream::whereProfileId($request->user()->profile_id)->firstOrFail(); + $stream = LiveStream::whereProfileId($request->user()->profile_id)->firstOrFail(); - $payload = $request->input('message'); - $payload = json_encode($payload, JSON_UNESCAPED_SLASHES); - LiveStreamService::deleteComment($stream->profile_id, $payload); + $payload = $request->input('message'); + $payload = json_encode($payload, JSON_UNESCAPED_SLASHES); + LiveStreamService::deleteComment($stream->profile_id, $payload); - return; - } + return; + } - public function getConfig(Request $request) - { - abort_if(!config('livestreaming.enabled'), 400); - abort_if(!$request->user(), 403); + public function getConfig(Request $request) + { + abort_if(!config('livestreaming.enabled'), 400); + abort_if(!$request->user(), 403); - $res = [ - 'enabled' => config('livestreaming.enabled'), - 'broadcast' => [ - 'sources' => config('livestreaming.broadcast.sources'), - 'limits' => config('livestreaming.broadcast.limits') - ], - ]; + $res = [ + 'enabled' => config('livestreaming.enabled'), + 'broadcast' => [ + 'sources' => config('livestreaming.broadcast.sources'), + 'limits' => config('livestreaming.broadcast.limits') + ], + ]; - return response()->json($res, 200, [], JSON_UNESCAPED_SLASHES); - } + return response()->json($res, 200, [], JSON_UNESCAPED_SLASHES); + } + + public function clientBroadcastPublish(Request $request) + { + abort_if(!config('livestreaming.enabled'), 400); + $key = $request->input('name'); + $name = $request->input('name'); + + abort_if(!$name, 400); + + if(empty($key)) { + abort_if(!$request->filled('tcurl'), 400); + $url = $this->parseStreamUrl($request->input('tcurl')); + $key = $request->filled('name') ? $request->input('name') : $url['name']; + } + + $token = substr($name, 0, 10) === 'streamkey-'; + + if($token) { + $stream = LiveStream::whereStreamKey($key)->firstOrFail(); + return redirect($stream->getStreamRtmpUrl(), 301); + } else { + $stream = LiveStream::whereStreamId($key)->firstOrFail(); + } + + if($request->filled('name') && $token == false) { + $stream->live_at = now(); + $stream->save(); + return []; + } else { + abort(400); + } + + abort(400); + } + + public function clientBroadcastFinish(Request $request) + { + abort_if(!config('livestreaming.enabled'), 400); + abort_if(!$request->filled('tcurl'), 400); + $url = $this->parseStreamUrl($request->input('tcurl')); + $name = $url['name'] ?? $request->input('name'); + + $stream = LiveStream::whereStreamId($name)->whereStreamKey($url['key'])->firstOrFail(); + + if(config('livestreaming.broadcast.delete_token_after_finished')) { + $stream->delete(); + } else { + $stream->live_at = null; + $stream->save(); + } + + return []; + } + + protected function parseStreamUrl($url) + { + $name = null; + $key = null; + $query = parse_url($url, PHP_URL_QUERY); + $parts = explode('&', $query); + foreach($parts as $part) { + if (!strlen(trim($part))) { + continue; + } + $s = explode('=', $part); + if(in_array($s[0], ['name', 'key'])) { + if($s[0] === 'name') { + $name = $s[1]; + } + if($s[0] === 'key') { + $key = $s[1]; + } + } + } + + return ['name' => $name, 'key' => $key]; + } } diff --git a/app/Models/LiveStream.php b/app/Models/LiveStream.php index 197bd70ad..98b8d8211 100644 --- a/app/Models/LiveStream.php +++ b/app/Models/LiveStream.php @@ -8,40 +8,34 @@ use Storage; class LiveStream extends Model { - use HasFactory; + use HasFactory; - public function getHlsUrl() - { - $path = Storage::url("live-hls/{$this->stream_id}/index.m3u8"); - return url($path); - } + public function getHlsUrl() + { + $path = Storage::url("live-hls/{$this->stream_id}/index.m3u8"); + return url($path); + } - public function getStreamKeyUrl() - { - $proto = 'rtmp://'; - $host = config('livestreaming.server.host'); - $port = ':' . config('livestreaming.server.port'); - $path = '/' . config('livestreaming.server.path') . '?'; - $query = http_build_query([ - 'name' => $this->stream_id, - 'key' => $this->stream_key, - 'ts' => time() - ]); + public function getStreamServer() + { + $proto = 'rtmp://'; + $host = config('livestreaming.server.host'); + $port = ':' . config('livestreaming.server.port'); + $path = '/' . config('livestreaming.server.path'); + return $proto . $host . $port . $path; + } - return $proto . $host . $port . $path . $query; - } + public function getStreamKeyUrl() + { + $path = $this->getStreamServer() . '?'; + $query = http_build_query([ + 'name' => $this->stream_key, + ]); + return $path . $query; + } - public function getStreamRtmpUrl() - { - $proto = 'rtmp://'; - $host = config('livestreaming.server.host'); - $port = ':' . config('livestreaming.server.port'); - $path = '/' . config('livestreaming.server.path') . '/'. $this->stream_id . '?'; - $query = http_build_query([ - 'key' => $this->stream_key, - 'ts' => time() - ]); - - return $proto . $host . $port . $path . $query; - } + public function getStreamRtmpUrl() + { + return $this->getStreamServer() . '/' . $this->stream_id; + } } From f78aa1f674e38336e13351b12fb4df8aab58e3b7 Mon Sep 17 00:00:00 2001 From: Daniel Supernault Date: Wed, 15 Jun 2022 00:46:57 -0600 Subject: [PATCH 3/6] Update livestream api routes --- routes/api.php | 2 ++ 1 file changed, 2 insertions(+) diff --git a/routes/api.php b/routes/api.php index 844942c36..00fc628b4 100644 --- a/routes/api.php +++ b/routes/api.php @@ -106,5 +106,7 @@ Route::group(['prefix' => 'api'], function() use($middleware) { Route::post('chat/message', 'LiveStreamController@addChatComment')->middleware($middleware); Route::post('chat/delete', 'LiveStreamController@deleteChatComment')->middleware($middleware); Route::get('config', 'LiveStreamController@getConfig')->middleware($middleware); + Route::post('broadcast/publish', 'LiveStreamController@clientBroadcastPublish'); + Route::post('broadcast/finish', 'LiveStreamController@clientBroadcastFinish'); }); }); From 11e99d782fafecdc5dbfdc91d9e0e6ac91f8aee0 Mon Sep 17 00:00:00 2001 From: Daniel Supernault Date: Wed, 15 Jun 2022 01:10:39 -0600 Subject: [PATCH 4/6] Update ApiV1Controller, fix network timeline --- app/Http/Controllers/Api/ApiV1Controller.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/Http/Controllers/Api/ApiV1Controller.php b/app/Http/Controllers/Api/ApiV1Controller.php index 3b1edace0..8723dfd7d 100644 --- a/app/Http/Controllers/Api/ApiV1Controller.php +++ b/app/Http/Controllers/Api/ApiV1Controller.php @@ -1984,7 +1984,7 @@ class ApiV1Controller extends Controller $remote = ($request->has('remote') && $request->input('remote') == true) || ($request->filled('local') && $request->input('local') != true); $filtered = $user ? UserFilterService::filters($user->profile_id) : []; - if($remote && config('instance.timeline.network.cached')) { + if((!$request->has('local') || $remote) && config('instance.timeline.network.cached')) { Cache::remember('api:v1:timelines:network:cache_check', 10368000, function() { if(NetworkTimelineService::count() == 0) { NetworkTimelineService::warmCache(true, config('instance.timeline.network.cache_dropoff')); From 7803b692f8cd254b5f301b960ce74ef8896b4b73 Mon Sep 17 00:00:00 2001 From: Daniel Supernault Date: Wed, 15 Jun 2022 01:11:25 -0600 Subject: [PATCH 5/6] Update changelog --- CHANGELOG.md | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index b2a2a3c63..884a62f8a 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,6 +4,7 @@ ### New Features - Custom content warnings/spoiler text ([d4864213](https://github.com/pixelfed/pixelfed/commit/d4864213)) +- Add NetworkTimelineService cache ([1310d95c](https://github.com/pixelfed/pixelfed/commit/1310d95c)) ### Breaking - Replaced `predis` with `phpredis` as default redis driver due to predis being deprecated, install [phpredis](https://github.com/phpredis/phpredis/blob/develop/INSTALL.markdown) if you're still using predis. @@ -27,6 +28,13 @@ - Refactor AP profileFetch logic to fix race conditions and improve updating fields and avatars ([505261da](https://github.com/pixelfed/pixelfed/commit/505261da)) - Update network timeline api, limit falloff to 2 days ([13a66303](https://github.com/pixelfed/pixelfed/commit/13a66303)) - Update Inbox, store follow request activity ([c82f2085](https://github.com/pixelfed/pixelfed/commit/c82f2085)) +- Update UserFilterService, improve cache strategy by using in-memory state via UserFilterObserver for empty lists with a ttl of 90 days ([9c17def4](https://github.com/pixelfed/pixelfed/commit/9c17def4)) +- Update ApiV1Controller, add network timeline support via NetworkTimelineService ([f54fd6e9](https://github.com/pixelfed/pixelfed/commit/f54fd6e9)) +- Bump max_collection_length default to 100 from 18 ([65cf9cca](https://github.com/pixelfed/pixelfed/commit/65cf9cca)) +- Improve follow request flow, federate rejections and delete rejections from database to properly handle future follow requests from same actor ([4470981a](https://github.com/pixelfed/pixelfed/commit/4470981a)) +- Update follower counts on follow_request approval ([e97900a0](https://github.com/pixelfed/pixelfed/commit/e97900a0)) +- Update ApiV1Controller, improve local/remote logic in public timeline endpoint ([4ff179ad](https://github.com/pixelfed/pixelfed/commit/4ff179ad)) +- Update ApiV1Controller, fix network timeline ([11e99d78](https://github.com/pixelfed/pixelfed/commit/11e99d78)) - ([](https://github.com/pixelfed/pixelfed/commit/)) ## [v0.11.3 (2022-05-09)](https://github.com/pixelfed/pixelfed/compare/v0.11.2...v0.11.3) From 154f23416a1f5cf75aeb49e38804f1007452b55e Mon Sep 17 00:00:00 2001 From: Daniel Supernault Date: Wed, 15 Jun 2022 01:20:03 -0600 Subject: [PATCH 6/6] Add live-hls storage dir and gitignore --- storage/app/public/.gitignore | 1 + storage/app/public/live-hls/.gitignore | 2 ++ 2 files changed, 3 insertions(+) create mode 100644 storage/app/public/live-hls/.gitignore diff --git a/storage/app/public/.gitignore b/storage/app/public/.gitignore index fefdd6c6f..6cf5852e5 100755 --- a/storage/app/public/.gitignore +++ b/storage/app/public/.gitignore @@ -5,3 +5,4 @@ !emoji/ !textimg/ !headers/ +!live-hls/ diff --git a/storage/app/public/live-hls/.gitignore b/storage/app/public/live-hls/.gitignore new file mode 100644 index 000000000..d6b7ef32c --- /dev/null +++ b/storage/app/public/live-hls/.gitignore @@ -0,0 +1,2 @@ +* +!.gitignore