1
0
Эх сурвалжийг харах

Merge pull request #3562 from pixelfed/staging

Staging
daniel 3 жил өмнө
parent
commit
04caa30e9a
39 өөрчлөгдсөн 3494 нэмэгдсэн , 402 устгасан
  1. 58 32
      .env.example
  2. 325 190
      app/Console/Commands/Installer.php
  3. 5 0
      app/Console/Commands/UserCreate.php
  4. 51 0
      app/Events/LiveStream/BanUser.php
  5. 51 0
      app/Events/LiveStream/DeleteChatComment.php
  6. 51 0
      app/Events/LiveStream/NewChatComment.php
  7. 51 0
      app/Events/LiveStream/PinChatMessage.php
  8. 48 0
      app/Events/LiveStream/StreamEnd.php
  9. 48 0
      app/Events/LiveStream/StreamStart.php
  10. 51 0
      app/Events/LiveStream/UnpinChatMessage.php
  11. 10 0
      app/Http/Controllers/Api/ApiV1Controller.php
  12. 141 26
      app/Http/Controllers/LiveStreamController.php
  13. 75 0
      app/Providers/TelescopeServiceProvider.php
  14. 1 1
      app/Services/LiveStreamService.php
  15. 1 1
      app/Transformer/Api/StatusStatelessTransformer.php
  16. 1 1
      app/Transformer/Api/StatusTransformer.php
  17. 2 0
      composer.json
  18. 1323 46
      composer.lock
  19. 2 1
      config/app.php
  20. 4 8
      config/broadcasting.php
  21. 133 0
      config/telescope.php
  22. 258 84
      config/websockets.php
  23. BIN
      public/js/collectioncompose.js
  24. BIN
      public/js/collections.js
  25. BIN
      public/js/home-ojtjadoml.js
  26. BIN
      public/js/installer.js
  27. 1 0
      public/js/installer.js.LICENSE.txt
  28. BIN
      public/js/live-player.js
  29. BIN
      public/js/spa.js
  30. BIN
      public/js/vendor.js
  31. 686 0
      public/js/vendor.js.LICENSE.txt
  32. BIN
      public/mix-manifest.json
  33. 4 0
      resources/assets/js/live-player.js
  34. 62 8
      resources/views/admin/diagnostics/home.blade.php
  35. 33 0
      resources/views/live/player.blade.php
  36. 3 3
      resources/views/site/about.blade.php
  37. 6 1
      routes/api.php
  38. 8 0
      routes/channels.php
  39. 1 0
      routes/web.php

+ 58 - 32
.env.example

@@ -1,35 +1,62 @@
-APP_NAME="Pixelfed Prod"
-APP_ENV=production
+APP_NAME="Pixelfed"
+APP_ENV="production"
 APP_KEY=
-APP_DEBUG=false
+APP_DEBUG="false"
 
-APP_URL=http://localhost
+# Instance Configuration
+OPEN_REGISTRATION="false"
+ENFORCE_EMAIL_VERIFICATION="false"
+PF_MAX_USERS="1000"
+OAUTH_ENABLED="true"
+
+# Media Configuration
+PF_OPTIMIZE_IMAGES="true"
+IMAGE_QUALITY="80"
+MAX_PHOTO_SIZE="15000"
+MAX_CAPTION_LENGTH="500"
+MAX_ALBUM_LENGTH="4"
+
+# Instance URL Configuration
+APP_URL="http://localhost"
 APP_DOMAIN="localhost"
 ADMIN_DOMAIN="localhost"
 SESSION_DOMAIN="localhost"
 TRUST_PROXIES="*"
 
-LOG_CHANNEL=stack
-
-DB_CONNECTION=mysql
-DB_HOST=127.0.0.1
-DB_PORT=3306
-DB_DATABASE=pixelfed
-DB_USERNAME=pixelfed
-DB_PASSWORD=pixelfed
+# Database Configuration
+DB_CONNECTION="mysql"
+DB_HOST="127.0.0.1"
+DB_PORT="3306"
+DB_DATABASE="pixelfed"
+DB_USERNAME="pixelfed"
+DB_PASSWORD="pixelfed"
 
-BROADCAST_DRIVER=log
-CACHE_DRIVER=redis
-SESSION_DRIVER=database
-QUEUE_DRIVER=redis
+# Redis Configuration
+REDIS_CLIENT="predis"
+REDIS_SCHEME="tcp"
+REDIS_HOST="127.0.0.1"
+REDIS_PASSWORD="null"
+REDIS_PORT="6379"
 
+# Laravel Configuration
+SESSION_DRIVER="database"
+CACHE_DRIVER="redis"
+QUEUE_DRIVER="redis"
+BROADCAST_DRIVER="log"
+LOG_CHANNEL="stack"
 HORIZON_PREFIX="horizon-"
 
-REDIS_SCHEME=tcp
-REDIS_HOST=127.0.0.1
-REDIS_PASSWORD=null
-REDIS_PORT=6379
+# ActivityPub Configuration
+ACTIVITY_PUB="false"
+AP_REMOTE_FOLLOW="false"
+AP_INBOX="false"
+AP_OUTBOX="false"
+AP_SHAREDINBOX="false"
 
+# Experimental Configuration
+EXP_EMC="true"
+
+## Mail Configuration (Post-Installer)
 MAIL_DRIVER=log
 MAIL_HOST=smtp.mailtrap.io
 MAIL_PORT=2525
@@ -39,15 +66,14 @@ MAIL_ENCRYPTION=null
 MAIL_FROM_ADDRESS="pixelfed@example.com"
 MAIL_FROM_NAME="Pixelfed"
 
-OPEN_REGISTRATION=true
-ENFORCE_EMAIL_VERIFICATION=true
-PF_MAX_USERS=1000
-
-MAX_PHOTO_SIZE=15000
-MAX_CAPTION_LENGTH=150
-MAX_ALBUM_LENGTH=4
-
-ACTIVITY_PUB=false
-AP_REMOTE_FOLLOW=false
-AP_INBOX=false
-PF_COSTAR_ENABLED=false
+## S3 Configuration (Post-Installer)
+PF_ENABLE_CLOUD=false
+FILESYSTEM_DRIVER=local
+FILESYSTEM_CLOUD=s3
+#AWS_ACCESS_KEY_ID=
+#AWS_SECRET_ACCESS_KEY=
+#AWS_DEFAULT_REGION=
+#AWS_BUCKET=<BucketName>
+#AWS_URL=
+#AWS_ENDPOINT=
+#AWS_USE_PATH_STYLE_ENDPOINT=false

+ 325 - 190
app/Console/Commands/Installer.php

@@ -57,52 +57,81 @@ class Installer extends Command
         $this->info(' ');
         $this->info('Pixelfed version: ' . config('pixelfed.version'));
         $this->line(' ');
+        $this->installerSteps();
+    }
+
+    protected function installerSteps()
+    {
         $this->envCheck();
+        $this->envCreate();
+        $this->installType();
+
+        if ($this->installType === 'Advanced') {
+            $this->info('Installer: Advanced...');
+            $this->checkPHPRequiredDependencies();
+            $this->checkFFmpegDependencies();
+            $this->checkOptimiseDependencies();
+            $this->checkDiskPermissions();
+            $this->envProd();
+            $this->instanceDB();
+            $this->instanceRedis();
+            $this->instanceURL();
+            $this->activityPubSettings();
+            $this->laravelSettings();
+            $this->instanceSettings();
+            $this->mediaSettings();
+            $this->dbMigrations();
+            $this->validateEnv();
+            $this->resetArtisanCache();
+        } else {
+            $this->info('Installer: Simple...');
+            $this->checkDiskPermissions();
+            $this->envProd();
+            $this->instanceDB();
+            $this->instanceRedis();
+            $this->instanceURL();
+            $this->activityPubSettings();
+            $this->instanceSettings();
+            $this->dbMigrations();
+            $this->validateEnv();
+            $this->resetArtisanCache();
+        }
     }
 
     protected function envCheck()
     {
-        if( file_exists(base_path('.env')) &&
-        	filesize(base_path('.env')) !== 0 &&
-        	!$this->option('dangerously-overwrite-env')
+        if (file_exists(base_path('.env')) &&
+            filesize(base_path('.env')) !== 0 &&
+            !$this->option('dangerously-overwrite-env')
         ) {
             $this->line('');
-            $this->error('Installation aborted, found existing .env file');
-            $this->line('Run the following command to re-run the installer:');
-            $this->line('');
-            $this->info('php artisan install --dangerously-overwrite-env');
+            $this->error('Existing .env File Found - Installation Aborted');
+            $this->line('Run the following command to re-run the installer: php artisan install --dangerously-overwrite-env');
             $this->line('');
             exit;
         }
-        $this->installType();
     }
 
-    protected function installType()
+    protected function envCreate()
     {
-    	$type = $this->choice('Select installation type', ['Simple', 'Advanced'], 0);
-		$this->installType = $type;
-        $this->preflightCheck();
+        $this->line('');
+        $this->info('Creating .env if required');
+        if (!file_exists(app()->environmentFilePath())) {
+            exec('cp .env.example .env');
+        }
     }
 
-    protected function preflightCheck()
+    protected function installType()
     {
-        if($this->installType === 'Advanced') {
-			$this->info('Scanning system...');
-			$this->line(' ');
-			$this->info('Checking for installed dependencies...');
-	        $redis = Redis::connection();
-	        if($redis->ping()) {
-	            $this->info('- Found redis!');
-	        } else {
-	            $this->error('- Redis not found, aborting installation');
-	            exit;
-	        }
-        }
-        $this->checkPhpDependencies();
+        $type = $this->choice('Select installation type', ['Simple', 'Advanced'], 1);
+        $this->installType = $type;
     }
 
-    protected function checkPhpDependencies()
+    protected function checkPHPRequiredDependencies()
     {
+        $this->line(' ');
+        $this->info('Checking for Required PHP Extensions...');
+
         $extensions = [
             'bcmath',
             'ctype',
@@ -110,197 +139,326 @@ class Installer extends Command
             'json',
             'mbstring',
             'openssl',
+            'gd',
+            'intl',
+            'xml',
+            'zip',
+            'redis',
         ];
-        if($this->installType === 'Advanced') {
-	        $ffmpeg = exec('which ffmpeg');
-	        if(empty($ffmpeg)) {
-	            $this->error('FFmpeg not found, please install it.');
-	            $this->error('Cancelling installation.');
-	            exit;
-	        } else {
-	            $this->info('- Found FFmpeg!');
-	        }
-	        $this->line('');
-        	$this->info('Checking for required php extensions...');
-	    }
-        foreach($extensions as $ext) {
-            if(extension_loaded($ext) == false) {
-                $this->error("\"{$ext}\" PHP extension not found, aborting installation");
-                exit;
+
+        foreach ($extensions as $ext) {
+            if (extension_loaded($ext) == false) {
+                $this->error("- \"{$ext}\" not found");
+            } else {
+                $this->info("- \"{$ext}\" found");
             }
         }
-        if($this->installType === 'Advanced') {
-	        $this->info("- Required PHP extensions found!");
-	    }
 
-	    $this->checkPermissions();
+        $continue = $this->choice('Do you wish to continue?', ['yes', 'no'], 0);
+        $this->continue = $continue;
+        if ($this->continue === 'no') {
+            $this->info('Exiting Installer.');
+            exit;
+        }
+
+    }
+
+    protected function checkFFmpegDependencies()
+    {
+        $this->line(' ');
+        $this->info('Checking for Required FFmpeg dependencies...');
+
+        $ffmpeg = exec('which ffmpeg');
+        if (empty($ffmpeg)) {
+            $this->error("- \"{$ext}\" FFmpeg not found, aborting installation");
+            exit;
+        } else {
+            $this->info('- Found FFmpeg!');
+        }
     }
 
-    protected function checkPermissions()
+    protected function checkOptimiseDependencies()
     {
-    	if($this->installType === 'Advanced') {
-	        $this->line('');
-	        $this->info('Checking for proper filesystem permissions...');
-	    }
+        $this->line(' ');
+        $this->info('Checking for Optional Media Optimisation dependencies...');
+
+        $dependencies = [
+            'jpegoptim',
+            'optipng',
+            'pngquant',
+            'gifsicle',
+        ];
+
+        foreach ($dependencies as $dep) {
+            $which = exec("which $dep");
+            if (empty($which)) {
+                $this->error("- \"{$dep}\" not found");
+            } else {
+                $this->info("- \"{$dep}\" found");
+            }
+        }
+    }
+
+    protected function checkDiskPermissions()
+    {
+        $this->line('');
+        $this->info('Checking for proper filesystem permissions...');
+        $this->callSilently('storage:link');
 
         $paths = [
             base_path('bootstrap'),
-            base_path('storage')
+            base_path('storage'),
         ];
 
-        foreach($paths as $path) {
-            if(is_writeable($path) == false) {
+        foreach ($paths as $path) {
+            if (is_writeable($path) == false) {
                 $this->error("- Invalid permission found! Aborting installation.");
                 $this->error("  Please make the following path writeable by the web server:");
                 $this->error("  $path");
                 exit;
             } else {
-            	if($this->installType === 'Advanced') {
-	                $this->info("- Found valid permissions for {$path}");
-	            }
+                $this->info("- Found valid permissions for {$path}");
             }
         }
-
-        $this->createEnv();
     }
 
-    protected function createEnv()
+    protected function envProd()
     {
         $this->line('');
-        if(!file_exists(app()->environmentFilePath())) {
-            exec('cp .env.example .env');
-            $this->updateEnvFile('APP_ENV', 'setup');
-            $this->call('key:generate');
-        }
+        $this->info('Enabling production');
 
-        $name = $this->ask('Site name [ex: Pixelfed]');
-        $this->updateEnvFile('APP_NAME', $name ?? 'pixelfed');
-
-        $domain = $this->ask('Site Domain [ex: pixelfed.com]');
-        if(empty($domain)) {
-        	$this->error('You must set the site domain');
-        	exit;
-        }
-        if(starts_with($domain, 'http')) {
-        	$this->error('The site domain cannot start with https://, you must use the FQDN (eg: example.org)');
-        	exit;
-        }
-        if(strpos($domain, '.') == false) {
-        	$this->error('You must enter a valid site domain');
-        	exit;
-        }
-        $this->updateEnvFile('APP_DOMAIN', $domain ?? 'example.org');
-        $this->updateEnvFile('ADMIN_DOMAIN', $domain ?? 'example.org');
-        $this->updateEnvFile('SESSION_DOMAIN', $domain ?? 'example.org');
-        $this->updateEnvFile('APP_URL', 'https://' . $domain);
+        $this->updateEnvFile('APP_ENV', 'production');
+        $this->updateEnvFile('APP_DEBUG', 'false');
+        $this->call('key:generate', ['--force' => true]);
+    }
 
+    protected function instanceDB()
+    {
+        $this->line('');
+        $this->info('Database Settings:');
         $database = $this->choice('Select database driver', ['mysql', 'pgsql'], 0);
-        $this->updateEnvFile('DB_CONNECTION', $database ?? 'mysql');
-
         $database_host = $this->ask('Select database host', '127.0.0.1');
-        $this->updateEnvFile('DB_HOST', $database_host ?? 'mysql');
-
         $database_port_default = $database === 'mysql' ? 3306 : 5432;
         $database_port = $this->ask('Select database port', $database_port_default);
-        $this->updateEnvFile('DB_PORT', $database_port ?? $database_port_default);
 
         $database_db = $this->ask('Select database', 'pixelfed');
-        $this->updateEnvFile('DB_DATABASE', $database_db ?? 'pixelfed');
-
         $database_username = $this->ask('Select database username', 'pixelfed');
-        $this->updateEnvFile('DB_USERNAME', $database_username ?? 'pixelfed');
+        $database_password = $this->secret('Select database password');
 
-        $db_pass = str_random(64);
-        $database_password = $this->secret('Select database password', $db_pass);
+        $this->updateEnvFile('DB_CONNECTION', $database);
+        $this->updateEnvFile('DB_HOST', $database_host);
+        $this->updateEnvFile('DB_PORT', $database_port);
+        $this->updateEnvFile('DB_DATABASE', $database_db);
+        $this->updateEnvFile('DB_USERNAME', $database_username);
         $this->updateEnvFile('DB_PASSWORD', $database_password);
 
+        $this->info('Testing Database...');
         $dsn = "{$database}:dbname={$database_db};host={$database_host};port={$database_port};";
         try {
-        	$dbh = new PDO($dsn, $database_username, $database_password, [PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION]);
+            $dbh = new PDO($dsn, $database_username, $database_password, [PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION]);
         } catch (\PDOException $e) {
-        	$this->error('Cannot connect to database, check your credentials and try again');
-        	exit;
+            $this->error('Cannot connect to database, check your details and try again');
+            exit;
         }
+        $this->info('- Connected to DB Successfully');
+    }
 
-        if($this->installType === 'Advanced') {
-	        $cache = $this->choice('Select cache driver', ["redis", "apc", "array", "database", "file", "memcached"], 0);
-	        $this->updateEnvFile('CACHE_DRIVER', $cache ?? 'redis');
+    protected function instanceRedis()
+    {
+        $this->line('');
+        $this->info('Redis Settings:');
+        $redis_client = $this->choice('Set redis client (PHP extension)', ['phpredis', 'predis'], 0);
+        $redis_host = $this->ask('Set redis host', 'localhost');
+        $redis_password = $this->ask('Set redis password', 'null');
+        $redis_port = $this->ask('Set redis port', 6379);
+
+        $this->updateEnvFile('REDIS_CLIENT', $redis_client);
+        $this->updateEnvFile('REDIS_SCHEME', 'tcp');
+        $this->updateEnvFile('REDIS_HOST', $redis_host);
+        $this->updateEnvFile('REDIS_PASSWORD', $redis_password);
+        $this->updateEnvFile('REDIS_PORT', $redis_port);
+
+        $this->info('Testing Redis...');
+        $redis = Redis::connection();
+        if ($redis->ping()) {
+            $this->info('- Connected to Redis Successfully!');
+        } else {
+            $this->error('Cannot connect to Redis, check your details and try again');
+            exit;
+        }
+    }
 
-	        $session = $this->choice('Select session driver', ["redis", "file", "cookie", "database", "apc", "memcached", "array"], 0);
-	        $this->updateEnvFile('SESSION_DRIVER', $session ?? 'redis');
+    protected function instanceURL()
+    {
+        $this->line('');
+        $this->info('Instance URL Settings:');
+        $name = $this->ask('Site name [ex: Pixelfed]', 'Pixelfed');
 
-	        $redis_host = $this->ask('Set redis host', 'localhost');
-	        $this->updateEnvFile('REDIS_HOST', $redis_host);
+        $domain = $this->ask('Site Domain [ex: pixelfed.com]');
+        $domain = strtolower($domain);
+        if (empty($domain)) {
+            $this->error('You must set the site domain');
+            exit;
+        }
+        if (starts_with($domain, 'http')) {
+            $this->error('The site domain cannot start with https://, you must use the FQDN (eg: example.org)');
+            exit;
+        }
+        if (strpos($domain, '.') == false) {
+            $this->error('You must enter a valid site domain');
+            exit;
+        }
 
-	        $redis_password = $this->ask('Set redis password', 'null');
-	        $this->updateEnvFile('REDIS_PASSWORD', $redis_password);
+        $this->updateEnvFile('APP_NAME', $name);
+        $this->updateEnvFile('APP_URL', 'https://' . $domain);
+        $this->updateEnvFile('APP_DOMAIN', $domain);
+        $this->updateEnvFile('ADMIN_DOMAIN', $domain);
+        $this->updateEnvFile('SESSION_DOMAIN', $domain);
+    }
 
-	        $redis_port = $this->ask('Set redis port', 6379);
-	        $this->updateEnvFile('REDIS_PORT', $redis_port);
-	    }
+    protected function laravelSettings()
+    {
+        $this->line('');
+        $this->info('Laravel Settings (Defaults are recommended):');
+        $session = $this->choice('Select session driver', ["database", "file", "cookie", "redis", "memcached", "array"], 0);
+        $cache = $this->choice('Select cache driver', ["redis", "apc", "array", "database", "file", "memcached"], 0);
+        $queue = $this->choice('Select queue driver', ["redis", "database", "sync", "beanstalkd", "sqs", "null"], 0);
+        $broadcast = $this->choice('Select broadcast driver', ["log", "redis", "pusher", "null"], 0);
+        $log = $this->choice('Select Log Channel', ["stack", "single", "daily", "stderr", "syslog", "null"], 0);
+        $horizon = $this->ask('Set Horizon Prefix [ex: horizon-]', 'horizon-');
+
+        $this->updateEnvFile('SESSION_DRIVER', $session);
+        $this->updateEnvFile('CACHE_DRIVER', $cache);
+        $this->updateEnvFile('QUEUE_DRIVER', $queue);
+        $this->updateEnvFile('BROADCAST_DRIVER', $broadcast);
+        $this->updateEnvFile('LOG_CHANNEL', $log);
+        $this->updateEnvFile('HORIZON_PREFIX', $horizon);
+    }
 
+    protected function instanceSettings()
+    {
+        $this->line('');
+        $this->info('Instance Settings:');
+        $max_registration = $this->ask('Set Maximum users on this instance.', '1000');
         $open_registration = $this->choice('Allow new registrations?', ['false', 'true'], 0);
+        $enforce_email_verification = $this->choice('Enforce email verification?', ['false', 'true'], 0);
+        $enable_mobile_apis = $this->choice('Enable mobile app/apis support?', ['false', 'true'], 1);
+
+        $this->updateEnvFile('PF_MAX_USERS', $max_registration);
         $this->updateEnvFile('OPEN_REGISTRATION', $open_registration);
+        $this->updateEnvFile('ENFORCE_EMAIL_VERIFICATION', $enforce_email_verification);
+        $this->updateEnvFile('OAUTH_ENABLED', $enable_mobile_apis);
+        $this->updateEnvFile('EXP_EMC', $enable_mobile_apis);
+    }
 
+    protected function activityPubSettings()
+    {
+        $this->line('');
+        $this->info('Federation Settings:');
         $activitypub_federation = $this->choice('Enable ActivityPub federation?', ['false', 'true'], 1);
+
         $this->updateEnvFile('ACTIVITY_PUB', $activitypub_federation);
+        $this->updateEnvFile('AP_REMOTE_FOLLOW', $activitypub_federation);
         $this->updateEnvFile('AP_INBOX', $activitypub_federation);
+        $this->updateEnvFile('AP_OUTBOX', $activitypub_federation);
         $this->updateEnvFile('AP_SHAREDINBOX', $activitypub_federation);
-        $this->updateEnvFile('AP_REMOTE_FOLLOW', $activitypub_federation);
+    }
 
-        $enforce_email_verification = $this->choice('Enforce email verification?', ['false', 'true'], 1);
-        $this->updateEnvFile('ENFORCE_EMAIL_VERIFICATION', $enforce_email_verification);
+    protected function mediaSettings()
+    {
+        $this->line('');
+        $this->info('Media Settings:');
+        $optimize_media = $this->choice('Optimize media uploads? Requires jpegoptim and other dependencies!', ['false', 'true'], 1);
+        $image_quality = $this->ask('Set image optimization quality between 1-100. Default is 80%, lower values use less disk space at the expense of image quality.', '80');
+        if ($image_quality < 1) {
+            $this->error('Min image quality is 1. You should avoid such a low value, 60 at minimum is recommended.');
+            exit;
+        }
+        if ($image_quality > 100) {
+            $this->error('Max image quality is 100');
+            exit;
+        }
+        $this->info('Note: Max photo size cannot exceed `post_max_size` in php.ini.');
+        $max_photo_size = $this->ask('Max photo upload size in kilobytes. Default 15000 which is equal to 15MB', '15000');
 
-        $enable_mobile_apis = $this->choice('Enable mobile app/apis support?', ['false', 'true'], 1);
-        $this->updateEnvFile('OAUTH_ENABLED', $enable_mobile_apis);
-        $this->updateEnvFile('EXP_EMC', $enable_mobile_apis);
+        $max_caption_length = $this->ask('Max caption limit. Default to 500, max 5000.', '500');
+        if ($max_caption_length > 5000) {
+            $this->error('Max caption length is 5000 characters.');
+            exit;
+        }
 
-    	$optimize_media = $this->choice('Optimize media uploads? Requires jpegoptim and other dependencies!', ['false', 'true'], 0);
-    	$this->updateEnvFile('PF_OPTIMIZE_IMAGES', $optimize_media);
-
-        if($this->installType === 'Advanced') {
-
-        	if($optimize_media === 'true') {
-	        	$image_quality = $this->ask('Set image optimization quality between 1-100. Default is 80%, lower values use less disk space at the expense of image quality.', '80');
-	        	if($image_quality < 1) {
-	        		$this->error('Min image quality is 1. You should avoid such a low value, 60 at minimum is recommended.');
-	        		exit;
-	        	}
-	        	if($image_quality > 100) {
-	        		$this->error('Max image quality is 100');
-	        		exit;
-	        	}
-	    		$this->updateEnvFile('IMAGE_QUALITY', $image_quality);
-        	}
-
-        	$max_photo_size = $this->ask('Max photo upload size in kilobytes. Default 15000 which is equal to 15MB', '15000');
-        	if($max_photo_size * 1024 > $this->parseSize(ini_get('post_max_size'))) {
-        		$this->error('Max photo size (' . (round($max_photo_size / 1000)) . 'M) cannot exceed php.ini `post_max_size` of ' . ini_get('post_max_size'));
-        		exit;
-        	}
-        	$this->updateEnvFile('MAX_PHOTO_SIZE', $max_photo_size);
-
-        	$max_caption_length = $this->ask('Max caption limit. Default to 500, max 5000.', '500');
-        	if($max_caption_length > 5000) {
-        		$this->error('Max caption length is 5000 characters.');
-        		exit;
-        	}
-        	$this->updateEnvFile('MAX_CAPTION_LENGTH', $max_caption_length);
-
-        	$max_album_length = $this->ask('Max photos allowed per album. Choose a value between 1 and 10.', '4');
-        	if($max_album_length < 1) {
-        		$this->error('Min album length is 1 photos per album.');
-        		exit;
-        	}
-        	if($max_album_length > 10) {
-        		$this->error('Max album length is 10 photos per album.');
-        		exit;
-        	}
-        	$this->updateEnvFile('MAX_ALBUM_LENGTH', $max_album_length);
+        $max_album_length = $this->ask('Max photos allowed per album. Choose a value between 1 and 10.', '4');
+        if ($max_album_length < 1) {
+            $this->error('Min album length is 1 photos per album.');
+            exit;
+        }
+        if ($max_album_length > 10) {
+            $this->error('Max album length is 10 photos per album.');
+            exit;
         }
 
-        $this->updateEnvFile('APP_ENV', 'production');
-        $this->postInstall();
+        $this->updateEnvFile('PF_OPTIMIZE_IMAGES', $optimize_media);
+        $this->updateEnvFile('IMAGE_QUALITY', $image_quality);
+        $this->updateEnvFile('MAX_PHOTO_SIZE', $max_photo_size);
+        $this->updateEnvFile('MAX_CAPTION_LENGTH', $max_caption_length);
+        $this->updateEnvFile('MAX_ALBUM_LENGTH', $max_album_length);
+    }
+
+    protected function dbMigrations()
+    {
+        $this->line('');
+        $this->info('Note: We recommend running database migrations now!');
+        $confirm = $this->choice('Do you want to run the database migrations?', ['Yes', 'No'], 0);
+
+        if ($confirm === 'Yes') {
+            sleep(3);
+            $this->line('');
+            $this->info('Migrating DB:');
+            $this->call('migrate', ['--force' => true]);
+            $this->line('');
+            $this->info('Importing Cities:');
+            $this->call('import:cities');
+            $this->line('');
+            $this->info('Creating Federation Instance Actor:');
+            $this->call('instance:actor');
+            $this->line('');
+            $this->info('Creating Password Keys for API:');
+            $this->call('passport:keys', ['--force' => true]);
+
+            $confirm = $this->choice('Do you want to create an admin account?', ['Yes', 'No'], 0);
+            if ($confirm === 'Yes') {
+                $this->call('user:create');
+            }
+        }
+    }
+
+    protected function resetArtisanCache()
+    {
+        $this->call('config:cache');
+        $this->call('route:cache');
+        $this->call('view:cache');
+    }
+
+    protected function validateEnv()
+    {
+        $this->checkEnvKeys('APP_KEY', "key:generate failed?");
+        $this->checkEnvKeys('APP_ENV', "APP_ENV value should be production");
+        $this->checkEnvKeys('APP_DEBUG', "APP_DEBUG value should be false");
+    }
+
+#####
+    # Installer Functions
+    #####
+
+    protected function checkEnvKeys($key, $error)
+    {
+        $envPath = app()->environmentFilePath();
+        $payload = file_get_contents($envPath);
+
+        if ($existing = $this->existingEnv($key, $payload)) {
+        } else {
+            $this->error("$key empty - $error");
+        }
     }
 
     protected function updateEnvFile($key, $value)
@@ -333,37 +491,14 @@ class Installer extends Command
         fclose($file);
     }
 
-    protected function postInstall()
+    protected function parseSize($size)
     {
-        $this->line('');
-        $this->info('We recommend running database migrations now, or you can do it manually later.');
-        $confirm = $this->choice('Do you want to run the database migrations?', ['No', 'Yes'], 0);
-        if($confirm === 'Yes') {
-        	$this->callSilently('config:clear');
-        	sleep(3);
-        	$this->call('migrate', ['--force' => true]);
-	        $this->callSilently('instance:actor');
-	        $this->callSilently('passport:install');
-
-	        $confirm = $this->choice('Do you want to create an admin account?', ['No', 'Yes'], 0);
-	        if($confirm === 'Yes') {
-	        	$this->call('user:create');
-	        }
+        $unit = preg_replace('/[^bkmgtpezy]/i', '', $size);
+        $size = preg_replace('/[^0-9\.]/', '', $size);
+        if ($unit) {
+            return round($size * pow(1024, stripos('bkmgtpezy', $unit[0])));
         } else {
-        	$this->callSilently('config:cache');
+            return round($size);
         }
-
-        $this->info('Pixelfed has been successfully installed!');
-    }
-
-    protected function parseSize($size) {
-    	$unit = preg_replace('/[^bkmgtpezy]/i', '', $size);
-    	$size = preg_replace('/[^0-9\.]/', '', $size);
-    	if ($unit) {
-    		return round($size * pow(1024, stripos('bkmgtpezy', $unit[0])));
-    	}
-    	else {
-    		return round($size);
-    	}
     }
 }

+ 5 - 0
app/Console/Commands/UserCreate.php

@@ -84,6 +84,11 @@ class UserCreate extends Command
             exit;
         }
         
+        if (strlen($password) < 6) {
+            $this->error('Must be 6 or more characters, please try again...');
+            exit;
+        }
+        
         $is_admin = $this->confirm('Make this user an admin?');
         $confirm_email = $this->confirm('Manually verify email address?');
 

+ 51 - 0
app/Events/LiveStream/BanUser.php

@@ -0,0 +1,51 @@
+<?php
+
+namespace App\Events\LiveStream;
+
+use Illuminate\Broadcasting\Channel;
+use Illuminate\Broadcasting\InteractsWithSockets;
+use Illuminate\Broadcasting\PresenceChannel;
+use Illuminate\Broadcasting\PrivateChannel;
+use Illuminate\Contracts\Broadcasting\ShouldBroadcast;
+use Illuminate\Foundation\Events\Dispatchable;
+use Illuminate\Queue\SerializesModels;
+use App\Models\LiveStream;
+
+class BanUser implements ShouldBroadcast
+{
+    use Dispatchable, InteractsWithSockets, SerializesModels;
+
+    public $livestream;
+    public $profileId;
+
+    /**
+     * Create a new event instance.
+     *
+     * @return void
+     */
+    public function __construct(LiveStream $livestream, $profileId)
+    {
+        $this->livestream = $livestream;
+        $this->profileId = $profileId;
+    }
+
+    /**
+     * Get the channels the event should broadcast on.
+     *
+     * @return \Illuminate\Broadcasting\Channel|array
+     */
+    public function broadcastOn()
+    {
+        return new Channel('live.chat.' . $this->livestream->profile_id);
+    }
+
+    public function broadcastAs()
+    {
+        return 'chat.ban-user';
+    }
+
+    public function broadcastWith()
+    {
+        return ['id' => $this->profileId];
+    }
+}

+ 51 - 0
app/Events/LiveStream/DeleteChatComment.php

@@ -0,0 +1,51 @@
+<?php
+
+namespace App\Events\LiveStream;
+
+use Illuminate\Broadcasting\Channel;
+use Illuminate\Broadcasting\InteractsWithSockets;
+use Illuminate\Broadcasting\PresenceChannel;
+use Illuminate\Broadcasting\PrivateChannel;
+use Illuminate\Contracts\Broadcasting\ShouldBroadcast;
+use Illuminate\Foundation\Events\Dispatchable;
+use Illuminate\Queue\SerializesModels;
+use App\Models\LiveStream;
+
+class DeleteChatComment implements ShouldBroadcast
+{
+    use Dispatchable, InteractsWithSockets, SerializesModels;
+
+    public $livestream;
+    public $chatmsg;
+
+    /**
+     * Create a new event instance.
+     *
+     * @return void
+     */
+    public function __construct(LiveStream $livestream, $chatmsg)
+    {
+        $this->livestream = $livestream;
+        $this->chatmsg = $chatmsg;
+    }
+
+    /**
+     * Get the channels the event should broadcast on.
+     *
+     * @return \Illuminate\Broadcasting\Channel|array
+     */
+    public function broadcastOn()
+    {
+        return new Channel('live.chat.' . $this->livestream->profile_id);
+    }
+
+    public function broadcastAs()
+    {
+        return 'chat.delete-message';
+    }
+
+    public function broadcastWith()
+    {
+        return ['id' => $this->chatmsg['id']];
+    }
+}

+ 51 - 0
app/Events/LiveStream/NewChatComment.php

@@ -0,0 +1,51 @@
+<?php
+
+namespace App\Events\LiveStream;
+
+use Illuminate\Broadcasting\Channel;
+use Illuminate\Broadcasting\InteractsWithSockets;
+use Illuminate\Broadcasting\PresenceChannel;
+use Illuminate\Broadcasting\PrivateChannel;
+use Illuminate\Contracts\Broadcasting\ShouldBroadcast;
+use Illuminate\Foundation\Events\Dispatchable;
+use Illuminate\Queue\SerializesModels;
+use App\Models\LiveStream;
+
+class NewChatComment implements ShouldBroadcast
+{
+    use Dispatchable, InteractsWithSockets, SerializesModels;
+
+    public $livestream;
+    public $chatmsg;
+
+    /**
+     * Create a new event instance.
+     *
+     * @return void
+     */
+    public function __construct(LiveStream $livestream, $chatmsg)
+    {
+        $this->livestream = $livestream;
+        $this->chatmsg = $chatmsg;
+    }
+
+    /**
+     * Get the channels the event should broadcast on.
+     *
+     * @return \Illuminate\Broadcasting\Channel|array
+     */
+    public function broadcastOn()
+    {
+        return new Channel('live.chat.' . $this->livestream->profile_id);
+    }
+
+    public function broadcastAs()
+    {
+        return 'chat.new-message';
+    }
+
+    public function broadcastWith()
+    {
+        return ['msg' => $this->chatmsg];
+    }
+}

+ 51 - 0
app/Events/LiveStream/PinChatMessage.php

@@ -0,0 +1,51 @@
+<?php
+
+namespace App\Events\LiveStream;
+
+use Illuminate\Broadcasting\Channel;
+use Illuminate\Broadcasting\InteractsWithSockets;
+use Illuminate\Broadcasting\PresenceChannel;
+use Illuminate\Broadcasting\PrivateChannel;
+use Illuminate\Contracts\Broadcasting\ShouldBroadcast;
+use Illuminate\Foundation\Events\Dispatchable;
+use Illuminate\Queue\SerializesModels;
+use App\Models\LiveStream;
+
+class PinChatMessage implements ShouldBroadcast
+{
+    use Dispatchable, InteractsWithSockets, SerializesModels;
+
+    public $livestream;
+    public $chatmsg;
+
+    /**
+     * Create a new event instance.
+     *
+     * @return void
+     */
+    public function __construct(LiveStream $livestream, $chatmsg)
+    {
+        $this->livestream = $livestream;
+        $this->chatmsg = $chatmsg;
+    }
+
+    /**
+     * Get the channels the event should broadcast on.
+     *
+     * @return \Illuminate\Broadcasting\Channel|array
+     */
+    public function broadcastOn()
+    {
+        return new Channel('live.chat.' . $this->livestream->profile_id);
+    }
+
+    public function broadcastAs()
+    {
+        return 'chat.pin-message';
+    }
+
+    public function broadcastWith()
+    {
+        return $this->chatmsg;
+    }
+}

+ 48 - 0
app/Events/LiveStream/StreamEnd.php

@@ -0,0 +1,48 @@
+<?php
+
+namespace App\Events\LiveStream;
+
+use Illuminate\Broadcasting\Channel;
+use Illuminate\Broadcasting\InteractsWithSockets;
+use Illuminate\Broadcasting\PresenceChannel;
+use Illuminate\Broadcasting\PrivateChannel;
+use Illuminate\Contracts\Broadcasting\ShouldBroadcast;
+use Illuminate\Foundation\Events\Dispatchable;
+use Illuminate\Queue\SerializesModels;
+
+class StreamEnd implements ShouldBroadcast
+{
+    use Dispatchable, InteractsWithSockets, SerializesModels;
+
+    public $id;
+
+    /**
+     * Create a new event instance.
+     *
+     * @return void
+     */
+    public function __construct($id)
+    {
+        $this->id = $id;
+    }
+
+    /**
+     * Get the channels the event should broadcast on.
+     *
+     * @return \Illuminate\Broadcasting\Channel|array
+     */
+    public function broadcastOn()
+    {
+        return new Channel('live.chat.' . $this->id);
+    }
+
+    public function broadcastAs()
+    {
+        return 'stream.end';
+    }
+
+    public function broadcastWith()
+    {
+        return ['ts' => time() ];
+    }
+}

+ 48 - 0
app/Events/LiveStream/StreamStart.php

@@ -0,0 +1,48 @@
+<?php
+
+namespace App\Events\LiveStream;
+
+use Illuminate\Broadcasting\Channel;
+use Illuminate\Broadcasting\InteractsWithSockets;
+use Illuminate\Broadcasting\PresenceChannel;
+use Illuminate\Broadcasting\PrivateChannel;
+use Illuminate\Contracts\Broadcasting\ShouldBroadcast;
+use Illuminate\Foundation\Events\Dispatchable;
+use Illuminate\Queue\SerializesModels;
+
+class StreamStart implements ShouldBroadcast
+{
+    use Dispatchable, InteractsWithSockets, SerializesModels;
+
+    public $id;
+
+    /**
+     * Create a new event instance.
+     *
+     * @return void
+     */
+    public function __construct($id)
+    {
+        $this->id = $id;
+    }
+
+    /**
+     * Get the channels the event should broadcast on.
+     *
+     * @return \Illuminate\Broadcasting\Channel|array
+     */
+    public function broadcastOn()
+    {
+        return new Channel('live.chat.' . $this->id);
+    }
+
+    public function broadcastAs()
+    {
+        return 'stream.start';
+    }
+
+    public function broadcastWith()
+    {
+        return ['ts' => time() ];
+    }
+}

+ 51 - 0
app/Events/LiveStream/UnpinChatMessage.php

@@ -0,0 +1,51 @@
+<?php
+
+namespace App\Events\LiveStream;
+
+use Illuminate\Broadcasting\Channel;
+use Illuminate\Broadcasting\InteractsWithSockets;
+use Illuminate\Broadcasting\PresenceChannel;
+use Illuminate\Broadcasting\PrivateChannel;
+use Illuminate\Contracts\Broadcasting\ShouldBroadcast;
+use Illuminate\Foundation\Events\Dispatchable;
+use Illuminate\Queue\SerializesModels;
+use App\Models\LiveStream;
+
+class UnpinChatMessage implements ShouldBroadcast
+{
+    use Dispatchable, InteractsWithSockets, SerializesModels;
+
+    public $livestream;
+    public $chatmsg;
+
+    /**
+     * Create a new event instance.
+     *
+     * @return void
+     */
+    public function __construct(LiveStream $livestream, $chatmsg)
+    {
+        $this->livestream = $livestream;
+        $this->chatmsg = $chatmsg;
+    }
+
+    /**
+     * Get the channels the event should broadcast on.
+     *
+     * @return \Illuminate\Broadcasting\Channel|array
+     */
+    public function broadcastOn()
+    {
+        return new Channel('live.chat.' . $this->livestream->profile_id);
+    }
+
+    public function broadcastAs()
+    {
+        return 'chat.unpin-message';
+    }
+
+    public function broadcastWith()
+    {
+        return $this->chatmsg;
+    }
+}

+ 10 - 0
app/Http/Controllers/Api/ApiV1Controller.php

@@ -101,6 +101,16 @@ class ApiV1Controller extends Controller
 		return response()->json($res, $code, $headers, JSON_UNESCAPED_SLASHES);
 	}
 
+	public function getWebsocketConfig()
+	{
+		return config('broadcasting.default') === 'pusher' ? [
+			'host' => config('broadcasting.connections.pusher.options.host'),
+			'port' => config('broadcasting.connections.pusher.options.port'),
+			'key' => config('broadcasting.connections.pusher.key'),
+			'cluster' => config('broadcasting.connections.pusher.options.cluster')
+		] : [];
+	}
+
 	public function getApp(Request $request)
 	{
 		if(!$request->user()) {

+ 141 - 26
app/Http/Controllers/LiveStreamController.php

@@ -9,6 +9,14 @@ use Illuminate\Support\Facades\Storage;
 use App\Services\AccountService;
 use App\Services\FollowerService;
 use App\Services\LiveStreamService;
+use App\User;
+use App\Events\LiveStream\NewChatComment;
+use App\Events\LiveStream\DeleteChatComment;
+use App\Events\LiveStream\BanUser;
+use App\Events\LiveStream\PinChatMessage;
+use App\Events\LiveStream\UnpinChatMessage;
+use App\Events\LiveStream\StreamStart;
+use App\Events\LiveStream\StreamEnd;
 
 class LiveStreamController extends Controller
 {
@@ -63,32 +71,48 @@ class LiveStreamController extends Controller
 		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'))
+			->whereNotNull('live_at')
+			->orderByDesc('live_at')
+			->first();
 
 		if(!$stream) {
 			return [];
 		}
 
 		$res = [];
-		$owner = $stream->profile_id == $request->user()->profile_id;
+		$owner = $request->user() ? $stream->profile_id == $request->user()->profile_id : false;
 
 		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();
-		}
+		$res = [
+			'hls_url' => $stream->getHlsUrl(),
+			'name' => $stream->name,
+			'description' => $stream->description
+		];
+
+		return response()->json($res, 200, [], JSON_UNESCAPED_SLASHES);
+	}
+
+
+	public function getUserStreamAsGuest(Request $request)
+	{
+		abort_if(!config('livestreaming.enabled'), 400);
 
-		if($stream->live_at == null) {
-			$res['hls_url'] = null;
-			$res['name'] = $stream->name;
-			$res['description'] = $stream->description;
-			return $res;
+		$stream = LiveStream::whereProfileId($request->input('profile_id'))
+			->whereVisibility('public')
+			->whereNotNull('live_at')
+			->orderByDesc('live_at')
+			->first();
+
+		if(!$stream) {
+			return [];
 		}
 
+		$res = [];
+
 		$res = [
 			'hls_url' => $stream->getHlsUrl(),
 			'name' => $stream->name,
@@ -98,6 +122,21 @@ class LiveStreamController extends Controller
 		return response()->json($res, 200, [], JSON_UNESCAPED_SLASHES);
 	}
 
+	public function showProfilePlayer(Request $request, $username)
+	{
+		abort_if(!config('livestreaming.enabled'), 400);
+
+		$user = User::whereUsername($username)->firstOrFail();
+		$id = (string) $user->profile_id;
+		$stream = LiveStream::whereProfileId($id)
+			->whereNotNull('live_at')
+			->first();
+
+		abort_if(!$request->user() && $stream && $stream->visibility !== 'public', 404);
+
+		return view('live.player', compact('id'));
+	}
+
 	public function deleteStream(Request $request)
 	{
 		abort_if(!config('livestreaming.enabled'), 400);
@@ -107,6 +146,8 @@ class LiveStreamController extends Controller
 			->get()
 			->each(function($stream) {
 				Storage::deleteDirectory("public/live-hls/{$stream->stream_id}");
+				LiveStreamService::clearChat($stream->profile_id);
+				StreamEnd::dispatch($stream->profile_id);
 				$stream->delete();
 			});
 
@@ -118,7 +159,7 @@ class LiveStreamController extends Controller
 		abort_if(!config('livestreaming.enabled'), 400);
 		abort_if(!$request->user(), 403);
 
-		return LiveStream::whereVisibility('local')->whereNotNull('live_at')->get()->map(function($stream) {
+		return LiveStream::whereIn('visibility', ['local', 'public'])->whereNotNull('live_at')->get()->map(function($stream) {
 			return [
 				'account' => AccountService::get($stream->profile_id),
 				'stream_id' => $stream->stream_id
@@ -162,22 +203,30 @@ class LiveStreamController extends Controller
 			'message' => 'required|max:140'
 		]);
 
-		$stream = LiveStream::whereProfileId($request->input('profile_id'))->firstOrFail();
+		$stream = LiveStream::whereProfileId($request->input('profile_id'))
+			->whereNotNull('live_at')
+			->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');
+			abort_if(!$owner && !FollowerService::follows($request->user()->profile_id, $stream->profile_id), 403);
 		}
 
+		$user = AccountService::get($request->user()->profile_id);
+
+		abort_if(!$user, 422);
+
 		$res = [
+			'id' => (string) Str::uuid(),
 			'pid' => (string) $request->user()->profile_id,
-			'username' => $request->user()->username,
+			'avatar' => $user['avatar'],
+			'username' => $user['username'],
 			'text' => $request->input('message'),
 			'ts' => now()->timestamp
 		];
 
 		LiveStreamService::addComment($stream->profile_id, json_encode($res, JSON_UNESCAPED_SLASHES));
-
+		NewChatComment::dispatch($stream, $res);
 		return $res;
 	}
 
@@ -209,24 +258,86 @@ class LiveStreamController extends Controller
 			'message' => 'required'
 		]);
 
-		abort_if($request->user()->profile_id != $request->input('profile_id'), 403);
+		$uid = $request->user()->profile_id;
+		$pid = $request->input('profile_id');
+		$msg = $request->input('message');
+		$admin = $uid == $request->input('profile_id');
+		$owner = $uid == $msg['pid'];
+		abort_if(!$admin && !$owner, 403);
 
-		$stream = LiveStream::whereProfileId($request->user()->profile_id)->firstOrFail();
+		$stream = LiveStream::whereProfileId($pid)->firstOrFail();
 
 		$payload = $request->input('message');
+		DeleteChatComment::dispatch($stream, $payload);
 		$payload = json_encode($payload, JSON_UNESCAPED_SLASHES);
 		LiveStreamService::deleteComment($stream->profile_id, $payload);
+		return;
+	}
+
+	public function banChatUser(Request $request)
+	{
+		abort_if(!config('livestreaming.enabled'), 400);
+		abort_if(!$request->user(), 403);
 
+		$this->validate($request, [
+			'profile_id' => 'required|exists:profiles,id',
+		]);
+
+		abort_if($request->user()->profile_id == $request->input('profile_id'), 403);
+
+		$stream = LiveStream::whereProfileId($request->user()->profile_id)->firstOrFail();
+		$pid = $request->input('profile_id');
+
+		BanUser::dispatch($stream, $pid);
 		return;
 	}
 
-	public function getConfig(Request $request)
+	public function pinChatComment(Request $request)
 	{
 		abort_if(!config('livestreaming.enabled'), 400);
 		abort_if(!$request->user(), 403);
 
+		$this->validate($request, [
+			'profile_id' => 'required|exists:profiles,id',
+			'message' => 'required'
+		]);
+
+		$uid = $request->user()->profile_id;
+		$pid = $request->input('profile_id');
+		$msg = $request->input('message');
+
+		abort_if($uid != $pid, 403);
+
+		$stream = LiveStream::whereProfileId($request->user()->profile_id)->firstOrFail();
+		PinChatMessage::dispatch($stream, $msg);
+		return;
+	}
+
+	public function unpinChatComment(Request $request)
+	{
+		abort_if(!config('livestreaming.enabled'), 400);
+		abort_if(!$request->user(), 403);
+
+		$this->validate($request, [
+			'profile_id' => 'required|exists:profiles,id',
+			'message' => 'required'
+		]);
+
+		$uid = $request->user()->profile_id;
+		$pid = $request->input('profile_id');
+		$msg = $request->input('message');
+
+		abort_if($uid != $pid, 403);
+
+		$stream = LiveStream::whereProfileId($request->user()->profile_id)->firstOrFail();
+		UnpinChatMessage::dispatch($stream, $msg);
+		return;
+	}
+
+	public function getConfig(Request $request)
+	{
 		$res = [
-			'enabled' => config('livestreaming.enabled'),
+			'enabled' => (bool) config('livestreaming.enabled'),
 			'broadcast' => [
 				'sources' => config('livestreaming.broadcast.sources'),
 				'limits' => config('livestreaming.broadcast.limits')
@@ -239,6 +350,7 @@ class LiveStreamController extends Controller
 	public function clientBroadcastPublish(Request $request)
 	{
 		abort_if(!config('livestreaming.enabled'), 400);
+		abort_if($request->ip() != '127.0.0.1', 400);
 		$key = $request->input('name');
 		$name = $request->input('name');
 
@@ -259,9 +371,12 @@ class LiveStreamController extends Controller
 			$stream = LiveStream::whereStreamId($key)->firstOrFail();
 		}
 
+		StreamStart::dispatch($stream->profile_id);
+
 		if($request->filled('name') && $token == false) {
 			$stream->live_at = now();
 			$stream->save();
+
 			return [];
 		} else {
 			abort(400);
@@ -273,11 +388,11 @@ class LiveStreamController extends Controller
 	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();
+		abort_if($request->ip() != '127.0.0.1', 400);
+		$name = $request->input('name');
+		$stream = LiveStream::whereStreamId($name)->firstOrFail();
+		StreamEnd::dispatch($stream->profile_id);
+		LiveStreamService::clearChat($stream->profile_id);
 
 		if(config('livestreaming.broadcast.delete_token_after_finished')) {
 			$stream->delete();

+ 75 - 0
app/Providers/TelescopeServiceProvider.php

@@ -0,0 +1,75 @@
+<?php
+
+namespace App\Providers;
+
+use Illuminate\Support\Facades\Gate;
+use Laravel\Telescope\IncomingEntry;
+use Laravel\Telescope\Telescope;
+use Laravel\Telescope\TelescopeApplicationServiceProvider;
+
+class TelescopeServiceProvider extends TelescopeApplicationServiceProvider
+{
+    /**
+     * Register any application services.
+     *
+     * @return void
+     */
+    public function register()
+    {
+        // Telescope::night();
+
+        $this->hideSensitiveRequestDetails();
+
+        Telescope::filter(function (IncomingEntry $entry) {
+            if ($this->app->environment('local')) {
+                return true;
+            }
+
+            return $entry->isReportableException() ||
+                   $entry->isFailedRequest() ||
+                   $entry->isFailedJob() ||
+                   $entry->isScheduledTask() ||
+                   $entry->hasMonitoredTag();
+        });
+    }
+
+    /**
+     * Prevent sensitive request details from being logged by Telescope.
+     *
+     * @return void
+     */
+    protected function hideSensitiveRequestDetails()
+    {
+        if ($this->app->environment('local')) {
+            return;
+        }
+
+        Telescope::hideRequestParameters(['_token']);
+
+        Telescope::hideRequestHeaders([
+            'cookie',
+            'x-csrf-token',
+            'x-xsrf-token',
+        ]);
+    }
+
+    /**
+     * Register the Telescope gate.
+     *
+     * This gate determines who can access Telescope in non-local environments.
+     *
+     * @return void
+     */
+    protected function gate()
+    {
+        Gate::define('viewTelescope', function ($user) {
+        	if(!config('telescope.enabled')) {
+        		return false;
+        	}
+            return in_array($user->email, [
+                'danielsupernault@gmail.com',
+                'me@dansup.com'
+            ]);
+        });
+    }
+}

+ 1 - 1
app/Services/LiveStreamService.php

@@ -39,7 +39,7 @@ class LiveStreamService
 		return Redis::lrem($key, 0, $val);
 	}
 
-	public static function clearChat($id, $val)
+	public static function clearChat($id)
 	{
 		$key = self::CACHE_KEY . 'chat:' . $id;
 		return Redis::del($key);

+ 1 - 1
app/Transformer/Api/StatusStatelessTransformer.php

@@ -53,7 +53,7 @@ class StatusStatelessTransformer extends Fractal\TransformerAbstract
 			'mentions'                  => StatusMentionService::get($status->id),
 			'pf_type'                   => $status->type ?? $status->setType(),
 			'reply_count'               => (int) $status->reply_count,
-			'comments_disabled'         => $status->comments_disabled ? true : false,
+			'comments_disabled'         => (bool) $status->comments_disabled,
 			'thread'                    => false,
 			'replies'                   => [],
 			'parent'                    => [],

+ 1 - 1
app/Transformer/Api/StatusTransformer.php

@@ -56,7 +56,7 @@ class StatusTransformer extends Fractal\TransformerAbstract
 			'mentions'                  => StatusMentionService::get($status->id),
 			'pf_type'                   => $status->type ?? $status->setType(),
 			'reply_count'               => (int) $status->reply_count,
-			'comments_disabled'         => $status->comments_disabled ? true : false,
+			'comments_disabled'         => (bool) $status->comments_disabled,
 			'thread'                    => false,
 			'replies'                   => [],
 			'parent'                    => [],

+ 2 - 0
composer.json

@@ -14,6 +14,7 @@
 		"ext-mbstring": "*",
 		"ext-openssl": "*",
 		"bacon/bacon-qr-code": "^2.0.3",
+		"beyondcode/laravel-websockets": "^1.13",
 		"brick/math": "^0.9.3",
 		"buzz/laravel-h-captcha": "1.0.3",
 		"doctrine/dbal": "^2.7",
@@ -45,6 +46,7 @@
 	"require-dev": {
 		"brianium/paratest": "^6.1",
 		"facade/ignition": "^2.3.6",
+		"laravel/telescope": "^4.9",
 		"mockery/mockery": "^1.0",
 		"nunomaduro/collision": "^5.0",
 		"phpunit/phpunit": "^9.0"

Файлын зөрүү хэтэрхий том тул дарагдсан байна
+ 1323 - 46
composer.lock


+ 2 - 1
config/app.php

@@ -154,10 +154,11 @@ return [
          */
         App\Providers\AppServiceProvider::class,
         App\Providers\AuthServiceProvider::class,
-        // App\Providers\BroadcastServiceProvider::class,
+        App\Providers\BroadcastServiceProvider::class,
         App\Providers\HorizonServiceProvider::class,
         App\Providers\EventServiceProvider::class,
         App\Providers\RouteServiceProvider::class,
+        App\Providers\TelescopeServiceProvider::class,
         App\Providers\PassportServiceProvider::class,
 
     ],

+ 4 - 8
config/broadcasting.php

@@ -37,14 +37,10 @@ return [
             'app_id'  => env('PUSHER_APP_ID'),
             'options' => [
                 'cluster'   => env('PUSHER_APP_CLUSTER'),
-                'encrypted' => true,
-                'host' => env('APP_DOMAIN'),
-                'port' => 6001,
-                'scheme' => 'https',
-                'curl_options' => [
-                    CURLOPT_SSL_VERIFYHOST => 0,
-                    CURLOPT_SSL_VERIFYPEER => 0,
-                ]
+                'encrypted' => env('PUSHER_APP_ENCRYPTED', false),
+                'host' => env('PUSHER_HOST', env('APP_DOMAIN')),
+                'port' => env('PUSHER_PORT', 443),
+                'scheme' => env('PUSHER_SCHEME', 'https')
             ],
         ],
 

+ 133 - 0
config/telescope.php

@@ -0,0 +1,133 @@
+<?php
+
+use Laravel\Telescope\Watchers;
+use Laravel\Telescope\Http\Middleware\Authorize;
+
+return [
+
+	'path' => 'telescope',
+
+	/*
+	|--------------------------------------------------------------------------
+	| Telescope Storage Driver
+	|--------------------------------------------------------------------------
+	|
+	| This configuration options determines the storage driver that will
+	| be used to store Telescope's data. In addition, you may set any
+	| custom options as needed by the particular driver you choose.
+	|
+	*/
+
+	'driver' => env('TELESCOPE_DRIVER', 'database'),
+
+	'storage' => [
+		'database' => [
+			'connection' => env('DB_CONNECTION', 'mysql'),
+		],
+	],
+
+	/*
+	|--------------------------------------------------------------------------
+	| Telescope Master Switch
+	|--------------------------------------------------------------------------
+	|
+	| This option may be used to disable all Telescope watchers regardless
+	| of their individual configuration, which simply provides a single
+	| and convenient way to enable or disable Telescope data storage.
+	|
+	*/
+
+	'enabled' => env('TELESCOPE_ENABLED', false),
+
+	/*
+	|--------------------------------------------------------------------------
+	| Telescope Route Middleware
+	|--------------------------------------------------------------------------
+	|
+	| These middleware will be assigned to every Telescope route, giving you
+	| the chance to add your own middleware to this list or change any of
+	| the existing middleware. Or, you can simply stick with this list.
+	|
+	*/
+
+	'middleware' => [
+		'web',
+		Authorize::class,
+	],
+
+	/*
+	|--------------------------------------------------------------------------
+	| Ignored Paths & Commands
+	|--------------------------------------------------------------------------
+	|
+	| The following array lists the URI paths and Artisan commands that will
+	| not be watched by Telescope. In addition to this list, some Laravel
+	| commands, like migrations and queue commands, are always ignored.
+	|
+	*/
+
+	'ignore_paths' => [
+		'js*',
+		'i*'
+	],
+
+	'ignore_commands' => [
+		//
+	],
+
+	/*
+	|--------------------------------------------------------------------------
+	| Telescope Watchers
+	|--------------------------------------------------------------------------
+	|
+	| The following array lists the "watchers" that will be registered with
+	| Telescope. The watchers gather the application's profile data when
+	| a request or task is executed. Feel free to customize this list.
+	|
+	*/
+
+	'watchers' => [
+		Watchers\CacheWatcher::class => env('TELESCOPE_CACHE_WATCHER', true),
+
+		Watchers\CommandWatcher::class => [
+			'enabled' => env('TELESCOPE_COMMAND_WATCHER', true),
+			'ignore' => [],
+		],
+
+		Watchers\DumpWatcher::class => env('TELESCOPE_DUMP_WATCHER', true),
+		Watchers\EventWatcher::class => env('TELESCOPE_EVENT_WATCHER', true),
+		Watchers\ExceptionWatcher::class => env('TELESCOPE_EXCEPTION_WATCHER', true),
+		Watchers\JobWatcher::class => env('TELESCOPE_JOB_WATCHER', true),
+		Watchers\LogWatcher::class => env('TELESCOPE_LOG_WATCHER', true),
+		Watchers\MailWatcher::class => env('TELESCOPE_MAIL_WATCHER', true),
+		Watchers\ClientRequestWatcher::class =>true,
+
+		Watchers\ModelWatcher::class => [
+			'enabled' => env('TELESCOPE_MODEL_WATCHER', true),
+			'events' => ['eloquent.*'],
+		],
+
+		Watchers\NotificationWatcher::class => env('TELESCOPE_NOTIFICATION_WATCHER', true),
+
+		Watchers\QueryWatcher::class => [
+			'enabled' => env('TELESCOPE_QUERY_WATCHER', true),
+			'ignore_packages' => true,
+			'slow' => 100,
+		],
+
+		Watchers\RedisWatcher::class => env('TELESCOPE_REDIS_WATCHER', true),
+
+		Watchers\RequestWatcher::class => [
+			'enabled' => env('TELESCOPE_REQUEST_WATCHER', true),
+			'size_limit' => env('TELESCOPE_RESPONSE_SIZE_LIMIT', 64),
+		],
+
+		Watchers\GateWatcher::class => [
+			'enabled' => env('TELESCOPE_GATE_WATCHER', true),
+			'ignore_abilities' => [],
+			'ignore_packages' => true,
+		],
+
+		Watchers\ScheduleWatcher::class => env('TELESCOPE_SCHEDULE_WATCHER', true),
+	],
+];

+ 258 - 84
config/websockets.php

@@ -1,125 +1,299 @@
 <?php
 
-use BeyondCode\LaravelWebSockets\Dashboard\Http\Middleware\Authorize;
-
 return [
 
     /*
-     * This package comes with multi tenancy out of the box. Here you can
-     * configure the different apps that can use the webSockets server.
-     *
-     * Optionally you can disable client events so clients cannot send
-     * messages to each other via the webSockets.
-     */
+    |--------------------------------------------------------------------------
+    | Dashboard Settings
+    |--------------------------------------------------------------------------
+    |
+    | You can configure the dashboard settings from here.
+    |
+    */
+
+    'dashboard' => [
+
+        'port' => env('LARAVEL_WEBSOCKETS_PORT', 6001),
+
+        'domain' => env('LARAVEL_WEBSOCKETS_DOMAIN'),
+
+        'path' => env('LARAVEL_WEBSOCKETS_PATH', 'laravel-websockets'),
+
+        'middleware' => [
+            'web',
+            \BeyondCode\LaravelWebSockets\Dashboard\Http\Middleware\Authorize::class,
+        ],
+
+    ],
+
+    'managers' => [
+
+        /*
+        |--------------------------------------------------------------------------
+        | Application Manager
+        |--------------------------------------------------------------------------
+        |
+        | An Application manager determines how your websocket server allows
+        | the use of the TCP protocol based on, for example, a list of allowed
+        | applications.
+        | By default, it uses the defined array in the config file, but you can
+        | anytime implement the same interface as the class and add your own
+        | custom method to retrieve the apps.
+        |
+        */
+
+        'app' => \BeyondCode\LaravelWebSockets\Apps\ConfigAppManager::class,
+
+    ],
+
+    /*
+    |--------------------------------------------------------------------------
+    | Applications Repository
+    |--------------------------------------------------------------------------
+    |
+    | By default, the only allowed app is the one you define with
+    | your PUSHER_* variables from .env.
+    | You can configure to use multiple apps if you need to, or use
+    | a custom App Manager that will handle the apps from a database, per se.
+    |
+    | You can apply multiple settings, like the maximum capacity, enable
+    | client-to-client messages or statistics.
+    |
+    */
+
     'apps' => [
         [
             'id' => env('PUSHER_APP_ID'),
             'name' => env('APP_NAME'),
+            'host' => env('PUSHER_APP_HOST'),
             'key' => env('PUSHER_APP_KEY'),
             'secret' => env('PUSHER_APP_SECRET'),
-            'enable_client_messages' => env('WSS_CM', false),
-            'enable_statistics' => env('WSS_STATS', false),
+            'path' => env('PUSHER_APP_PATH'),
+            'capacity' => null,
+            'enable_client_messages' => false,
+            'enable_statistics' => false,
+            'allowed_origins' => [
+                // env('LARAVEL_WEBSOCKETS_DOMAIN'),
+            ],
         ],
     ],
 
     /*
-     * This class is responsible for finding the apps. The default provider
-     * will use the apps defined in this config file.
-     *
-     * You can create a custom provider by implementing the
-     * `AppProvider` interface.
-     */
-    'app_provider' => BeyondCode\LaravelWebSockets\Apps\ConfigAppProvider::class,
+    |--------------------------------------------------------------------------
+    | Broadcasting Replication PubSub
+    |--------------------------------------------------------------------------
+    |
+    | You can enable replication to publish and subscribe to
+    | messages across the driver.
+    |
+    | By default, it is set to 'local', but you can configure it to use drivers
+    | like Redis to ensure connection between multiple instances of
+    | WebSocket servers. Just set the driver to 'redis' to enable the PubSub using Redis.
+    |
+    */
 
-    /*
-     * This array contains the hosts of which you want to allow incoming requests.
-     * Leave this empty if you want to accept requests from all hosts.
-     */
-    'allowed_origins' => [
-        //
-    ],
+    'replication' => [
 
-    /*
-     * The maximum request size in kilobytes that is allowed for an incoming WebSocket request.
-     */
-    'max_request_size_in_kb' => 250,
+        'mode' => env('WEBSOCKETS_REPLICATION_MODE', 'local'),
 
-    /*
-     * This path will be used to register the necessary routes for the package.
-     */
-    'path' => 'pxws',
+        'modes' => [
+
+            /*
+            |--------------------------------------------------------------------------
+            | Local Replication
+            |--------------------------------------------------------------------------
+            |
+            | Local replication is actually a null replicator, meaning that it
+            | is the default behaviour of storing the connections into an array.
+            |
+            */
+
+            'local' => [
+
+                /*
+                |--------------------------------------------------------------------------
+                | Channel Manager
+                |--------------------------------------------------------------------------
+                |
+                | The channel manager is responsible for storing, tracking and retrieving
+                | the channels as long as their members and connections.
+                |
+                */
+
+                'channel_manager' => \BeyondCode\LaravelWebSockets\ChannelManagers\LocalChannelManager::class,
+
+                /*
+                |--------------------------------------------------------------------------
+                | Statistics Collector
+                |--------------------------------------------------------------------------
+                |
+                | The Statistics Collector will, by default, handle the incoming statistics,
+                | storing them until they will become dumped into another database, usually
+                | a MySQL database or a time-series database.
+                |
+                */
+
+                'collector' => \BeyondCode\LaravelWebSockets\Statistics\Collectors\MemoryCollector::class,
+
+            ],
+
+            'redis' => [
+
+                'connection' => env('WEBSOCKETS_REDIS_REPLICATION_CONNECTION', 'default'),
+
+                /*
+                |--------------------------------------------------------------------------
+                | Channel Manager
+                |--------------------------------------------------------------------------
+                |
+                | The channel manager is responsible for storing, tracking and retrieving
+                | the channels as long as their members and connections.
+                |
+                */
+
+                'channel_manager' => \BeyondCode\LaravelWebSockets\ChannelManagers\RedisChannelManager::class,
+
+                /*
+                |--------------------------------------------------------------------------
+                | Statistics Collector
+                |--------------------------------------------------------------------------
+                |
+                | The Statistics Collector will, by default, handle the incoming statistics,
+                | storing them until they will become dumped into another database, usually
+                | a MySQL database or a time-series database.
+                |
+                */
+
+                'collector' => \BeyondCode\LaravelWebSockets\Statistics\Collectors\RedisCollector::class,
+
+            ],
+
+        ],
 
-    /*
-     * Dashboard Routes Middleware
-     *
-     * These middleware will be assigned to every dashboard route, giving you
-     * the chance to add your own middleware to this list or change any of
-     * the existing middleware. Or, you can simply stick with this list.
-     */
-    'middleware' => [
-        'web',
-        Authorize::class,
     ],
 
     'statistics' => [
+
         /*
-         * This model will be used to store the statistics of the WebSocketsServer.
-         * The only requirement is that the model should extend
-         * `WebSocketsStatisticsEntry` provided by this package.
-         */
-        'model' => \BeyondCode\LaravelWebSockets\Statistics\Models\WebSocketsStatisticsEntry::class,
+        |--------------------------------------------------------------------------
+        | Statistics Store
+        |--------------------------------------------------------------------------
+        |
+        | The Statistics Store is the place where all the temporary stats will
+        | be dumped. This is a much reliable store and will be used to display
+        | graphs or handle it later on your app.
+        |
+        */
+
+        'store' => \BeyondCode\LaravelWebSockets\Statistics\Stores\DatabaseStore::class,
 
         /*
-         * Here you can specify the interval in seconds at which statistics should be logged.
-         */
+        |--------------------------------------------------------------------------
+        | Statistics Interval Period
+        |--------------------------------------------------------------------------
+        |
+        | Here you can specify the interval in seconds at which
+        | statistics should be logged.
+        |
+        */
+
         'interval_in_seconds' => 60,
 
         /*
-         * When the clean-command is executed, all recorded statistics older than
-         * the number of days specified here will be deleted.
-         */
+        |--------------------------------------------------------------------------
+        | Statistics Deletion Period
+        |--------------------------------------------------------------------------
+        |
+        | When the clean-command is executed, all recorded statistics older than
+        | the number of days specified here will be deleted.
+        |
+        */
+
         'delete_statistics_older_than_days' => 60,
 
-        /*
-         * Use an DNS resolver to make the requests to the statistics logger
-         * default is to resolve everything to 127.0.0.1.
-         */
-        'perform_dns_lookup' => false,
     ],
 
     /*
-     * Define the optional SSL context for your WebSocket connections.
-     * You can see all available options at: http://php.net/manual/en/context.ssl.php
-     */
+    |--------------------------------------------------------------------------
+    | Maximum Request Size
+    |--------------------------------------------------------------------------
+    |
+    | The maximum request size in kilobytes that is allowed for
+    | an incoming WebSocket request.
+    |
+    */
+
+    'max_request_size_in_kb' => 250,
+
+    /*
+    |--------------------------------------------------------------------------
+    | SSL Configuration
+    |--------------------------------------------------------------------------
+    |
+    | By default, the configuration allows only on HTTP. For SSL, you need
+    | to set up the the certificate, the key, and optionally, the passphrase
+    | for the private key.
+    | You will need to restart the server for the settings to take place.
+    |
+    */
+
     'ssl' => [
-        /*
-         * Path to local certificate file on filesystem. It must be a PEM encoded file which
-         * contains your certificate and private key. It can optionally contain the
-         * certificate chain of issuers. The private key also may be contained
-         * in a separate file specified by local_pk.
-         */
-        'local_cert' => env('WSS_LOCAL_CERT', null),
 
-        /*
-         * Path to local private key file on filesystem in case of separate files for
-         * certificate (local_cert) and private key.
-         */
-        'local_pk' => env('WSS_LOCAL_PK', null),
+        'local_cert' => env('LARAVEL_WEBSOCKETS_SSL_LOCAL_CERT', null),
 
-        /*
-         * Passphrase for your local_cert file.
-         */
-        'passphrase' => env('WSS_PASSPHRASE', null),
+        'capath' => env('LARAVEL_WEBSOCKETS_SSL_CA', null),
+
+        'local_pk' => env('LARAVEL_WEBSOCKETS_SSL_LOCAL_PK', null),
+
+        'passphrase' => env('LARAVEL_WEBSOCKETS_SSL_PASSPHRASE', null),
+
+        'verify_peer' => env('APP_ENV') === 'production',
+
+        'allow_self_signed' => env('APP_ENV') !== 'production',
 
-        'verify_peer' => env('WSS_VERIFY_PEER', false),
     ],
 
     /*
-     * Channel Manager
-     * This class handles how channel persistence is handled.
-     * By default, persistence is stored in an array by the running webserver.
-     * The only requirement is that the class should implement
-     * `ChannelManager` interface provided by this package.
-     */
-    'channel_manager' => \BeyondCode\LaravelWebSockets\WebSockets\Channels\ChannelManagers\ArrayChannelManager::class,
+    |--------------------------------------------------------------------------
+    | Route Handlers
+    |--------------------------------------------------------------------------
+    |
+    | Here you can specify the route handlers that will take over
+    | the incoming/outgoing websocket connections. You can extend the
+    | original class and implement your own logic, alongside
+    | with the existing logic.
+    |
+    */
+
+    'handlers' => [
+
+        'websocket' => \BeyondCode\LaravelWebSockets\Server\WebSocketHandler::class,
+
+        'health' => \BeyondCode\LaravelWebSockets\Server\HealthHandler::class,
+
+        'trigger_event' => \BeyondCode\LaravelWebSockets\API\TriggerEvent::class,
+
+        'fetch_channels' => \BeyondCode\LaravelWebSockets\API\FetchChannels::class,
+
+        'fetch_channel' => \BeyondCode\LaravelWebSockets\API\FetchChannel::class,
+
+        'fetch_users' => \BeyondCode\LaravelWebSockets\API\FetchUsers::class,
+
+    ],
+
+    /*
+    |--------------------------------------------------------------------------
+    | Promise Resolver
+    |--------------------------------------------------------------------------
+    |
+    | The promise resolver is a class that takes a input value and is
+    | able to make sure the PHP code runs async by using ->then(). You can
+    | use your own Promise Resolver. This is usually changed when you want to
+    | intercept values by the promises throughout the app, like in testing
+    | to switch from async to sync.
+    |
+    */
+
+    'promise_resolver' => \React\Promise\FulfilledPromise::class,
+
 ];

BIN
public/js/collectioncompose.js


BIN
public/js/collections.js


BIN
public/js/home-ojtjadoml.js


BIN
public/js/installer.js


+ 1 - 0
public/js/installer.js.LICENSE.txt

@@ -0,0 +1 @@
+/*! @source http://purl.eligrey.com/github/canvas-toBlob.js/blob/master/canvas-toBlob.js */

BIN
public/js/live-player.js


BIN
public/js/spa.js


BIN
public/js/vendor.js


+ 686 - 0
public/js/vendor.js.LICENSE.txt

@@ -48,6 +48,14 @@
  * Released under the MIT license
  */
 
+/*!
+ * Pusher JavaScript Library v7.1.1-beta
+ * https://pusher.com/
+ *
+ * Copyright 2020, Pusher
+ * Released under the MIT licence.
+ */
+
 /*!
  * Scroll Lock v3.1.3
  * https://github.com/MohammadYounes/jquery-scrollLock
@@ -155,8 +163,686 @@ See the Apache Version 2.0 License for specific language governing permissions
 and limitations under the License.
 ***************************************************************************** */
 
+/*! ../controller/level-helper */
+
+/*! ../crypt/decrypter */
+
+/*! ../demux/aacdemuxer */
+
+/*! ../demux/chunk-cache */
+
+/*! ../demux/id3 */
+
+/*! ../demux/mp3demuxer */
+
+/*! ../demux/mp4demuxer */
+
+/*! ../demux/transmuxer */
+
+/*! ../demux/transmuxer-interface */
+
+/*! ../demux/transmuxer-worker.ts */
+
+/*! ../demux/tsdemuxer */
+
+/*! ../errors */
+
+/*! ../events */
+
+/*! ../is-supported */
+
+/*! ../loader/fragment */
+
+/*! ../loader/fragment-loader */
+
+/*! ../loader/load-stats */
+
+/*! ../remux/mp4-remuxer */
+
+/*! ../remux/passthrough-remuxer */
+
+/*! ../task-loop */
+
+/*! ../types/cmcd */
+
+/*! ../types/level */
+
+/*! ../types/loader */
+
+/*! ../types/transmuxer */
+
+/*! ../utils/attr-list */
+
+/*! ../utils/binary-search */
+
+/*! ../utils/buffer-helper */
+
+/*! ../utils/cea-608-parser */
+
+/*! ../utils/codecs */
+
+/*! ../utils/discontinuities */
+
+/*! ../utils/ewma */
+
+/*! ../utils/ewma-bandwidth-estimator */
+
+/*! ../utils/imsc1-ttml-parser */
+
+/*! ../utils/logger */
+
+/*! ../utils/mediakeys-helper */
+
+/*! ../utils/mediasource-helper */
+
+/*! ../utils/mp4-tools */
+
+/*! ../utils/output-filter */
+
+/*! ../utils/texttrack-utils */
+
+/*! ../utils/time-ranges */
+
+/*! ../utils/timescale-conversion */
+
+/*! ../utils/typed-array */
+
+/*! ../utils/webvtt-parser */
+
+/*! ./aac-helper */
+
+/*! ./adts */
+
+/*! ./aes-crypto */
+
+/*! ./aes-decryptor */
+
+/*! ./base-audio-demuxer */
+
+/*! ./base-playlist-controller */
+
+/*! ./base-stream-controller */
+
+/*! ./buffer-operation-queue */
+
+/*! ./chunk-cache */
+
+/*! ./config */
+
+/*! ./controller/abr-controller */
+
+/*! ./controller/audio-stream-controller */
+
+/*! ./controller/audio-track-controller */
+
+/*! ./controller/buffer-controller */
+
+/*! ./controller/cap-level-controller */
+
+/*! ./controller/cmcd-controller */
+
+/*! ./controller/eme-controller */
+
+/*! ./controller/fps-controller */
+
+/*! ./controller/fragment-tracker */
+
+/*! ./controller/id3-track-controller */
+
+/*! ./controller/latency-controller */
+
+/*! ./controller/level-controller */
+
+/*! ./controller/stream-controller */
+
+/*! ./controller/subtitle-stream-controller */
+
+/*! ./controller/subtitle-track-controller */
+
+/*! ./controller/timeline-controller */
+
+/*! ./dummy-demuxed-track */
+
+/*! ./errors */
+
+/*! ./events */
+
+/*! ./exp-golomb */
+
+/*! ./fast-aes-key */
+
+/*! ./fragment */
+
+/*! ./fragment-finders */
+
+/*! ./fragment-tracker */
+
+/*! ./gap-controller */
+
+/*! ./id3 */
+
+/*! ./is-supported */
+
+/*! ./level-details */
+
+/*! ./level-helper */
+
+/*! ./level-key */
+
+/*! ./load-stats */
+
+/*! ./loader/key-loader */
+
+/*! ./loader/playlist-loader */
+
+/*! ./logger */
+
+/*! ./m3u8-parser */
+
+/*! ./mp4-generator */
+
+/*! ./mp4-tools */
+
+/*! ./mpegaudio */
+
+/*! ./sample-aes */
+
+/*! ./src/polyfills/number */
+
+/*! ./texttrack-utils */
+
+/*! ./timescale-conversion */
+
+/*! ./tsdemuxer */
+
+/*! ./typed-array */
+
+/*! ./utils/cues */
+
+/*! ./utils/fetch-loader */
+
+/*! ./utils/logger */
+
+/*! ./utils/mediakeys-helper */
+
+/*! ./utils/mediasource-helper */
+
+/*! ./utils/xhr-loader */
+
+/*! ./vttcue */
+
+/*! ./vttparser */
+
+/*! ./webvtt-parser */
+
+/*! eventemitter3 */
+
+/*! exports provided: AttrList */
+
+/*! exports provided: BufferHelper */
+
+/*! exports provided: CMCDVersion, CMCDObjectType, CMCDStreamingFormat, CMCDStreamType */
+
+/*! exports provided: ChunkMetadata */
+
+/*! exports provided: ElementaryStreamTypes, BaseSegment, Fragment, Part */
+
+/*! exports provided: ErrorTypes, ErrorDetails */
+
+/*! exports provided: Events */
+
+/*! exports provided: FragmentState, FragmentTracker */
+
+/*! exports provided: HlsSkip, getSkipValue, HlsUrlParameters, Level */
+
+/*! exports provided: IMSC1_CODEC, parseIMSC1 */
+
+/*! exports provided: KeySystems, requestMediaKeySystemAccess */
+
+/*! exports provided: LevelDetails */
+
+/*! exports provided: LevelKey */
+
+/*! exports provided: LoadStats */
+
+/*! exports provided: PlaylistContextType, PlaylistLevelType */
+
+/*! exports provided: Row, CaptionScreen, default */
+
+/*! exports provided: STALL_MINIMUM_DURATION_MS, MAX_START_GAP_JUMP, SKIP_BUFFER_HOLE_STEP_SECONDS, SKIP_BUFFER_RANGE_START, default */
+
+/*! exports provided: State, default */
+
+/*! exports provided: SubtitleStreamController */
+
+/*! exports provided: TimelineController */
+
+/*! exports provided: addGroupId, assignTrackIdsByGroup, updatePTS, updateFragPTSDTS, mergeDetails, mapPartIntersection, mapFragmentIntersection, adjustSliding, addSliding, computeReloadInterval, getFragmentWithSN, getPartWith */
+
+/*! exports provided: appendFrame, parseHeader, isHeaderPattern, isHeader, canParse, probe */
+
+/*! exports provided: bin2str, readUint16, readUint32, writeUint32, findBox, parseSegmentIndex, parseInitSegment, getStartDTS, getDuration, computeRawDurationFromSamples, offsetStartDTS, segmentValidRange, appendUint8Array */
+
+/*! exports provided: default */
+
+/*! exports provided: default, LoadError */
+
+/*! exports provided: default, isPromise, TransmuxConfig, TransmuxState */
+
+/*! exports provided: default, normalizePts */
+
+/*! exports provided: discardEPB, default */
+
+/*! exports provided: dummyTrack */
+
+/*! exports provided: enableLogs, logger */
+
+/*! exports provided: fetchSupported, default */
+
+/*! exports provided: findFirstFragWithCC, shouldAlignOnDiscontinuities, findDiscontinuousReferenceFrag, adjustSlidingStart, alignStream, alignPDT, alignFragmentByPDTDelta, alignMediaPlaylistByPDT */
+
+/*! exports provided: findFragmentByPDT, findFragmentByPTS, fragmentWithinToleranceTest, pdtWithinToleranceTest, findFragWithCC */
+
+/*! exports provided: generateCueId, parseWebVTT */
+
+/*! exports provided: getAudioConfig, isHeaderPattern, getHeaderLength, getFullFrameLength, canGetFrameLength, isHeader, canParse, probe, initTrackConfig, getFrameDuration, parseFrameHeader, appendFrame */
+
+/*! exports provided: getMediaSource */
+
+/*! exports provided: hlsDefaultConfig, mergeConfig, enableStreamingMode */
+
+/*! exports provided: initPTSFn, default */
+
+/*! exports provided: isCodecType, isCodecSupportedInMp4 */
+
+/*! exports provided: isFiniteNumber, MAX_SAFE_INTEGER */
+
+/*! exports provided: isHeader, isFooter, getID3Data, canParse, getTimeStamp, isTimeStampFrame, getID3Frames, decodeFrame, utf8ArrayToStr, testables */
+
+/*! exports provided: isSupported, changeTypeSupported */
+
+/*! exports provided: parseTimeStamp, fixLineBreaks, VTTParser */
+
+/*! exports provided: removePadding, default */
+
+/*! exports provided: sendAddTrackEvent, addCueToTrack, clearCurrentCues, removeCuesInRange, getCuesInRange */
+
+/*! exports provided: sliceUint8 */
+
+/*! exports provided: toTimescaleFromBase, toTimescaleFromScale, toMsFromMpegTsClock, toMpegTsClockFromTimescale */
+
 /*! https://mths.be/punycode v1.4.1 by @mathias */
 
+/*! no static exports found */
+
+/*! url-toolkit */
+
+/*! webworkify-webpack */
+
+/*!********************!*\
+  !*** ./src/hls.ts ***!
+  \********************/
+
+/*!***********************!*\
+  !*** ./src/config.ts ***!
+  \***********************/
+
+/*!***********************!*\
+  !*** ./src/errors.ts ***!
+  \***********************/
+
+/*!***********************!*\
+  !*** ./src/events.ts ***!
+  \***********************/
+
+/*!**************************!*\
+  !*** ./src/demux/id3.ts ***!
+  \**************************/
+
+/*!**************************!*\
+  !*** ./src/task-loop.ts ***!
+  \**************************/
+
+/*!***************************!*\
+  !*** ./src/demux/adts.ts ***!
+  \***************************/
+
+/*!***************************!*\
+  !*** ./src/types/cmcd.ts ***!
+  \***************************/
+
+/*!***************************!*\
+  !*** ./src/utils/cues.ts ***!
+  \***************************/
+
+/*!***************************!*\
+  !*** ./src/utils/ewma.ts ***!
+  \***************************/
+
+/*!****************************!*\
+  !*** ./src/types/level.ts ***!
+  \****************************/
+
+/*!*****************************!*\
+  !*** ./src/is-supported.ts ***!
+  \*****************************/
+
+/*!*****************************!*\
+  !*** ./src/types/loader.ts ***!
+  \*****************************/
+
+/*!*****************************!*\
+  !*** ./src/utils/codecs.ts ***!
+  \*****************************/
+
+/*!*****************************!*\
+  !*** ./src/utils/logger.ts ***!
+  \*****************************/
+
+/*!*****************************!*\
+  !*** ./src/utils/vttcue.ts ***!
+  \*****************************/
+
+/*!********************************!*\
+  !*** ./src/crypt/decrypter.ts ***!
+  \********************************/
+
+/*!********************************!*\
+  !*** ./src/demux/mpegaudio.ts ***!
+  \********************************/
+
+/*!********************************!*\
+  !*** ./src/demux/tsdemuxer.ts ***!
+  \********************************/
+
+/*!********************************!*\
+  !*** ./src/loader/fragment.ts ***!
+  \********************************/
+
+/*!********************************!*\
+  !*** ./src/utils/attr-list.ts ***!
+  \********************************/
+
+/*!********************************!*\
+  !*** ./src/utils/mp4-tools.ts ***!
+  \********************************/
+
+/*!********************************!*\
+  !*** ./src/utils/vttparser.ts ***!
+  \********************************/
+
+/*!*********************************!*\
+  !*** ./src/crypt/aes-crypto.ts ***!
+  \*********************************/
+
+/*!*********************************!*\
+  !*** ./src/demux/aacdemuxer.ts ***!
+  \*********************************/
+
+/*!*********************************!*\
+  !*** ./src/demux/exp-golomb.ts ***!
+  \*********************************/
+
+/*!*********************************!*\
+  !*** ./src/demux/mp3demuxer.ts ***!
+  \*********************************/
+
+/*!*********************************!*\
+  !*** ./src/demux/mp4demuxer.ts ***!
+  \*********************************/
+
+/*!*********************************!*\
+  !*** ./src/demux/sample-aes.ts ***!
+  \*********************************/
+
+/*!*********************************!*\
+  !*** ./src/demux/transmuxer.ts ***!
+  \*********************************/
+
+/*!*********************************!*\
+  !*** ./src/loader/level-key.ts ***!
+  \*********************************/
+
+/*!*********************************!*\
+  !*** ./src/polyfills/number.ts ***!
+  \*********************************/
+
+/*!*********************************!*\
+  !*** ./src/remux/aac-helper.ts ***!
+  \*********************************/
+
+/*!*********************************!*\
+  !*** ./src/types/transmuxer.ts ***!
+  \*********************************/
+
+/*!*********************************!*\
+  !*** ./src/utils/xhr-loader.ts ***!
+  \*********************************/
+
+/*!**********************************!*\
+  !*** ./src/demux/chunk-cache.ts ***!
+  \**********************************/
+
+/*!**********************************!*\
+  !*** ./src/loader/key-loader.ts ***!
+  \**********************************/
+
+/*!**********************************!*\
+  !*** ./src/loader/load-stats.ts ***!
+  \**********************************/
+
+/*!**********************************!*\
+  !*** ./src/remux/mp4-remuxer.ts ***!
+  \**********************************/
+
+/*!**********************************!*\
+  !*** ./src/utils/time-ranges.ts ***!
+  \**********************************/
+
+/*!**********************************!*\
+  !*** ./src/utils/typed-array.ts ***!
+  \**********************************/
+
+/*!***********************************!*\
+  !*** ./src/crypt/fast-aes-key.ts ***!
+  \***********************************/
+
+/*!***********************************!*\
+  !*** ./src/loader/m3u8-parser.ts ***!
+  \***********************************/
+
+/*!***********************************!*\
+  !*** ./src/utils/fetch-loader.ts ***!
+  \***********************************/
+
+/*!************************************!*\
+  !*** ./src/crypt/aes-decryptor.ts ***!
+  \************************************/
+
+/*!************************************!*\
+  !*** ./src/remux/mp4-generator.ts ***!
+  \************************************/
+
+/*!************************************!*\
+  !*** ./src/utils/binary-search.ts ***!
+  \************************************/
+
+/*!************************************!*\
+  !*** ./src/utils/buffer-helper.ts ***!
+  \************************************/
+
+/*!************************************!*\
+  !*** ./src/utils/output-filter.ts ***!
+  \************************************/
+
+/*!************************************!*\
+  !*** ./src/utils/webvtt-parser.ts ***!
+  \************************************/
+
+/*!*************************************!*\
+  !*** ./src/loader/level-details.ts ***!
+  \*************************************/
+
+/*!*************************************!*\
+  !*** ./src/utils/cea-608-parser.ts ***!
+  \*************************************/
+
+/*!**************************************!*\
+  !*** ./src/utils/discontinuities.ts ***!
+  \**************************************/
+
+/*!**************************************!*\
+  !*** ./src/utils/texttrack-utils.ts ***!
+  \**************************************/
+
+/*!***************************************!*\
+  !*** ./src/loader/fragment-loader.ts ***!
+  \***************************************/
+
+/*!***************************************!*\
+  !*** ./src/loader/playlist-loader.ts ***!
+  \***************************************/
+
+/*!***************************************!*\
+  !*** ./src/utils/mediakeys-helper.ts ***!
+  \***************************************/
+
+/*!****************************************!*\
+  !*** ./src/controller/level-helper.ts ***!
+  \****************************************/
+
+/*!****************************************!*\
+  !*** ./src/demux/transmuxer-worker.ts ***!
+  \****************************************/
+
+/*!****************************************!*\
+  !*** ./src/utils/imsc1-ttml-parser.ts ***!
+  \****************************************/
+
+/*!*****************************************!*\
+  !*** ./src/demux/base-audio-demuxer.ts ***!
+  \*****************************************/
+
+/*!*****************************************!*\
+  !*** ./src/utils/mediasource-helper.ts ***!
+  \*****************************************/
+
+/*!******************************************!*\
+  !*** ./src/controller/abr-controller.ts ***!
+  \******************************************/
+
+/*!******************************************!*\
+  !*** ./src/controller/eme-controller.ts ***!
+  \******************************************/
+
+/*!******************************************!*\
+  !*** ./src/controller/fps-controller.ts ***!
+  \******************************************/
+
+/*!******************************************!*\
+  !*** ./src/controller/gap-controller.ts ***!
+  \******************************************/
+
+/*!******************************************!*\
+  !*** ./src/demux/dummy-demuxed-track.ts ***!
+  \******************************************/
+
+/*!******************************************!*\
+  !*** ./src/remux/passthrough-remuxer.ts ***!
+  \******************************************/
+
+/*!*******************************************!*\
+  !*** ./src/controller/cmcd-controller.ts ***!
+  \*******************************************/
+
+/*!*******************************************!*\
+  !*** ./src/demux/transmuxer-interface.ts ***!
+  \*******************************************/
+
+/*!*******************************************!*\
+  !*** ./src/utils/timescale-conversion.ts ***!
+  \*******************************************/
+
+/*!********************************************!*\
+  !*** ./src/controller/fragment-finders.ts ***!
+  \********************************************/
+
+/*!********************************************!*\
+  !*** ./src/controller/fragment-tracker.ts ***!
+  \********************************************/
+
+/*!********************************************!*\
+  !*** ./src/controller/level-controller.ts ***!
+  \********************************************/
+
+/*!*********************************************!*\
+  !*** ./node_modules/eventemitter3/index.js ***!
+  \*********************************************/
+
+/*!*********************************************!*\
+  !*** ./src/controller/buffer-controller.ts ***!
+  \*********************************************/
+
+/*!*********************************************!*\
+  !*** ./src/controller/stream-controller.ts ***!
+  \*********************************************/
+
+/*!**********************************************!*\
+  !*** ./src/controller/latency-controller.ts ***!
+  \**********************************************/
+
+/*!***********************************************!*\
+  !*** ./src/controller/timeline-controller.ts ***!
+  \***********************************************/
+
+/*!***********************************************!*\
+  !*** ./src/utils/ewma-bandwidth-estimator.ts ***!
+  \***********************************************/
+
+/*!************************************************!*\
+  !*** ./src/controller/cap-level-controller.ts ***!
+  \************************************************/
+
+/*!************************************************!*\
+  !*** ./src/controller/id3-track-controller.ts ***!
+  \************************************************/
+
+/*!**************************************************!*\
+  !*** ./node_modules/webworkify-webpack/index.js ***!
+  \**************************************************/
+
+/*!**************************************************!*\
+  !*** ./src/controller/audio-track-controller.ts ***!
+  \**************************************************/
+
+/*!**************************************************!*\
+  !*** ./src/controller/base-stream-controller.ts ***!
+  \**************************************************/
+
+/*!**************************************************!*\
+  !*** ./src/controller/buffer-operation-queue.ts ***!
+  \**************************************************/
+
+/*!***************************************************!*\
+  !*** ./src/controller/audio-stream-controller.ts ***!
+  \***************************************************/
+
+/*!****************************************************!*\
+  !*** ./src/controller/base-playlist-controller.ts ***!
+  \****************************************************/
+
+/*!*****************************************************!*\
+  !*** ./node_modules/url-toolkit/src/url-toolkit.js ***!
+  \*****************************************************/
+
+/*!*****************************************************!*\
+  !*** ./src/controller/subtitle-track-controller.ts ***!
+  \*****************************************************/
+
+/*!******************************************************!*\
+  !*** ./src/controller/subtitle-stream-controller.ts ***!
+  \******************************************************/
+
 /**
   * vue-class-component v7.2.3
   * (c) 2015-present Evan You

BIN
public/mix-manifest.json


+ 4 - 0
resources/assets/js/live-player.js

@@ -0,0 +1,4 @@
+Vue.component(
+    'live-player',
+    require('./../components/LivePlayer.vue').default
+);

+ 62 - 8
resources/views/admin/diagnostics/home.blade.php

@@ -58,15 +58,15 @@
 		</li>
 		<li>
 			<strong><span class="badge badge-primary">ACTIVITYPUB</span> instance actor created: </strong>
-			<span>{{ \App\Models\InstanceActor::count() ? '✅' : '❌' }}</span>
+			<span>{{ \App\Models\InstanceActor::count() ? '✅ true' : '❌ false' }}</span>
 		</li>
 		<li>
 			<strong><span class="badge badge-primary">ACTIVITYPUB</span> instance actor cached: </strong>
-			<span>{{ Cache::get(\App\Models\InstanceActor::PROFILE_KEY) ? '✅' : '❌' }}</span>
+			<span>{{ Cache::get(\App\Models\InstanceActor::PROFILE_KEY) ? '✅ true' : '❌ false' }}</span>
 		</li>
 		<li>
 			<strong><span class="badge badge-primary">OAUTH</span> enabled: </strong>
-			<span>{{ config_cache('pixelfed.oauth_enabled') ? '✅' : '❌' }}</span>
+			<span>{{ config_cache('pixelfed.oauth_enabled') ? '✅ true' : '❌ false' }}</span>
 		</li>
 		<li>
 			<strong><span class="badge badge-primary">OAUTH</span> token_expiration</strong>
@@ -74,11 +74,11 @@
 		</li>
 		<li>
 			<strong><span class="badge badge-primary">OAUTH</span> public key exists: </strong>
-			<span>{{ file_exists(storage_path('oauth-public.key')) ? '✅' : '❌' }}</span>
+			<span>{{ file_exists(storage_path('oauth-public.key')) ? '✅ true' : '❌ false' }}</span>
 		</li>
 		<li>
 			<strong><span class="badge badge-primary">OAUTH</span> private key exists: </strong>
-			<span>{{ file_exists(storage_path('oauth-private.key')) ? '✅' : '❌' }}</span>
+			<span>{{ file_exists(storage_path('oauth-private.key')) ? '✅ true' : '❌ false' }}</span>
 		</li>		
 		
 		<hr>
@@ -153,11 +153,43 @@
 			<strong><span class="badge badge-primary">PHP INI</span> max_input_time:</strong>
 			<span>{{ ini_get('max_input_time') }}</span>
 		</li>
+
 		<li>
-			<strong><span class="badge badge-primary">PHP INI</span> file_uploads:</strong>
-			<span>{{ ini_get('file_uploads') ? '✅' : '❌' }}</span>
+			<strong><span class="badge badge-primary">PHP INI</span> file_uploads (On):</strong>
+			<span>{{ ini_get('file_uploads') }}</span>
+		</li>
+		<li>
+			<strong><span class="badge badge-primary">PHP INI - Security</span> allow_url_fopen (true):</strong>
+			<span>{{ ini_get('allow_url_fopen') }}</span>
+		</li>
+		<li>
+			<strong><span class="badge badge-primary">PHP INI - Security</span> allow_url_include (false):</strong>
+			<span>{{ ini_get('allow_url_include') }}</span>
+		</li>
+		<li>
+			<strong><span class="badge badge-primary">PHP INI - Security</span> expose_php (false):</strong>
+			<span>{{ ini_get('expose_php') }}</span>
+		</li>
+		<li>
+			<strong><span class="badge badge-primary">PHP INI - Security</span> display_errors (false):</strong>
+			<span>{{ ini_get('display_errors') }}</span>
+		</li>
+		<li>
+			<strong><span class="badge badge-primary">PHP INI - Security</span> display_startup_errors (false):</strong>
+			<span>{{ ini_get('display_startup_errors') }}</span>
+		</li>
+		<li>
+			<strong><span class="badge badge-primary">PHP INI - Security</span> log_errors (true):</strong>
+			<span>{{ ini_get('log_errors') }}</span>
+		</li>
+		<li>
+			<strong><span class="badge badge-primary">PHP INI - Security</span> ignore_repeated_errors (false):</strong>
+			<span>{{ ini_get('ignore_repeated_errors') }}</span>
+		</li>
+		<li>
+			<strong><span class="badge badge-primary">PHP INI - Security</span> disable_functions:</strong>
+			<span>{{ ini_get('disable_functions') }}</span>
 		</li>
-
 
 	<hr>
 	<p class="font-weight-bold text-muted">
@@ -328,6 +360,11 @@
 		<td><strong>PF_NETWORK_TIMELINE</strong></td>
 		<td><span>{{config_cache('federation.network_timeline') ? '✅ true' : '❌ false' }}</span></td>
 	</tr>
+	<tr>
+		<td><span class="badge badge-primary">FEDERATION</span></td>
+		<td><strong>PF_NETWORK_TIMELINE_DAYS_FALLOFF</strong></td>
+		<td><span>{{config('federation.network_timeline_days_falloff') }}</span></td>
+	</tr>
 	<tr>
 		<td><span class="badge badge-primary">FEDERATION</span></td>
 		<td><strong>CUSTOM_EMOJI</strong></td>
@@ -453,6 +490,23 @@
 		<td><strong>INSTANCE_PUBLIC_LOCAL_TIMELINE</strong></td>
 		<td><span>{{config_cache('instance.timeline.local.is_public') ? '✅ true' : '❌ false' }}</span></td>
 	</tr>
+
+	<tr>
+		<td><span class="badge badge-primary">INSTANCE</span></td>
+		<td><strong>INSTANCE_NETWORK_TIMELINE_CACHED</strong></td>
+		<td><span>{{config('instance.timeline.network.cached') }}</span></td>
+	</tr>
+	<tr>
+		<td><span class="badge badge-primary">INSTANCE</span></td>
+		<td><strong>INSTANCE_NETWORK_TIMELINE_CACHE_DROPOFF</strong></td>
+		<td><span>{{config('instance.timeline.network.cache_dropoff') }}</span></td>
+	</tr>
+	<tr>
+		<td><span class="badge badge-primary">INSTANCE</span></td>
+		<td><strong>INSTANCE_NETWORK_TIMELINE_CACHE_MAX_HOUR_INGEST</strong></td>
+		<td><span>{{config('instance.timeline.network.max_hours_old') }}</span></td>
+	</tr>
+
 	<tr>
 		<td><span class="badge badge-primary">INSTANCE</span></td>
 		<td><strong>PAGE_404_HEADER</strong></td>

+ 33 - 0
resources/views/live/player.blade.php

@@ -0,0 +1,33 @@
+@extends('layouts.blank')
+
+@section('content')
+<div class="force-dark-mode">
+    <live-player id="{{ $id }}"></live-player>
+</div>
+@endsection
+
+@push('scripts')
+<script type="text/javascript" src="/js/live-player.js?v={{ time() }}"></script>
+<script type="text/javascript">App.boot();</script>
+@endpush
+
+@push('meta')
+<script type="text/javascript">
+    window._pushr = {
+        host: "{{ config('broadcasting.connections.pusher.options.host')}}",
+        port: "{{ config('broadcasting.connections.pusher.options.port')}}",
+        key: "{{ config('broadcasting.connections.pusher.key')}}",
+        cluster: "{{ config('broadcasting.connections.pusher.options.cluster')}}"
+    };
+</script>
+@endpush
+
+@push('styles')
+<link rel="stylesheet" type="text/css" href="{{ mix('css/spa.css') }}">
+<style type="text/css">
+body {
+    background-color: #000000;
+    background-image: radial-gradient(circle, #0f172a 0%, #000000 74%);
+}
+</style>
+@endpush

+ 3 - 3
resources/views/site/about.blade.php

@@ -6,9 +6,9 @@
 	<meta http-equiv="X-UA-Compatible" content="IE=edge">
 	<meta name="viewport" content="width=device-width, initial-scale=1">
 	<meta name="mobile-web-app-capable" content="yes">
-	<title>{{ config('app.name', 'Pixelfed') }}</title>
-	<meta property="og:site_name" content="{{ config_cache('app.name', 'pixelfed') }}">
-	<meta property="og:title" content="{{ config_cache('app.name', 'pixelfed') }}">
+	<title>{{ config_cache('app.name') ?? 'pixelfed' }}</title>
+	<meta property="og:site_name" content="{{ config_cache('app.name') ?? 'pixelfed' }}">
+	<meta property="og:title" content="{{ config_cache('app.name') ?? 'pixelfed' }}">
 	<meta property="og:type" content="article">
 	<meta property="og:url" content="{{route('site.about')}}">
 	<meta property="og:description" content="{{config_cache('app.short_description')}}">

+ 6 - 1
routes/api.php

@@ -94,6 +94,7 @@ Route::group(['prefix' => 'api'], function() use($middleware) {
 	Route::group(['prefix' => 'v2'], function() use($middleware) {
 		Route::get('search', 'Api\ApiV1Controller@searchV2')->middleware($middleware);
 		Route::post('media', 'Api\ApiV1Controller@mediaUploadV2')->middleware($middleware);
+		Route::get('streaming/config', 'Api\ApiV1Controller@getWebsocketConfig');
 	});
 
 	Route::group(['prefix' => 'live'], function() use($middleware) {
@@ -101,11 +102,15 @@ Route::group(['prefix' => 'api'], function() use($middleware) {
 		Route::post('stream/edit', 'LiveStreamController@editStream')->middleware($middleware);
 		Route::get('active/list', 'LiveStreamController@getActiveStreams')->middleware($middleware);
 		Route::get('accounts/stream', 'LiveStreamController@getUserStream')->middleware($middleware);
+		Route::get('accounts/stream/guest', 'LiveStreamController@getUserStreamAsGuest');
 		Route::delete('accounts/stream', 'LiveStreamController@deleteStream')->middleware($middleware);
 		Route::get('chat/latest', 'LiveStreamController@getLatestChat')->middleware($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('chat/ban-user', 'LiveStreamController@banChatUser')->middleware($middleware);
+		Route::post('chat/pin', 'LiveStreamController@pinChatComment')->middleware($middleware);
+		Route::post('chat/unpin', 'LiveStreamController@unpinChatComment')->middleware($middleware);
+		Route::get('config', 'LiveStreamController@getConfig');
 		Route::post('broadcast/publish', 'LiveStreamController@clientBroadcastPublish');
 		Route::post('broadcast/finish', 'LiveStreamController@clientBroadcastFinish');
 	});

+ 8 - 0
routes/channels.php

@@ -14,3 +14,11 @@
 Broadcast::channel('App.User.{id}', function ($user, $id) {
     return (int) $user->id === (int) $id;
 });
+
+Broadcast::channel('live.chat.{id}', function ($user, $id) {
+    return true;
+}, ['guards' => ['web', 'api']]);
+
+Broadcast::channel('live.presence.{id}', function ($user, $id) {
+    return [ $user->profile_id ];
+}, ['guards' => ['web', 'api']]);

+ 1 - 0
routes/web.php

@@ -540,6 +540,7 @@ Route::domain(config('pixelfed.domain.app'))->middleware(['validemail', 'twofact
 	Route::get('p/{username}/{id}.json', 'StatusController@showObject');
 	Route::get('p/{username}/{id}', 'StatusController@show');
 	Route::get('{username}/embed', 'ProfileController@embed');
+	Route::get('{username}/live', 'LiveStreamController@showProfilePlayer');
 	Route::get('@{username}@{domain}', 'SiteController@legacyWebfingerRedirect');
 	Route::get('@{username}', 'SiteController@legacyProfileRedirect');
 	Route::get('{username}', 'ProfileController@show');

Энэ ялгаанд хэт олон файл өөрчлөгдсөн тул зарим файлыг харуулаагүй болно