2019-02-07 18:59:38 +00:00
< ? php
namespace App\Http\Controllers\Admin ;
2021-05-08 04:19:58 +00:00
use App\Models\ConfigCache ;
2024-03-05 07:05:05 +00:00
use App\Models\InstanceActor ;
use App\Page ;
use App\Profile ;
2023-05-07 11:08:45 +00:00
use App\Services\AccountService ;
2024-03-14 11:03:19 +00:00
use App\Services\AdminSettingsService ;
2021-05-08 04:19:58 +00:00
use App\Services\ConfigCacheService ;
2024-03-18 11:55:38 +00:00
use App\Services\FilesystemService ;
2024-03-05 07:05:05 +00:00
use App\User ;
2021-05-19 08:01:33 +00:00
use App\Util\Site\Config ;
2024-03-05 07:05:05 +00:00
use Artisan ;
use Cache ;
use DB ;
use Illuminate\Http\Request ;
2019-02-07 18:59:38 +00:00
trait AdminSettingsController
{
2024-02-19 11:00:31 +00:00
public function settings ( Request $request )
{
$cloud_storage = ConfigCacheService :: get ( 'pixelfed.cloud_storage' );
$cloud_disk = config ( 'filesystems.cloud' );
2024-03-05 07:05:05 +00:00
$cloud_ready = ! empty ( config ( 'filesystems.disks.' . $cloud_disk . '.key' )) && ! empty ( config ( 'filesystems.disks.' . $cloud_disk . '.secret' ));
2024-02-19 11:00:31 +00:00
$types = explode ( ',' , ConfigCacheService :: get ( 'pixelfed.media_types' ));
$rules = ConfigCacheService :: get ( 'app.rules' ) ? json_decode ( ConfigCacheService :: get ( 'app.rules' ), true ) : null ;
$jpeg = in_array ( 'image/jpg' , $types ) || in_array ( 'image/jpeg' , $types );
$png = in_array ( 'image/png' , $types );
$gif = in_array ( 'image/gif' , $types );
$mp4 = in_array ( 'video/mp4' , $types );
$webp = in_array ( 'image/webp' , $types );
$availableAdmins = User :: whereIsAdmin ( true ) -> get ();
$currentAdmin = config_cache ( 'instance.admin.pid' ) ? AccountService :: get ( config_cache ( 'instance.admin.pid' ), true ) : null ;
$openReg = ( bool ) config_cache ( 'pixelfed.open_registration' );
$curOnboarding = ( bool ) config_cache ( 'instance.curated_registration.enabled' );
$regState = $openReg ? 'open' : ( $curOnboarding ? 'filtered' : 'closed' );
2024-03-05 07:05:05 +00:00
$accountMigration = ( bool ) config_cache ( 'federation.migration' );
2024-02-19 11:00:31 +00:00
return view ( 'admin.settings.home' , compact (
'jpeg' ,
'png' ,
'gif' ,
'mp4' ,
'webp' ,
'rules' ,
'cloud_storage' ,
'cloud_disk' ,
'cloud_ready' ,
'availableAdmins' ,
'currentAdmin' ,
2024-03-05 07:05:05 +00:00
'regState' ,
'accountMigration'
2024-02-19 11:00:31 +00:00
));
}
public function settingsHomeStore ( Request $request )
{
$this -> validate ( $request , [
'name' => 'nullable|string' ,
'short_description' => 'nullable' ,
'long_description' => 'nullable' ,
'max_photo_size' => 'nullable|integer|min:1' ,
'max_album_length' => 'nullable|integer|min:1|max:100' ,
'image_quality' => 'nullable|integer|min:1|max:100' ,
'type_jpeg' => 'nullable' ,
'type_png' => 'nullable' ,
'type_gif' => 'nullable' ,
'type_mp4' => 'nullable' ,
'type_webp' => 'nullable' ,
'admin_account_id' => 'nullable' ,
2024-03-05 07:05:05 +00:00
'regs' => 'required|in:open,filtered,closed' ,
'account_migration' => 'nullable' ,
2024-03-18 11:55:38 +00:00
'rule_delete' => 'sometimes' ,
2024-02-19 11:00:31 +00:00
]);
$orb = false ;
$cob = false ;
2024-03-05 07:05:05 +00:00
switch ( $request -> input ( 'regs' )) {
2024-02-19 11:00:31 +00:00
case 'open' :
$orb = true ;
$cob = false ;
2024-03-05 07:05:05 +00:00
break ;
2024-02-19 11:00:31 +00:00
case 'filtered' :
$orb = false ;
$cob = true ;
2024-03-05 07:05:05 +00:00
break ;
2024-02-19 11:00:31 +00:00
case 'closed' :
$orb = false ;
$cob = false ;
2024-03-05 07:05:05 +00:00
break ;
2024-02-19 11:00:31 +00:00
}
ConfigCacheService :: put ( 'pixelfed.open_registration' , ( bool ) $orb );
ConfigCacheService :: put ( 'instance.curated_registration.enabled' , ( bool ) $cob );
2024-03-05 07:05:05 +00:00
if ( $request -> filled ( 'admin_account_id' )) {
2024-02-19 11:00:31 +00:00
ConfigCacheService :: put ( 'instance.admin.pid' , $request -> admin_account_id );
Cache :: forget ( 'api:v1:instance-data:contact' );
Cache :: forget ( 'api:v1:instance-data-response-v1' );
}
2024-03-05 07:05:05 +00:00
if ( $request -> filled ( 'rule_delete' )) {
2024-02-19 11:00:31 +00:00
$index = ( int ) $request -> input ( 'rule_delete' );
$rules = ConfigCacheService :: get ( 'app.rules' );
$json = json_decode ( $rules , true );
2024-03-05 07:05:05 +00:00
if ( ! $rules || empty ( $json )) {
2024-02-19 11:00:31 +00:00
return ;
}
unset ( $json [ $index ]);
$json = json_encode ( array_values ( $json ));
ConfigCacheService :: put ( 'app.rules' , $json );
Cache :: forget ( 'api:v1:instance-data:rules' );
Cache :: forget ( 'api:v1:instance-data-response-v1' );
2024-03-05 07:05:05 +00:00
2024-02-19 11:00:31 +00:00
return 200 ;
}
$media_types = explode ( ',' , config_cache ( 'pixelfed.media_types' ));
$media_types_original = $media_types ;
$mimes = [
'type_jpeg' => 'image/jpeg' ,
'type_png' => 'image/png' ,
'type_gif' => 'image/gif' ,
'type_mp4' => 'video/mp4' ,
'type_webp' => 'image/webp' ,
];
foreach ( $mimes as $key => $value ) {
2024-03-05 07:05:05 +00:00
if ( $request -> input ( $key ) == 'on' ) {
if ( ! in_array ( $value , $media_types )) {
2024-02-19 11:00:31 +00:00
array_push ( $media_types , $value );
}
} else {
$media_types = array_diff ( $media_types , [ $value ]);
}
}
2024-03-05 07:05:05 +00:00
if ( $media_types !== $media_types_original ) {
2024-02-19 11:00:31 +00:00
ConfigCacheService :: put ( 'pixelfed.media_types' , implode ( ',' , array_unique ( $media_types )));
}
$keys = [
'name' => 'app.name' ,
'short_description' => 'app.short_description' ,
'long_description' => 'app.description' ,
'max_photo_size' => 'pixelfed.max_photo_size' ,
'max_album_length' => 'pixelfed.max_album_length' ,
'image_quality' => 'pixelfed.image_quality' ,
'account_limit' => 'pixelfed.max_account_size' ,
'custom_css' => 'uikit.custom.css' ,
'custom_js' => 'uikit.custom.js' ,
2024-03-05 07:05:05 +00:00
'about_title' => 'about.title' ,
2024-02-19 11:00:31 +00:00
];
foreach ( $keys as $key => $value ) {
$cc = ConfigCache :: whereK ( $value ) -> first ();
$val = $request -> input ( $key );
2024-03-05 07:05:05 +00:00
if ( $cc && $cc -> v != $val ) {
2024-02-19 11:00:31 +00:00
ConfigCacheService :: put ( $value , $val );
2024-03-05 07:05:05 +00:00
} elseif ( ! empty ( $val )) {
2024-02-19 11:00:31 +00:00
ConfigCacheService :: put ( $value , $val );
}
}
$bools = [
'activitypub' => 'federation.activitypub.enabled' ,
// 'open_registration' => 'pixelfed.open_registration',
'mobile_apis' => 'pixelfed.oauth_enabled' ,
'stories' => 'instance.stories.enabled' ,
'ig_import' => 'pixelfed.import.instagram.enabled' ,
'spam_detection' => 'pixelfed.bouncer.enabled' ,
'require_email_verification' => 'pixelfed.enforce_email_verification' ,
'enforce_account_limit' => 'pixelfed.enforce_account_limit' ,
'show_custom_css' => 'uikit.show_custom.css' ,
'show_custom_js' => 'uikit.show_custom.js' ,
'cloud_storage' => 'pixelfed.cloud_storage' ,
'account_autofollow' => 'account.autofollow' ,
'show_directory' => 'instance.landing.show_directory' ,
'show_explore_feed' => 'instance.landing.show_explore' ,
2024-03-05 07:05:05 +00:00
'account_migration' => 'federation.migration' ,
2024-02-19 11:00:31 +00:00
];
foreach ( $bools as $key => $value ) {
$active = $request -> input ( $key ) == 'on' ;
2024-03-05 07:05:05 +00:00
if ( $key == 'activitypub' && $active && ! InstanceActor :: exists ()) {
2024-02-19 11:00:31 +00:00
Artisan :: call ( 'instance:actor' );
}
2024-03-05 07:05:05 +00:00
if ( $key == 'mobile_apis' &&
2024-02-19 11:00:31 +00:00
$active &&
2024-03-05 07:05:05 +00:00
! file_exists ( storage_path ( 'oauth-public.key' )) &&
2024-08-18 18:50:04 +00:00
! config_cache ( 'passport.public_key' ) &&
! file_exists ( storage_path ( 'oauth-private.key' )) &&
! config_cache ( 'passport.private_key' )
2024-02-19 11:00:31 +00:00
) {
Artisan :: call ( 'passport:keys' );
Artisan :: call ( 'route:cache' );
}
2024-03-05 07:05:05 +00:00
if ( config_cache ( $value ) !== $active ) {
2024-02-19 11:00:31 +00:00
ConfigCacheService :: put ( $value , ( bool ) $active );
}
}
2024-03-05 07:05:05 +00:00
if ( $request -> filled ( 'new_rule' )) {
2024-02-19 11:00:31 +00:00
$rules = ConfigCacheService :: get ( 'app.rules' );
$val = $request -> input ( 'new_rule' );
2024-03-05 07:05:05 +00:00
if ( ! $rules ) {
2024-02-19 11:00:31 +00:00
ConfigCacheService :: put ( 'app.rules' , json_encode ([ $val ]));
} else {
$json = json_decode ( $rules , true );
$json [] = $val ;
ConfigCacheService :: put ( 'app.rules' , json_encode ( array_values ( $json )));
}
Cache :: forget ( 'api:v1:instance-data:rules' );
Cache :: forget ( 'api:v1:instance-data-response-v1' );
}
2024-03-05 07:05:05 +00:00
if ( $request -> filled ( 'account_autofollow_usernames' )) {
2024-02-19 11:00:31 +00:00
$usernames = explode ( ',' , $request -> input ( 'account_autofollow_usernames' ));
$names = [];
2024-03-05 07:05:05 +00:00
foreach ( $usernames as $n ) {
2024-02-19 11:00:31 +00:00
$p = Profile :: whereUsername ( $n ) -> first ();
2024-03-05 07:05:05 +00:00
if ( ! $p ) {
2024-02-19 11:00:31 +00:00
continue ;
}
array_push ( $names , $p -> username );
}
ConfigCacheService :: put ( 'account.autofollow_usernames' , implode ( ',' , $names ));
}
Cache :: forget ( Config :: CACHE_KEY );
return redirect ( '/i/admin/settings' ) -> with ( 'status' , 'Successfully updated settings!' );
}
public function settingsBackups ( Request $request )
{
$path = storage_path ( 'app/' . config ( 'app.name' ));
$files = is_dir ( $path ) ? new \DirectoryIterator ( $path ) : [];
2024-03-05 07:05:05 +00:00
2024-02-19 11:00:31 +00:00
return view ( 'admin.settings.backups' , compact ( 'files' ));
}
public function settingsMaintenance ( Request $request )
{
return view ( 'admin.settings.maintenance' );
}
public function settingsStorage ( Request $request )
{
$storage = [];
2024-03-05 07:05:05 +00:00
2024-02-19 11:00:31 +00:00
return view ( 'admin.settings.storage' , compact ( 'storage' ));
}
public function settingsFeatures ( Request $request )
{
return view ( 'admin.settings.features' );
}
public function settingsPages ( Request $request )
{
$pages = Page :: orderByDesc ( 'updated_at' ) -> paginate ( 10 );
2024-03-05 07:05:05 +00:00
2024-02-19 11:00:31 +00:00
return view ( 'admin.pages.home' , compact ( 'pages' ));
}
public function settingsPageEdit ( Request $request )
{
return view ( 'admin.pages.edit' );
}
public function settingsSystem ( Request $request )
{
$sys = [
'pixelfed' => config ( 'pixelfed.version' ),
'php' => phpversion (),
'laravel' => app () -> version (),
];
switch ( config ( 'database.default' )) {
case 'pgsql' :
2024-03-05 07:05:05 +00:00
$exp = DB :: raw ( 'select version();' );
$expQuery = $exp -> getValue ( DB :: connection () -> getQueryGrammar ());
$sys [ 'database' ] = [
'name' => 'Postgres' ,
'version' => explode ( ' ' , DB :: select ( $expQuery )[ 0 ] -> version )[ 1 ],
];
break ;
2024-02-19 11:00:31 +00:00
case 'mysql' :
2024-03-05 07:05:05 +00:00
$exp = DB :: raw ( 'select version()' );
$expQuery = $exp -> getValue ( DB :: connection () -> getQueryGrammar ());
$sys [ 'database' ] = [
'name' => 'MySQL' ,
'version' => DB :: select ( $expQuery )[ 0 ] -> { 'version()' },
];
break ;
2024-02-19 11:00:31 +00:00
default :
2024-03-05 07:05:05 +00:00
$sys [ 'database' ] = [
'name' => 'Unknown' ,
'version' => '?' ,
];
break ;
2024-02-19 11:00:31 +00:00
}
2024-03-05 07:05:05 +00:00
2024-02-19 11:00:31 +00:00
return view ( 'admin.settings.system' , compact ( 'sys' ));
}
2024-03-14 11:03:19 +00:00
public function settingsApiFetch ( Request $request )
{
$cloud_storage = ConfigCacheService :: get ( 'pixelfed.cloud_storage' );
$cloud_disk = config ( 'filesystems.cloud' );
$cloud_ready = ! empty ( config ( 'filesystems.disks.' . $cloud_disk . '.key' )) && ! empty ( config ( 'filesystems.disks.' . $cloud_disk . '.secret' ));
$types = explode ( ',' , ConfigCacheService :: get ( 'pixelfed.media_types' ));
$rules = ConfigCacheService :: get ( 'app.rules' ) ? json_decode ( ConfigCacheService :: get ( 'app.rules' ), true ) : [];
$jpeg = in_array ( 'image/jpg' , $types ) || in_array ( 'image/jpeg' , $types );
$png = in_array ( 'image/png' , $types );
$gif = in_array ( 'image/gif' , $types );
$mp4 = in_array ( 'video/mp4' , $types );
$webp = in_array ( 'image/webp' , $types );
$availableAdmins = User :: whereIsAdmin ( true ) -> get ();
$currentAdmin = config_cache ( 'instance.admin.pid' ) ? AccountService :: get ( config_cache ( 'instance.admin.pid' ), true ) : null ;
$openReg = ( bool ) config_cache ( 'pixelfed.open_registration' );
$curOnboarding = ( bool ) config_cache ( 'instance.curated_registration.enabled' );
$regState = $openReg ? 'open' : ( $curOnboarding ? 'filtered' : 'closed' );
$accountMigration = ( bool ) config_cache ( 'federation.migration' );
$autoFollow = config_cache ( 'account.autofollow_usernames' );
2024-03-18 11:55:38 +00:00
if ( strlen ( $autoFollow ) > 3 ) {
2024-03-14 11:03:19 +00:00
$autoFollow = explode ( ',' , $autoFollow );
}
$res = AdminSettingsService :: getAll ();
return response () -> json ( $res );
}
public function settingsApiRulesAdd ( Request $request )
{
$this -> validate ( $request , [
2024-03-18 11:55:38 +00:00
'rule' => 'required|string|min:5|max:1000' ,
2024-03-14 11:03:19 +00:00
]);
$rules = ConfigCacheService :: get ( 'app.rules' );
$val = $request -> input ( 'rule' );
if ( ! $rules ) {
ConfigCacheService :: put ( 'app.rules' , json_encode ([ $val ]));
} else {
$json = json_decode ( $rules , true );
$count = count ( $json );
2024-03-18 11:55:38 +00:00
if ( $count >= 30 ) {
2024-03-14 11:03:19 +00:00
return response () -> json ([ 'message' => 'Max rules limit reached, you can set up to 30 rules at a time.' ], 400 );
}
$json [] = $val ;
ConfigCacheService :: put ( 'app.rules' , json_encode ( array_values ( $json )));
}
Cache :: forget ( 'api:v1:instance-data:rules' );
Cache :: forget ( 'api:v1:instance-data-response-v1' );
Cache :: forget ( 'api:v2:instance-data-response-v2' );
Config :: refresh ();
2024-03-18 11:55:38 +00:00
2024-03-14 11:03:19 +00:00
return [ $val ];
}
public function settingsApiRulesDelete ( Request $request )
{
$this -> validate ( $request , [
'rule' => 'required|string' ,
]);
$rules = ConfigCacheService :: get ( 'app.rules' );
$val = $request -> input ( 'rule' );
if ( ! $rules ) {
return [];
} else {
$json = json_decode ( $rules , true );
$idx = array_search ( $val , $json );
2024-03-18 11:55:38 +00:00
if ( $idx !== false ) {
2024-03-14 11:03:19 +00:00
unset ( $json [ $idx ]);
$json = array_values ( $json );
}
ConfigCacheService :: put ( 'app.rules' , json_encode ( array_values ( $json )));
}
Cache :: forget ( 'api:v1:instance-data:rules' );
Cache :: forget ( 'api:v1:instance-data-response-v1' );
Cache :: forget ( 'api:v2:instance-data-response-v2' );
Config :: refresh ();
return response () -> json ( $json );
}
public function settingsApiRulesDeleteAll ( Request $request )
{
$rules = ConfigCacheService :: get ( 'app.rules' );
if ( ! $rules ) {
return [];
} else {
ConfigCacheService :: put ( 'app.rules' , json_encode ([]));
}
Cache :: forget ( 'api:v1:instance-data:rules' );
Cache :: forget ( 'api:v1:instance-data-response-v1' );
Cache :: forget ( 'api:v2:instance-data-response-v2' );
Config :: refresh ();
return response () -> json ([]);
}
public function settingsApiAutofollowDelete ( Request $request )
{
$this -> validate ( $request , [
'username' => 'required|string' ,
]);
$username = $request -> input ( 'username' );
$names = [];
$existing = config_cache ( 'account.autofollow_usernames' );
2024-03-18 11:55:38 +00:00
if ( $existing ) {
2024-03-14 11:03:19 +00:00
$names = explode ( ',' , $existing );
}
2024-03-18 11:55:38 +00:00
if ( in_array ( $username , $names )) {
2024-03-14 11:03:19 +00:00
$key = array_search ( $username , $names );
2024-03-18 11:55:38 +00:00
if ( $key !== false ) {
2024-03-14 11:03:19 +00:00
unset ( $names [ $key ]);
}
}
ConfigCacheService :: put ( 'account.autofollow_usernames' , implode ( ',' , $names ));
2024-03-18 11:55:38 +00:00
2024-03-14 11:03:19 +00:00
return response () -> json ([ 'accounts' => array_values ( $names )]);
}
public function settingsApiAutofollowAdd ( Request $request )
{
$this -> validate ( $request , [
'username' => 'required|string' ,
]);
$username = $request -> input ( 'username' );
$names = [];
$existing = config_cache ( 'account.autofollow_usernames' );
2024-03-18 11:55:38 +00:00
if ( $existing ) {
2024-03-14 11:03:19 +00:00
$names = explode ( ',' , $existing );
}
2024-03-18 11:55:38 +00:00
if ( $existing && count ( $names )) {
if ( count ( $names ) >= 5 ) {
return response () -> json ([ 'message' => 'You can only add up to 5 accounts to be autofollowed.' ], 400 );
}
if ( in_array ( strtolower ( $username ), array_map ( 'strtolower' , $names ))) {
return response () -> json ([ 'message' => 'User already exists, please try again.' ], 400 );
}
}
$p = User :: whereUsername ( $username ) -> whereNull ( 'status' ) -> first ();
2024-03-14 11:03:19 +00:00
if ( ! $p || in_array ( $p -> username , $names )) {
abort ( 404 );
}
2024-03-18 11:55:38 +00:00
array_push ( $names , $p -> username );
2024-03-14 11:03:19 +00:00
ConfigCacheService :: put ( 'account.autofollow_usernames' , implode ( ',' , $names ));
2024-03-18 11:55:38 +00:00
2024-03-14 11:03:19 +00:00
return response () -> json ([ 'accounts' => array_values ( $names )]);
}
public function settingsApiUpdateType ( Request $request , $type )
{
abort_unless ( in_array ( $type , [
'posts' ,
'platform' ,
'home' ,
'landing' ,
'branding' ,
'media' ,
'users' ,
'storage' ,
]), 400 );
switch ( $type ) {
case 'home' :
return $this -> settingsApiUpdateHomeType ( $request );
2024-03-18 11:55:38 +00:00
break ;
2024-03-14 11:03:19 +00:00
case 'landing' :
return $this -> settingsApiUpdateLandingType ( $request );
2024-03-18 11:55:38 +00:00
break ;
2024-03-14 11:03:19 +00:00
case 'posts' :
return $this -> settingsApiUpdatePostsType ( $request );
break ;
case 'platform' :
return $this -> settingsApiUpdatePlatformType ( $request );
break ;
case 'branding' :
return $this -> settingsApiUpdateBrandingType ( $request );
break ;
case 'media' :
return $this -> settingsApiUpdateMediaType ( $request );
break ;
case 'users' :
return $this -> settingsApiUpdateUsersType ( $request );
break ;
case 'storage' :
return $this -> settingsApiUpdateStorageType ( $request );
break ;
default :
abort ( 404 );
break ;
}
}
public function settingsApiUpdateHomeType ( $request )
{
$this -> validate ( $request , [
'registration_status' => 'required|in:open,filtered,closed' ,
'cloud_storage' => 'required' ,
'activitypub_enabled' => 'required' ,
2024-08-02 03:47:04 +00:00
'authorized_fetch' => 'required' ,
2024-03-14 11:03:19 +00:00
'account_migration' => 'required' ,
'mobile_apis' => 'required' ,
'stories' => 'required' ,
'instagram_import' => 'required' ,
'autospam_enabled' => 'required' ,
]);
$regStatus = $request -> input ( 'registration_status' );
ConfigCacheService :: put ( 'pixelfed.open_registration' , $regStatus === 'open' );
ConfigCacheService :: put ( 'instance.curated_registration.enabled' , $regStatus === 'filtered' );
$cloudStorage = $request -> boolean ( 'cloud_storage' );
2024-03-18 11:55:38 +00:00
if ( $cloudStorage !== ( bool ) config_cache ( 'pixelfed.cloud_storage' )) {
if ( ! $cloudStorage ) {
2024-03-14 11:03:19 +00:00
ConfigCacheService :: put ( 'pixelfed.cloud_storage' , false );
} else {
$cloud_disk = config ( 'filesystems.cloud' );
$cloud_ready = ! empty ( config ( 'filesystems.disks.' . $cloud_disk . '.key' )) && ! empty ( config ( 'filesystems.disks.' . $cloud_disk . '.secret' ));
2024-03-18 11:55:38 +00:00
if ( ! $cloud_ready ) {
2024-03-14 11:03:19 +00:00
return redirect () -> back () -> withErrors ([ 'cloud_storage' => 'Must configure cloud storage before enabling!' ]);
} else {
ConfigCacheService :: put ( 'pixelfed.cloud_storage' , true );
}
}
}
2024-08-02 03:47:04 +00:00
ConfigCacheService :: put ( 'federation.activitypub.authorized_fetch' , $request -> boolean ( 'authorized_fetch' ));
2024-03-14 11:03:19 +00:00
ConfigCacheService :: put ( 'federation.activitypub.enabled' , $request -> boolean ( 'activitypub_enabled' ));
ConfigCacheService :: put ( 'federation.migration' , $request -> boolean ( 'account_migration' ));
ConfigCacheService :: put ( 'pixelfed.oauth_enabled' , $request -> boolean ( 'mobile_apis' ));
ConfigCacheService :: put ( 'instance.stories.enabled' , $request -> boolean ( 'stories' ));
ConfigCacheService :: put ( 'pixelfed.import.instagram.enabled' , $request -> boolean ( 'instagram_import' ));
ConfigCacheService :: put ( 'pixelfed.bouncer.enabled' , $request -> boolean ( 'autospam_enabled' ));
Cache :: forget ( 'api:v1:instance-data-response-v1' );
Cache :: forget ( 'api:v2:instance-data-response-v2' );
Cache :: forget ( 'api:v1:instance-data:contact' );
Config :: refresh ();
2024-03-18 11:55:38 +00:00
2024-03-14 11:03:19 +00:00
return $request -> all ();
}
public function settingsApiUpdateLandingType ( $request )
{
$this -> validate ( $request , [
'current_admin' => 'required' ,
'show_directory' => 'required' ,
'show_explore' => 'required' ,
]);
ConfigCacheService :: put ( 'instance.admin.pid' , $request -> input ( 'current_admin' ));
ConfigCacheService :: put ( 'instance.landing.show_directory' , $request -> boolean ( 'show_directory' ));
ConfigCacheService :: put ( 'instance.landing.show_explore' , $request -> boolean ( 'show_explore' ));
Cache :: forget ( 'api:v1:instance-data:rules' );
Cache :: forget ( 'api:v1:instance-data-response-v1' );
Cache :: forget ( 'api:v2:instance-data-response-v2' );
Cache :: forget ( 'api:v1:instance-data:contact' );
Config :: refresh ();
return $request -> all ();
}
public function settingsApiUpdateMediaType ( $request )
{
$this -> validate ( $request , [
'image_quality' => 'required|integer|min:1|max:100' ,
'max_album_length' => 'required|integer|min:1|max:20' ,
2025-01-05 18:31:44 +00:00
'max_photo_size' => 'required|integer|min:100|max:1000000' ,
2024-03-14 11:03:19 +00:00
'media_types' => 'required' ,
'optimize_image' => 'required' ,
'optimize_video' => 'required' ,
]);
$mediaTypes = $request -> input ( 'media_types' );
$mediaArray = explode ( ',' , $mediaTypes );
foreach ( $mediaArray as $mediaType ) {
2024-03-18 11:55:38 +00:00
if ( ! in_array ( $mediaType , [ 'image/jpeg' , 'image/png' , 'image/gif' , 'image/webp' , 'video/mp4' ])) {
2024-03-14 11:03:19 +00:00
return redirect () -> back () -> withErrors ([ 'media_types' => 'Invalid media type' ]);
}
}
ConfigCacheService :: put ( 'pixelfed.media_types' , $request -> input ( 'media_types' ));
ConfigCacheService :: put ( 'pixelfed.image_quality' , $request -> input ( 'image_quality' ));
ConfigCacheService :: put ( 'pixelfed.max_album_length' , $request -> input ( 'max_album_length' ));
ConfigCacheService :: put ( 'pixelfed.max_photo_size' , $request -> input ( 'max_photo_size' ));
ConfigCacheService :: put ( 'pixelfed.optimize_image' , $request -> boolean ( 'optimize_image' ));
ConfigCacheService :: put ( 'pixelfed.optimize_video' , $request -> boolean ( 'optimize_video' ));
Cache :: forget ( 'api:v1:instance-data:rules' );
Cache :: forget ( 'api:v1:instance-data-response-v1' );
Cache :: forget ( 'api:v2:instance-data-response-v2' );
Cache :: forget ( 'api:v1:instance-data:contact' );
Config :: refresh ();
return $request -> all ();
}
public function settingsApiUpdateBrandingType ( $request )
{
$this -> validate ( $request , [
'name' => 'required' ,
'short_description' => 'required' ,
'long_description' => 'required' ,
]);
ConfigCacheService :: put ( 'app.name' , $request -> input ( 'name' ));
ConfigCacheService :: put ( 'app.short_description' , $request -> input ( 'short_description' ));
ConfigCacheService :: put ( 'app.description' , $request -> input ( 'long_description' ));
Cache :: forget ( 'api:v1:instance-data:rules' );
Cache :: forget ( 'api:v1:instance-data-response-v1' );
Cache :: forget ( 'api:v2:instance-data-response-v2' );
Cache :: forget ( 'api:v1:instance-data:contact' );
Config :: refresh ();
return $request -> all ();
}
public function settingsApiUpdatePostsType ( $request )
{
$this -> validate ( $request , [
'max_caption_length' => 'required|integer|min:5|max:10000' ,
'max_altext_length' => 'required|integer|min:5|max:40000' ,
]);
ConfigCacheService :: put ( 'pixelfed.max_caption_length' , $request -> input ( 'max_caption_length' ));
ConfigCacheService :: put ( 'pixelfed.max_altext_length' , $request -> input ( 'max_altext_length' ));
$res = [
'max_caption_length' => $request -> input ( 'max_caption_length' ),
'max_altext_length' => $request -> input ( 'max_altext_length' ),
];
Cache :: forget ( 'api:v1:instance-data:rules' );
Cache :: forget ( 'api:v1:instance-data-response-v1' );
Cache :: forget ( 'api:v2:instance-data-response-v2' );
Config :: refresh ();
2024-03-18 11:55:38 +00:00
2024-03-14 11:03:19 +00:00
return $res ;
}
public function settingsApiUpdatePlatformType ( $request )
{
$this -> validate ( $request , [
'allow_app_registration' => 'required' ,
'app_registration_rate_limit_attempts' => 'required|integer|min:1' ,
'app_registration_rate_limit_decay' => 'required|integer|min:1' ,
'app_registration_confirm_rate_limit_attempts' => 'required|integer|min:1' ,
'app_registration_confirm_rate_limit_decay' => 'required|integer|min:1' ,
'allow_post_embeds' => 'required' ,
'allow_profile_embeds' => 'required' ,
'captcha_enabled' => 'required' ,
'captcha_on_login' => 'required_if_accepted:captcha_enabled' ,
'captcha_on_register' => 'required_if_accepted:captcha_enabled' ,
'captcha_secret' => 'required_if_accepted:captcha_enabled' ,
'captcha_sitekey' => 'required_if_accepted:captcha_enabled' ,
'custom_emoji_enabled' => 'required' ,
]);
ConfigCacheService :: put ( 'pixelfed.allow_app_registration' , $request -> boolean ( 'allow_app_registration' ));
ConfigCacheService :: put ( 'pixelfed.app_registration_rate_limit_attempts' , $request -> input ( 'app_registration_rate_limit_attempts' ));
ConfigCacheService :: put ( 'pixelfed.app_registration_rate_limit_decay' , $request -> input ( 'app_registration_rate_limit_decay' ));
ConfigCacheService :: put ( 'pixelfed.app_registration_confirm_rate_limit_attempts' , $request -> input ( 'app_registration_confirm_rate_limit_attempts' ));
ConfigCacheService :: put ( 'pixelfed.app_registration_confirm_rate_limit_decay' , $request -> input ( 'app_registration_confirm_rate_limit_decay' ));
ConfigCacheService :: put ( 'instance.embed.post' , $request -> boolean ( 'allow_post_embeds' ));
ConfigCacheService :: put ( 'instance.embed.profile' , $request -> boolean ( 'allow_profile_embeds' ));
ConfigCacheService :: put ( 'federation.custom_emoji.enabled' , $request -> boolean ( 'custom_emoji_enabled' ));
$captcha = $request -> boolean ( 'captcha_enabled' );
2024-03-18 11:55:38 +00:00
if ( $captcha ) {
2024-03-14 11:03:19 +00:00
$secret = $request -> input ( 'captcha_secret' );
$sitekey = $request -> input ( 'captcha_sitekey' );
2024-03-18 11:55:38 +00:00
if ( config_cache ( 'captcha.secret' ) != $secret && strpos ( $secret , '*' ) === false ) {
2024-03-14 11:03:19 +00:00
ConfigCacheService :: put ( 'captcha.secret' , $secret );
}
2024-03-18 11:55:38 +00:00
if ( config_cache ( 'captcha.sitekey' ) != $sitekey && strpos ( $sitekey , '*' ) === false ) {
2024-03-14 11:03:19 +00:00
ConfigCacheService :: put ( 'captcha.sitekey' , $sitekey );
}
ConfigCacheService :: put ( 'captcha.active.login' , $request -> boolean ( 'captcha_on_login' ));
ConfigCacheService :: put ( 'captcha.active.register' , $request -> boolean ( 'captcha_on_register' ));
ConfigCacheService :: put ( 'captcha.triggers.login.enabled' , $request -> boolean ( 'captcha_on_login' ));
ConfigCacheService :: put ( 'captcha.enabled' , true );
} else {
ConfigCacheService :: put ( 'captcha.enabled' , false );
}
$res = [
'allow_app_registration' => $request -> boolean ( 'allow_app_registration' ),
'app_registration_rate_limit_attempts' => $request -> input ( 'app_registration_rate_limit_attempts' ),
'app_registration_rate_limit_decay' => $request -> input ( 'app_registration_rate_limit_decay' ),
'app_registration_confirm_rate_limit_attempts' => $request -> input ( 'app_registration_confirm_rate_limit_attempts' ),
'app_registration_confirm_rate_limit_decay' => $request -> input ( 'app_registration_confirm_rate_limit_decay' ),
'allow_post_embeds' => $request -> boolean ( 'allow_post_embeds' ),
'allow_profile_embeds' => $request -> boolean ( 'allow_profile_embeds' ),
'captcha_enabled' => $request -> boolean ( 'captcha_enabled' ),
'captcha_on_login' => $request -> boolean ( 'captcha_on_login' ),
'captcha_on_register' => $request -> boolean ( 'captcha_on_register' ),
'captcha_secret' => $request -> input ( 'captcha_secret' ),
'captcha_sitekey' => $request -> input ( 'captcha_sitekey' ),
'custom_emoji_enabled' => $request -> boolean ( 'custom_emoji_enabled' ),
];
Cache :: forget ( 'api:v1:instance-data:rules' );
Cache :: forget ( 'api:v1:instance-data-response-v1' );
Cache :: forget ( 'api:v2:instance-data-response-v2' );
Config :: refresh ();
2024-03-18 11:55:38 +00:00
2024-03-14 11:03:19 +00:00
return $res ;
}
public function settingsApiUpdateUsersType ( $request )
{
$this -> validate ( $request , [
'require_email_verification' => 'required' ,
'enforce_account_limit' => 'required' ,
2024-03-18 12:09:36 +00:00
'max_account_size' => 'required|integer|min:50000' ,
2024-03-16 11:13:06 +00:00
'admin_autofollow' => 'required' ,
2024-03-18 11:55:38 +00:00
'admin_autofollow_accounts' => 'sometimes' ,
2024-03-18 12:13:27 +00:00
'max_user_blocks' => 'required|integer|min:0|max:5000' ,
'max_user_mutes' => 'required|integer|min:0|max:5000' ,
'max_domain_blocks' => 'required|integer|min:0|max:5000' ,
2024-03-14 11:03:19 +00:00
]);
2024-03-18 11:55:38 +00:00
$adminAutofollow = $request -> boolean ( 'admin_autofollow' );
$adminAutofollowAccounts = $request -> input ( 'admin_autofollow_accounts' );
if ( $adminAutofollow ) {
if ( $request -> filled ( 'admin_autofollow_accounts' )) {
$names = [];
$existing = config_cache ( 'account.autofollow_usernames' );
if ( $existing ) {
$names = explode ( ',' , $existing );
foreach ( array_map ( 'strtolower' , $adminAutofollowAccounts ) as $afc ) {
if ( in_array ( strtolower ( $afc ), array_map ( 'strtolower' , $names ))) {
continue ;
}
$names [] = $afc ;
}
} else {
$names = $adminAutofollowAccounts ;
}
if ( ! $names || count ( $names ) == 0 ) {
return response () -> json ([ 'message' => 'You need to assign autofollow accounts before you can enable it.' ], 400 );
}
if ( count ( $names ) > 5 ) {
return response () -> json ([ 'message' => 'You can only add up to 5 accounts to be autofollowed.' . json_encode ( $names )], 400 );
}
$autofollows = User :: whereIn ( 'username' , $names ) -> whereNull ( 'status' ) -> pluck ( 'username' );
$adminAutofollowAccounts = $autofollows -> implode ( ',' );
ConfigCacheService :: put ( 'account.autofollow_usernames' , $adminAutofollowAccounts );
} else {
return response () -> json ([ 'message' => 'You need to assign autofollow accounts before you can enable it.' ], 400 );
}
}
2024-03-14 11:03:19 +00:00
ConfigCacheService :: put ( 'pixelfed.enforce_email_verification' , $request -> boolean ( 'require_email_verification' ));
ConfigCacheService :: put ( 'pixelfed.enforce_account_limit' , $request -> boolean ( 'enforce_account_limit' ));
2024-03-18 12:09:36 +00:00
ConfigCacheService :: put ( 'pixelfed.max_account_size' , $request -> input ( 'max_account_size' ));
2024-03-14 11:03:19 +00:00
ConfigCacheService :: put ( 'account.autofollow' , $request -> boolean ( 'admin_autofollow' ));
2024-03-16 11:13:06 +00:00
ConfigCacheService :: put ( 'instance.user_filters.max_user_blocks' , ( int ) $request -> input ( 'max_user_blocks' ));
ConfigCacheService :: put ( 'instance.user_filters.max_user_mutes' , ( int ) $request -> input ( 'max_user_mutes' ));
ConfigCacheService :: put ( 'instance.user_filters.max_domain_blocks' , ( int ) $request -> input ( 'max_domain_blocks' ));
2024-03-14 11:03:19 +00:00
$res = [
'require_email_verification' => $request -> boolean ( 'require_email_verification' ),
'enforce_account_limit' => $request -> boolean ( 'enforce_account_limit' ),
'admin_autofollow' => $request -> boolean ( 'admin_autofollow' ),
2024-03-18 11:55:38 +00:00
'admin_autofollow_accounts' => $adminAutofollowAccounts ,
2024-03-16 11:13:06 +00:00
'max_user_blocks' => $request -> input ( 'max_user_blocks' ),
'max_user_mutes' => $request -> input ( 'max_user_mutes' ),
'max_domain_blocks' => $request -> input ( 'max_domain_blocks' ),
2024-03-14 11:03:19 +00:00
];
Cache :: forget ( 'api:v1:instance-data:rules' );
Cache :: forget ( 'api:v1:instance-data-response-v1' );
Cache :: forget ( 'api:v2:instance-data-response-v2' );
Config :: refresh ();
2024-03-18 11:55:38 +00:00
2024-03-14 11:03:19 +00:00
return $res ;
}
public function settingsApiUpdateStorageType ( $request )
{
$this -> validate ( $request , [
'primary_disk' => 'required|in:local,cloud' ,
'update_disk' => 'sometimes' ,
'disk_config' => 'required_if_accepted:update_disk' ,
'disk_config.driver' => 'required|in:s3,spaces' ,
'disk_config.key' => 'required' ,
'disk_config.secret' => 'required' ,
'disk_config.region' => 'required' ,
'disk_config.bucket' => 'required' ,
'disk_config.visibility' => 'required' ,
'disk_config.endpoint' => 'required' ,
'disk_config.url' => 'nullable' ,
]);
ConfigCacheService :: put ( 'pixelfed.cloud_storage' , $request -> input ( 'primary_disk' ) === 'cloud' );
$res = [
'primary_disk' => $request -> input ( 'primary_disk' ),
];
2024-03-18 11:55:38 +00:00
if ( $request -> has ( 'update_disk' )) {
2024-03-14 11:03:19 +00:00
$res [ 'disk_config' ] = $request -> input ( 'disk_config' );
$changes = [];
$dkey = $request -> input ( 'disk_config.driver' ) === 's3' ? 'filesystems.disks.s3.' : 'filesystems.disks.spaces.' ;
$key = $request -> input ( 'disk_config.key' );
$ckey = null ;
$secret = $request -> input ( 'disk_config.secret' );
$csecret = null ;
$region = $request -> input ( 'disk_config.region' );
$bucket = $request -> input ( 'disk_config.bucket' );
$visibility = $request -> input ( 'disk_config.visibility' );
$url = $request -> input ( 'disk_config.url' );
$endpoint = $request -> input ( 'disk_config.endpoint' );
2024-03-18 11:55:38 +00:00
if ( strpos ( $key , '*' ) === false && $key != config_cache ( $dkey . 'key' )) {
2024-03-14 11:03:19 +00:00
array_push ( $changes , 'key' );
} else {
2024-03-18 11:55:38 +00:00
$ckey = config_cache ( $dkey . 'key' );
2024-03-14 11:03:19 +00:00
}
2024-03-18 11:55:38 +00:00
if ( strpos ( $secret , '*' ) === false && $secret != config_cache ( $dkey . 'secret' )) {
2024-03-14 11:03:19 +00:00
array_push ( $changes , 'secret' );
} else {
2024-03-18 11:55:38 +00:00
$csecret = config_cache ( $dkey . 'secret' );
2024-03-14 11:03:19 +00:00
}
2024-03-18 11:55:38 +00:00
if ( $region != config_cache ( $dkey . 'region' )) {
2024-03-14 11:03:19 +00:00
array_push ( $changes , 'region' );
}
2024-03-18 11:55:38 +00:00
if ( $bucket != config_cache ( $dkey . 'bucket' )) {
2024-03-14 11:03:19 +00:00
array_push ( $changes , 'bucket' );
}
2024-03-18 11:55:38 +00:00
if ( $visibility != config_cache ( $dkey . 'visibility' )) {
2024-03-14 11:03:19 +00:00
array_push ( $changes , 'visibility' );
}
2024-03-18 11:55:38 +00:00
if ( $url != config_cache ( $dkey . 'url' )) {
2024-03-14 11:03:19 +00:00
array_push ( $changes , 'url' );
}
2024-03-18 11:55:38 +00:00
if ( $endpoint != config_cache ( $dkey . 'endpoint' )) {
2024-03-14 11:03:19 +00:00
array_push ( $changes , 'endpoint' );
}
2024-03-18 11:55:38 +00:00
if ( $changes && count ( $changes )) {
2024-03-14 11:03:19 +00:00
$isValid = FilesystemService :: getVerifyCredentials (
$ckey ? ? $key ,
$csecret ? ? $secret ,
$region ,
$bucket ,
$endpoint ,
);
2024-03-18 11:55:38 +00:00
if ( ! $isValid ) {
2024-03-14 11:03:19 +00:00
return response () -> json ([ 'error' => true , 's3_vce' => true , 'message' => " <div class='border border-danger text-danger p-3 font-weight-bold rounded-lg'>The S3/Spaces credentials you provided are invalid, or the bucket does not have the proper permissions.</div><br/>Please check all fields and try again.<br/><br/><strong>Any cloud storage configuration changes you made have NOT been saved due to invalid credentials.</strong> " ], 400 );
}
}
$res [ 'changes' ] = json_encode ( $changes );
}
Cache :: forget ( 'api:v1:instance-data:rules' );
Cache :: forget ( 'api:v1:instance-data-response-v1' );
Cache :: forget ( 'api:v2:instance-data-response-v2' );
Config :: refresh ();
2024-03-18 11:55:38 +00:00
2024-03-14 11:03:19 +00:00
return $res ;
}
2021-05-08 04:19:58 +00:00
}