AdminSettingsController.php 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898
  1. <?php
  2. namespace App\Http\Controllers\Admin;
  3. use App\Models\ConfigCache;
  4. use App\Models\InstanceActor;
  5. use App\Page;
  6. use App\Profile;
  7. use App\Services\AccountService;
  8. use App\Services\AdminSettingsService;
  9. use App\Services\ConfigCacheService;
  10. use App\Services\FilesystemService;
  11. use App\User;
  12. use App\Util\Site\Config;
  13. use Artisan;
  14. use Cache;
  15. use DB;
  16. use Illuminate\Http\Request;
  17. trait AdminSettingsController
  18. {
  19. public function settings(Request $request)
  20. {
  21. $cloud_storage = ConfigCacheService::get('pixelfed.cloud_storage');
  22. $cloud_disk = config('filesystems.cloud');
  23. $cloud_ready = ! empty(config('filesystems.disks.'.$cloud_disk.'.key')) && ! empty(config('filesystems.disks.'.$cloud_disk.'.secret'));
  24. $types = explode(',', ConfigCacheService::get('pixelfed.media_types'));
  25. $rules = ConfigCacheService::get('app.rules') ? json_decode(ConfigCacheService::get('app.rules'), true) : null;
  26. $jpeg = in_array('image/jpg', $types) || in_array('image/jpeg', $types);
  27. $png = in_array('image/png', $types);
  28. $gif = in_array('image/gif', $types);
  29. $mp4 = in_array('video/mp4', $types);
  30. $webp = in_array('image/webp', $types);
  31. $availableAdmins = User::whereIsAdmin(true)->get();
  32. $currentAdmin = config_cache('instance.admin.pid') ? AccountService::get(config_cache('instance.admin.pid'), true) : null;
  33. $openReg = (bool) config_cache('pixelfed.open_registration');
  34. $curOnboarding = (bool) config_cache('instance.curated_registration.enabled');
  35. $regState = $openReg ? 'open' : ($curOnboarding ? 'filtered' : 'closed');
  36. $accountMigration = (bool) config_cache('federation.migration');
  37. return view('admin.settings.home', compact(
  38. 'jpeg',
  39. 'png',
  40. 'gif',
  41. 'mp4',
  42. 'webp',
  43. 'rules',
  44. 'cloud_storage',
  45. 'cloud_disk',
  46. 'cloud_ready',
  47. 'availableAdmins',
  48. 'currentAdmin',
  49. 'regState',
  50. 'accountMigration'
  51. ));
  52. }
  53. public function settingsHomeStore(Request $request)
  54. {
  55. $this->validate($request, [
  56. 'name' => 'nullable|string',
  57. 'short_description' => 'nullable',
  58. 'long_description' => 'nullable',
  59. 'max_photo_size' => 'nullable|integer|min:1',
  60. 'max_album_length' => 'nullable|integer|min:1|max:100',
  61. 'image_quality' => 'nullable|integer|min:1|max:100',
  62. 'type_jpeg' => 'nullable',
  63. 'type_png' => 'nullable',
  64. 'type_gif' => 'nullable',
  65. 'type_mp4' => 'nullable',
  66. 'type_webp' => 'nullable',
  67. 'type_avif' => 'nullable',
  68. 'admin_account_id' => 'nullable',
  69. 'regs' => 'required|in:open,filtered,closed',
  70. 'account_migration' => 'nullable',
  71. 'rule_delete' => 'sometimes',
  72. ]);
  73. $orb = false;
  74. $cob = false;
  75. switch ($request->input('regs')) {
  76. case 'open':
  77. $orb = true;
  78. $cob = false;
  79. break;
  80. case 'filtered':
  81. $orb = false;
  82. $cob = true;
  83. break;
  84. case 'closed':
  85. $orb = false;
  86. $cob = false;
  87. break;
  88. }
  89. ConfigCacheService::put('pixelfed.open_registration', (bool) $orb);
  90. ConfigCacheService::put('instance.curated_registration.enabled', (bool) $cob);
  91. if ($request->filled('admin_account_id')) {
  92. ConfigCacheService::put('instance.admin.pid', $request->admin_account_id);
  93. Cache::forget('api:v1:instance-data:contact');
  94. Cache::forget('api:v1:instance-data-response-v1');
  95. }
  96. if ($request->filled('rule_delete')) {
  97. $index = (int) $request->input('rule_delete');
  98. $rules = ConfigCacheService::get('app.rules');
  99. $json = json_decode($rules, true);
  100. if (! $rules || empty($json)) {
  101. return;
  102. }
  103. unset($json[$index]);
  104. $json = json_encode(array_values($json));
  105. ConfigCacheService::put('app.rules', $json);
  106. Cache::forget('api:v1:instance-data:rules');
  107. Cache::forget('api:v1:instance-data-response-v1');
  108. return 200;
  109. }
  110. $media_types = explode(',', config_cache('pixelfed.media_types'));
  111. $media_types_original = $media_types;
  112. $mimes = [
  113. 'type_jpeg' => 'image/jpeg',
  114. 'type_png' => 'image/png',
  115. 'type_gif' => 'image/gif',
  116. 'type_mp4' => 'video/mp4',
  117. 'type_webp' => 'image/webp',
  118. 'type_avif' => 'image/avif',
  119. ];
  120. foreach ($mimes as $key => $value) {
  121. if ($request->input($key) == 'on') {
  122. if (! in_array($value, $media_types)) {
  123. array_push($media_types, $value);
  124. }
  125. } else {
  126. $media_types = array_diff($media_types, [$value]);
  127. }
  128. }
  129. if ($media_types !== $media_types_original) {
  130. ConfigCacheService::put('pixelfed.media_types', implode(',', array_unique($media_types)));
  131. }
  132. $keys = [
  133. 'name' => 'app.name',
  134. 'short_description' => 'app.short_description',
  135. 'long_description' => 'app.description',
  136. 'max_photo_size' => 'pixelfed.max_photo_size',
  137. 'max_album_length' => 'pixelfed.max_album_length',
  138. 'image_quality' => 'pixelfed.image_quality',
  139. 'account_limit' => 'pixelfed.max_account_size',
  140. 'custom_css' => 'uikit.custom.css',
  141. 'custom_js' => 'uikit.custom.js',
  142. 'about_title' => 'about.title',
  143. ];
  144. foreach ($keys as $key => $value) {
  145. $cc = ConfigCache::whereK($value)->first();
  146. $val = $request->input($key);
  147. if ($cc && $cc->v != $val) {
  148. ConfigCacheService::put($value, $val);
  149. } elseif (! empty($val)) {
  150. ConfigCacheService::put($value, $val);
  151. }
  152. }
  153. $bools = [
  154. 'activitypub' => 'federation.activitypub.enabled',
  155. // 'open_registration' => 'pixelfed.open_registration',
  156. 'mobile_apis' => 'pixelfed.oauth_enabled',
  157. 'stories' => 'instance.stories.enabled',
  158. 'ig_import' => 'pixelfed.import.instagram.enabled',
  159. 'spam_detection' => 'pixelfed.bouncer.enabled',
  160. 'require_email_verification' => 'pixelfed.enforce_email_verification',
  161. 'enforce_account_limit' => 'pixelfed.enforce_account_limit',
  162. 'show_custom_css' => 'uikit.show_custom.css',
  163. 'show_custom_js' => 'uikit.show_custom.js',
  164. 'cloud_storage' => 'pixelfed.cloud_storage',
  165. 'account_autofollow' => 'account.autofollow',
  166. 'show_directory' => 'instance.landing.show_directory',
  167. 'show_explore_feed' => 'instance.landing.show_explore',
  168. 'account_migration' => 'federation.migration',
  169. ];
  170. foreach ($bools as $key => $value) {
  171. $active = $request->input($key) == 'on';
  172. if ($key == 'activitypub' && $active && ! InstanceActor::exists()) {
  173. Artisan::call('instance:actor');
  174. }
  175. if ($key == 'mobile_apis' &&
  176. $active &&
  177. ! file_exists(storage_path('oauth-public.key')) &&
  178. ! config_cache('passport.public_key') &&
  179. ! file_exists(storage_path('oauth-private.key')) &&
  180. ! config_cache('passport.private_key')
  181. ) {
  182. Artisan::call('passport:keys');
  183. Artisan::call('route:cache');
  184. }
  185. if (config_cache($value) !== $active) {
  186. ConfigCacheService::put($value, (bool) $active);
  187. }
  188. }
  189. if ($request->filled('new_rule')) {
  190. $rules = ConfigCacheService::get('app.rules');
  191. $val = $request->input('new_rule');
  192. if (! $rules) {
  193. ConfigCacheService::put('app.rules', json_encode([$val]));
  194. } else {
  195. $json = json_decode($rules, true);
  196. $json[] = $val;
  197. ConfigCacheService::put('app.rules', json_encode(array_values($json)));
  198. }
  199. Cache::forget('api:v1:instance-data:rules');
  200. Cache::forget('api:v1:instance-data-response-v1');
  201. }
  202. if ($request->filled('account_autofollow_usernames')) {
  203. $usernames = explode(',', $request->input('account_autofollow_usernames'));
  204. $names = [];
  205. foreach ($usernames as $n) {
  206. $p = Profile::whereUsername($n)->first();
  207. if (! $p) {
  208. continue;
  209. }
  210. array_push($names, $p->username);
  211. }
  212. ConfigCacheService::put('account.autofollow_usernames', implode(',', $names));
  213. }
  214. Cache::forget(Config::CACHE_KEY);
  215. return redirect('/i/admin/settings')->with('status', 'Successfully updated settings!');
  216. }
  217. public function settingsBackups(Request $request)
  218. {
  219. $path = storage_path('app/'.config('app.name'));
  220. $files = is_dir($path) ? new \DirectoryIterator($path) : [];
  221. return view('admin.settings.backups', compact('files'));
  222. }
  223. public function settingsMaintenance(Request $request)
  224. {
  225. return view('admin.settings.maintenance');
  226. }
  227. public function settingsStorage(Request $request)
  228. {
  229. $storage = [];
  230. return view('admin.settings.storage', compact('storage'));
  231. }
  232. public function settingsFeatures(Request $request)
  233. {
  234. return view('admin.settings.features');
  235. }
  236. public function settingsPages(Request $request)
  237. {
  238. $pages = Page::orderByDesc('updated_at')->paginate(10);
  239. return view('admin.pages.home', compact('pages'));
  240. }
  241. public function settingsPageEdit(Request $request)
  242. {
  243. return view('admin.pages.edit');
  244. }
  245. public function settingsSystem(Request $request)
  246. {
  247. $sys = [
  248. 'pixelfed' => config('pixelfed.version'),
  249. 'php' => phpversion(),
  250. 'laravel' => app()->version(),
  251. ];
  252. switch (config('database.default')) {
  253. case 'pgsql':
  254. $exp = DB::raw('select version();');
  255. $expQuery = $exp->getValue(DB::connection()->getQueryGrammar());
  256. $sys['database'] = [
  257. 'name' => 'Postgres',
  258. 'version' => explode(' ', DB::select($expQuery)[0]->version)[1],
  259. ];
  260. break;
  261. case 'mysql':
  262. $exp = DB::raw('select version()');
  263. $expQuery = $exp->getValue(DB::connection()->getQueryGrammar());
  264. $sys['database'] = [
  265. 'name' => 'MySQL',
  266. 'version' => DB::select($expQuery)[0]->{'version()'},
  267. ];
  268. break;
  269. default:
  270. $sys['database'] = [
  271. 'name' => 'Unknown',
  272. 'version' => '?',
  273. ];
  274. break;
  275. }
  276. return view('admin.settings.system', compact('sys'));
  277. }
  278. public function settingsApiFetch(Request $request)
  279. {
  280. $cloud_storage = ConfigCacheService::get('pixelfed.cloud_storage');
  281. $cloud_disk = config('filesystems.cloud');
  282. $cloud_ready = ! empty(config('filesystems.disks.'.$cloud_disk.'.key')) && ! empty(config('filesystems.disks.'.$cloud_disk.'.secret'));
  283. $types = explode(',', ConfigCacheService::get('pixelfed.media_types'));
  284. $rules = ConfigCacheService::get('app.rules') ? json_decode(ConfigCacheService::get('app.rules'), true) : [];
  285. $jpeg = in_array('image/jpg', $types) || in_array('image/jpeg', $types);
  286. $png = in_array('image/png', $types);
  287. $gif = in_array('image/gif', $types);
  288. $mp4 = in_array('video/mp4', $types);
  289. $webp = in_array('image/webp', $types);
  290. $availableAdmins = User::whereIsAdmin(true)->get();
  291. $currentAdmin = config_cache('instance.admin.pid') ? AccountService::get(config_cache('instance.admin.pid'), true) : null;
  292. $openReg = (bool) config_cache('pixelfed.open_registration');
  293. $curOnboarding = (bool) config_cache('instance.curated_registration.enabled');
  294. $regState = $openReg ? 'open' : ($curOnboarding ? 'filtered' : 'closed');
  295. $accountMigration = (bool) config_cache('federation.migration');
  296. $autoFollow = config_cache('account.autofollow_usernames');
  297. if (strlen($autoFollow) > 3) {
  298. $autoFollow = explode(',', $autoFollow);
  299. }
  300. $res = AdminSettingsService::getAll();
  301. return response()->json($res);
  302. }
  303. public function settingsApiRulesAdd(Request $request)
  304. {
  305. $this->validate($request, [
  306. 'rule' => 'required|string|min:5|max:1000',
  307. ]);
  308. $rules = ConfigCacheService::get('app.rules');
  309. $val = $request->input('rule');
  310. if (! $rules) {
  311. ConfigCacheService::put('app.rules', json_encode([$val]));
  312. } else {
  313. $json = json_decode($rules, true);
  314. $count = count($json);
  315. if ($count >= 30) {
  316. return response()->json(['message' => 'Max rules limit reached, you can set up to 30 rules at a time.'], 400);
  317. }
  318. $json[] = $val;
  319. ConfigCacheService::put('app.rules', json_encode(array_values($json)));
  320. }
  321. Cache::forget('api:v1:instance-data:rules');
  322. Cache::forget('api:v1:instance-data-response-v1');
  323. Cache::forget('api:v2:instance-data-response-v2');
  324. Config::refresh();
  325. return [$val];
  326. }
  327. public function settingsApiRulesDelete(Request $request)
  328. {
  329. $this->validate($request, [
  330. 'rule' => 'required|string',
  331. ]);
  332. $rules = ConfigCacheService::get('app.rules');
  333. $val = $request->input('rule');
  334. if (! $rules) {
  335. return [];
  336. } else {
  337. $json = json_decode($rules, true);
  338. $idx = array_search($val, $json);
  339. if ($idx !== false) {
  340. unset($json[$idx]);
  341. $json = array_values($json);
  342. }
  343. ConfigCacheService::put('app.rules', json_encode(array_values($json)));
  344. }
  345. Cache::forget('api:v1:instance-data:rules');
  346. Cache::forget('api:v1:instance-data-response-v1');
  347. Cache::forget('api:v2:instance-data-response-v2');
  348. Config::refresh();
  349. return response()->json($json);
  350. }
  351. public function settingsApiRulesDeleteAll(Request $request)
  352. {
  353. $rules = ConfigCacheService::get('app.rules');
  354. if (! $rules) {
  355. return [];
  356. } else {
  357. ConfigCacheService::put('app.rules', json_encode([]));
  358. }
  359. Cache::forget('api:v1:instance-data:rules');
  360. Cache::forget('api:v1:instance-data-response-v1');
  361. Cache::forget('api:v2:instance-data-response-v2');
  362. Config::refresh();
  363. return response()->json([]);
  364. }
  365. public function settingsApiAutofollowDelete(Request $request)
  366. {
  367. $this->validate($request, [
  368. 'username' => 'required|string',
  369. ]);
  370. $username = $request->input('username');
  371. $names = [];
  372. $existing = config_cache('account.autofollow_usernames');
  373. if ($existing) {
  374. $names = explode(',', $existing);
  375. }
  376. if (in_array($username, $names)) {
  377. $key = array_search($username, $names);
  378. if ($key !== false) {
  379. unset($names[$key]);
  380. }
  381. }
  382. ConfigCacheService::put('account.autofollow_usernames', implode(',', $names));
  383. return response()->json(['accounts' => array_values($names)]);
  384. }
  385. public function settingsApiAutofollowAdd(Request $request)
  386. {
  387. $this->validate($request, [
  388. 'username' => 'required|string',
  389. ]);
  390. $username = $request->input('username');
  391. $names = [];
  392. $existing = config_cache('account.autofollow_usernames');
  393. if ($existing) {
  394. $names = explode(',', $existing);
  395. }
  396. if ($existing && count($names)) {
  397. if (count($names) >= 5) {
  398. return response()->json(['message' => 'You can only add up to 5 accounts to be autofollowed.'], 400);
  399. }
  400. if (in_array(strtolower($username), array_map('strtolower', $names))) {
  401. return response()->json(['message' => 'User already exists, please try again.'], 400);
  402. }
  403. }
  404. $p = User::whereUsername($username)->whereNull('status')->first();
  405. if (! $p || in_array($p->username, $names)) {
  406. abort(404);
  407. }
  408. array_push($names, $p->username);
  409. ConfigCacheService::put('account.autofollow_usernames', implode(',', $names));
  410. return response()->json(['accounts' => array_values($names)]);
  411. }
  412. public function settingsApiUpdateType(Request $request, $type)
  413. {
  414. abort_unless(in_array($type, [
  415. 'posts',
  416. 'platform',
  417. 'home',
  418. 'landing',
  419. 'branding',
  420. 'media',
  421. 'users',
  422. 'storage',
  423. ]), 400);
  424. switch ($type) {
  425. case 'home':
  426. return $this->settingsApiUpdateHomeType($request);
  427. break;
  428. case 'landing':
  429. return $this->settingsApiUpdateLandingType($request);
  430. break;
  431. case 'posts':
  432. return $this->settingsApiUpdatePostsType($request);
  433. break;
  434. case 'platform':
  435. return $this->settingsApiUpdatePlatformType($request);
  436. break;
  437. case 'branding':
  438. return $this->settingsApiUpdateBrandingType($request);
  439. break;
  440. case 'media':
  441. return $this->settingsApiUpdateMediaType($request);
  442. break;
  443. case 'users':
  444. return $this->settingsApiUpdateUsersType($request);
  445. break;
  446. case 'storage':
  447. return $this->settingsApiUpdateStorageType($request);
  448. break;
  449. default:
  450. abort(404);
  451. break;
  452. }
  453. }
  454. public function settingsApiUpdateHomeType($request)
  455. {
  456. $this->validate($request, [
  457. 'registration_status' => 'required|in:open,filtered,closed',
  458. 'cloud_storage' => 'required',
  459. 'activitypub_enabled' => 'required',
  460. 'authorized_fetch' => 'required',
  461. 'account_migration' => 'required',
  462. 'mobile_apis' => 'required',
  463. 'stories' => 'required',
  464. 'instagram_import' => 'required',
  465. 'autospam_enabled' => 'required',
  466. ]);
  467. $regStatus = $request->input('registration_status');
  468. ConfigCacheService::put('pixelfed.open_registration', $regStatus === 'open');
  469. ConfigCacheService::put('instance.curated_registration.enabled', $regStatus === 'filtered');
  470. $cloudStorage = $request->boolean('cloud_storage');
  471. if ($cloudStorage !== (bool) config_cache('pixelfed.cloud_storage')) {
  472. if (! $cloudStorage) {
  473. ConfigCacheService::put('pixelfed.cloud_storage', false);
  474. } else {
  475. $cloud_disk = config('filesystems.cloud');
  476. $cloud_ready = ! empty(config('filesystems.disks.'.$cloud_disk.'.key')) && ! empty(config('filesystems.disks.'.$cloud_disk.'.secret'));
  477. if (! $cloud_ready) {
  478. return redirect()->back()->withErrors(['cloud_storage' => 'Must configure cloud storage before enabling!']);
  479. } else {
  480. ConfigCacheService::put('pixelfed.cloud_storage', true);
  481. }
  482. }
  483. }
  484. ConfigCacheService::put('federation.activitypub.authorized_fetch', $request->boolean('authorized_fetch'));
  485. ConfigCacheService::put('federation.activitypub.enabled', $request->boolean('activitypub_enabled'));
  486. ConfigCacheService::put('federation.migration', $request->boolean('account_migration'));
  487. ConfigCacheService::put('pixelfed.oauth_enabled', $request->boolean('mobile_apis'));
  488. ConfigCacheService::put('instance.stories.enabled', $request->boolean('stories'));
  489. ConfigCacheService::put('pixelfed.import.instagram.enabled', $request->boolean('instagram_import'));
  490. ConfigCacheService::put('pixelfed.bouncer.enabled', $request->boolean('autospam_enabled'));
  491. Cache::forget('api:v1:instance-data-response-v1');
  492. Cache::forget('api:v2:instance-data-response-v2');
  493. Cache::forget('api:v1:instance-data:contact');
  494. Config::refresh();
  495. return $request->all();
  496. }
  497. public function settingsApiUpdateLandingType($request)
  498. {
  499. $this->validate($request, [
  500. 'current_admin' => 'required',
  501. 'show_directory' => 'required',
  502. 'show_explore' => 'required',
  503. ]);
  504. ConfigCacheService::put('instance.admin.pid', $request->input('current_admin'));
  505. ConfigCacheService::put('instance.landing.show_directory', $request->boolean('show_directory'));
  506. ConfigCacheService::put('instance.landing.show_explore', $request->boolean('show_explore'));
  507. Cache::forget('api:v1:instance-data:rules');
  508. Cache::forget('api:v1:instance-data-response-v1');
  509. Cache::forget('api:v2:instance-data-response-v2');
  510. Cache::forget('api:v1:instance-data:contact');
  511. Config::refresh();
  512. return $request->all();
  513. }
  514. public function settingsApiUpdateMediaType($request)
  515. {
  516. $this->validate($request, [
  517. 'image_quality' => 'required|integer|min:1|max:100',
  518. 'max_album_length' => 'required|integer|min:1|max:20',
  519. 'max_photo_size' => 'required|integer|min:100|max:1000000',
  520. 'media_types' => 'required',
  521. 'optimize_image' => 'required',
  522. 'optimize_video' => 'required',
  523. ]);
  524. $mediaTypes = $request->input('media_types');
  525. $mediaArray = explode(',', $mediaTypes);
  526. foreach ($mediaArray as $mediaType) {
  527. if (! in_array($mediaType, ['image/jpeg', 'image/png', 'image/gif', 'image/webp', 'video/mp4', 'image/avif'])) {
  528. return redirect()->back()->withErrors(['media_types' => 'Invalid media type']);
  529. }
  530. }
  531. ConfigCacheService::put('pixelfed.media_types', $request->input('media_types'));
  532. ConfigCacheService::put('pixelfed.image_quality', $request->input('image_quality'));
  533. ConfigCacheService::put('pixelfed.max_album_length', $request->input('max_album_length'));
  534. ConfigCacheService::put('pixelfed.max_photo_size', $request->input('max_photo_size'));
  535. ConfigCacheService::put('pixelfed.optimize_image', $request->boolean('optimize_image'));
  536. ConfigCacheService::put('pixelfed.optimize_video', $request->boolean('optimize_video'));
  537. Cache::forget('api:v1:instance-data:rules');
  538. Cache::forget('api:v1:instance-data-response-v1');
  539. Cache::forget('api:v2:instance-data-response-v2');
  540. Cache::forget('api:v1:instance-data:contact');
  541. Config::refresh();
  542. return $request->all();
  543. }
  544. public function settingsApiUpdateBrandingType($request)
  545. {
  546. $this->validate($request, [
  547. 'name' => 'required',
  548. 'short_description' => 'required',
  549. 'long_description' => 'required',
  550. ]);
  551. ConfigCacheService::put('app.name', $request->input('name'));
  552. ConfigCacheService::put('app.short_description', $request->input('short_description'));
  553. ConfigCacheService::put('app.description', $request->input('long_description'));
  554. Cache::forget('api:v1:instance-data:rules');
  555. Cache::forget('api:v1:instance-data-response-v1');
  556. Cache::forget('api:v2:instance-data-response-v2');
  557. Cache::forget('api:v1:instance-data:contact');
  558. Config::refresh();
  559. return $request->all();
  560. }
  561. public function settingsApiUpdatePostsType($request)
  562. {
  563. $this->validate($request, [
  564. 'max_caption_length' => 'required|integer|min:5|max:10000',
  565. 'max_altext_length' => 'required|integer|min:5|max:40000',
  566. ]);
  567. ConfigCacheService::put('pixelfed.max_caption_length', $request->input('max_caption_length'));
  568. ConfigCacheService::put('pixelfed.max_altext_length', $request->input('max_altext_length'));
  569. $res = [
  570. 'max_caption_length' => $request->input('max_caption_length'),
  571. 'max_altext_length' => $request->input('max_altext_length'),
  572. ];
  573. Cache::forget('api:v1:instance-data:rules');
  574. Cache::forget('api:v1:instance-data-response-v1');
  575. Cache::forget('api:v2:instance-data-response-v2');
  576. Config::refresh();
  577. return $res;
  578. }
  579. public function settingsApiUpdatePlatformType($request)
  580. {
  581. $this->validate($request, [
  582. 'allow_app_registration' => 'required',
  583. 'app_registration_rate_limit_attempts' => 'required|integer|min:1',
  584. 'app_registration_rate_limit_decay' => 'required|integer|min:1',
  585. 'app_registration_confirm_rate_limit_attempts' => 'required|integer|min:1',
  586. 'app_registration_confirm_rate_limit_decay' => 'required|integer|min:1',
  587. 'allow_post_embeds' => 'required',
  588. 'allow_profile_embeds' => 'required',
  589. 'captcha_enabled' => 'required',
  590. 'captcha_on_login' => 'required_if_accepted:captcha_enabled',
  591. 'captcha_on_register' => 'required_if_accepted:captcha_enabled',
  592. 'captcha_secret' => 'required_if_accepted:captcha_enabled',
  593. 'captcha_sitekey' => 'required_if_accepted:captcha_enabled',
  594. 'custom_emoji_enabled' => 'required',
  595. ]);
  596. ConfigCacheService::put('pixelfed.allow_app_registration', $request->boolean('allow_app_registration'));
  597. ConfigCacheService::put('pixelfed.app_registration_rate_limit_attempts', $request->input('app_registration_rate_limit_attempts'));
  598. ConfigCacheService::put('pixelfed.app_registration_rate_limit_decay', $request->input('app_registration_rate_limit_decay'));
  599. ConfigCacheService::put('pixelfed.app_registration_confirm_rate_limit_attempts', $request->input('app_registration_confirm_rate_limit_attempts'));
  600. ConfigCacheService::put('pixelfed.app_registration_confirm_rate_limit_decay', $request->input('app_registration_confirm_rate_limit_decay'));
  601. ConfigCacheService::put('instance.embed.post', $request->boolean('allow_post_embeds'));
  602. ConfigCacheService::put('instance.embed.profile', $request->boolean('allow_profile_embeds'));
  603. ConfigCacheService::put('federation.custom_emoji.enabled', $request->boolean('custom_emoji_enabled'));
  604. $captcha = $request->boolean('captcha_enabled');
  605. if ($captcha) {
  606. $secret = $request->input('captcha_secret');
  607. $sitekey = $request->input('captcha_sitekey');
  608. if (config_cache('captcha.secret') != $secret && strpos($secret, '*') === false) {
  609. ConfigCacheService::put('captcha.secret', $secret);
  610. }
  611. if (config_cache('captcha.sitekey') != $sitekey && strpos($sitekey, '*') === false) {
  612. ConfigCacheService::put('captcha.sitekey', $sitekey);
  613. }
  614. ConfigCacheService::put('captcha.active.login', $request->boolean('captcha_on_login'));
  615. ConfigCacheService::put('captcha.active.register', $request->boolean('captcha_on_register'));
  616. ConfigCacheService::put('captcha.triggers.login.enabled', $request->boolean('captcha_on_login'));
  617. ConfigCacheService::put('captcha.enabled', true);
  618. } else {
  619. ConfigCacheService::put('captcha.enabled', false);
  620. }
  621. $res = [
  622. 'allow_app_registration' => $request->boolean('allow_app_registration'),
  623. 'app_registration_rate_limit_attempts' => $request->input('app_registration_rate_limit_attempts'),
  624. 'app_registration_rate_limit_decay' => $request->input('app_registration_rate_limit_decay'),
  625. 'app_registration_confirm_rate_limit_attempts' => $request->input('app_registration_confirm_rate_limit_attempts'),
  626. 'app_registration_confirm_rate_limit_decay' => $request->input('app_registration_confirm_rate_limit_decay'),
  627. 'allow_post_embeds' => $request->boolean('allow_post_embeds'),
  628. 'allow_profile_embeds' => $request->boolean('allow_profile_embeds'),
  629. 'captcha_enabled' => $request->boolean('captcha_enabled'),
  630. 'captcha_on_login' => $request->boolean('captcha_on_login'),
  631. 'captcha_on_register' => $request->boolean('captcha_on_register'),
  632. 'captcha_secret' => $request->input('captcha_secret'),
  633. 'captcha_sitekey' => $request->input('captcha_sitekey'),
  634. 'custom_emoji_enabled' => $request->boolean('custom_emoji_enabled'),
  635. ];
  636. Cache::forget('api:v1:instance-data:rules');
  637. Cache::forget('api:v1:instance-data-response-v1');
  638. Cache::forget('api:v2:instance-data-response-v2');
  639. Config::refresh();
  640. return $res;
  641. }
  642. public function settingsApiUpdateUsersType($request)
  643. {
  644. $this->validate($request, [
  645. 'require_email_verification' => 'required',
  646. 'enforce_account_limit' => 'required',
  647. 'max_account_size' => 'required|integer|min:50000',
  648. 'admin_autofollow' => 'required',
  649. 'admin_autofollow_accounts' => 'sometimes',
  650. 'max_user_blocks' => 'required|integer|min:0|max:5000',
  651. 'max_user_mutes' => 'required|integer|min:0|max:5000',
  652. 'max_domain_blocks' => 'required|integer|min:0|max:5000',
  653. ]);
  654. $adminAutofollow = $request->boolean('admin_autofollow');
  655. $adminAutofollowAccounts = $request->input('admin_autofollow_accounts');
  656. if ($adminAutofollow) {
  657. if ($request->filled('admin_autofollow_accounts')) {
  658. $names = [];
  659. $existing = config_cache('account.autofollow_usernames');
  660. if ($existing) {
  661. $names = explode(',', $existing);
  662. foreach (array_map('strtolower', $adminAutofollowAccounts) as $afc) {
  663. if (in_array(strtolower($afc), array_map('strtolower', $names))) {
  664. continue;
  665. }
  666. $names[] = $afc;
  667. }
  668. } else {
  669. $names = $adminAutofollowAccounts;
  670. }
  671. if (! $names || count($names) == 0) {
  672. return response()->json(['message' => 'You need to assign autofollow accounts before you can enable it.'], 400);
  673. }
  674. if (count($names) > 5) {
  675. return response()->json(['message' => 'You can only add up to 5 accounts to be autofollowed.'.json_encode($names)], 400);
  676. }
  677. $autofollows = User::whereIn('username', $names)->whereNull('status')->pluck('username');
  678. $adminAutofollowAccounts = $autofollows->implode(',');
  679. ConfigCacheService::put('account.autofollow_usernames', $adminAutofollowAccounts);
  680. } else {
  681. return response()->json(['message' => 'You need to assign autofollow accounts before you can enable it.'], 400);
  682. }
  683. }
  684. ConfigCacheService::put('pixelfed.enforce_email_verification', $request->boolean('require_email_verification'));
  685. ConfigCacheService::put('pixelfed.enforce_account_limit', $request->boolean('enforce_account_limit'));
  686. ConfigCacheService::put('pixelfed.max_account_size', $request->input('max_account_size'));
  687. ConfigCacheService::put('account.autofollow', $request->boolean('admin_autofollow'));
  688. ConfigCacheService::put('instance.user_filters.max_user_blocks', (int) $request->input('max_user_blocks'));
  689. ConfigCacheService::put('instance.user_filters.max_user_mutes', (int) $request->input('max_user_mutes'));
  690. ConfigCacheService::put('instance.user_filters.max_domain_blocks', (int) $request->input('max_domain_blocks'));
  691. $res = [
  692. 'require_email_verification' => $request->boolean('require_email_verification'),
  693. 'enforce_account_limit' => $request->boolean('enforce_account_limit'),
  694. 'admin_autofollow' => $request->boolean('admin_autofollow'),
  695. 'admin_autofollow_accounts' => $adminAutofollowAccounts,
  696. 'max_user_blocks' => $request->input('max_user_blocks'),
  697. 'max_user_mutes' => $request->input('max_user_mutes'),
  698. 'max_domain_blocks' => $request->input('max_domain_blocks'),
  699. ];
  700. Cache::forget('api:v1:instance-data:rules');
  701. Cache::forget('api:v1:instance-data-response-v1');
  702. Cache::forget('api:v2:instance-data-response-v2');
  703. Config::refresh();
  704. return $res;
  705. }
  706. public function settingsApiUpdateStorageType($request)
  707. {
  708. $this->validate($request, [
  709. 'primary_disk' => 'required|in:local,cloud',
  710. 'update_disk' => 'sometimes',
  711. 'disk_config' => 'required_if_accepted:update_disk',
  712. 'disk_config.driver' => 'required|in:s3,spaces',
  713. 'disk_config.key' => 'required',
  714. 'disk_config.secret' => 'required',
  715. 'disk_config.region' => 'required',
  716. 'disk_config.bucket' => 'required',
  717. 'disk_config.visibility' => 'required',
  718. 'disk_config.endpoint' => 'required',
  719. 'disk_config.url' => 'nullable',
  720. ]);
  721. ConfigCacheService::put('pixelfed.cloud_storage', $request->input('primary_disk') === 'cloud');
  722. $res = [
  723. 'primary_disk' => $request->input('primary_disk'),
  724. ];
  725. if ($request->has('update_disk')) {
  726. $res['disk_config'] = $request->input('disk_config');
  727. $changes = [];
  728. $dkey = $request->input('disk_config.driver') === 's3' ? 'filesystems.disks.s3.' : 'filesystems.disks.spaces.';
  729. $key = $request->input('disk_config.key');
  730. $ckey = null;
  731. $secret = $request->input('disk_config.secret');
  732. $csecret = null;
  733. $region = $request->input('disk_config.region');
  734. $bucket = $request->input('disk_config.bucket');
  735. $visibility = $request->input('disk_config.visibility');
  736. $url = $request->input('disk_config.url');
  737. $endpoint = $request->input('disk_config.endpoint');
  738. if (strpos($key, '*') === false && $key != config_cache($dkey.'key')) {
  739. array_push($changes, 'key');
  740. } else {
  741. $ckey = config_cache($dkey.'key');
  742. }
  743. if (strpos($secret, '*') === false && $secret != config_cache($dkey.'secret')) {
  744. array_push($changes, 'secret');
  745. } else {
  746. $csecret = config_cache($dkey.'secret');
  747. }
  748. if ($region != config_cache($dkey.'region')) {
  749. array_push($changes, 'region');
  750. }
  751. if ($bucket != config_cache($dkey.'bucket')) {
  752. array_push($changes, 'bucket');
  753. }
  754. if ($visibility != config_cache($dkey.'visibility')) {
  755. array_push($changes, 'visibility');
  756. }
  757. if ($url != config_cache($dkey.'url')) {
  758. array_push($changes, 'url');
  759. }
  760. if ($endpoint != config_cache($dkey.'endpoint')) {
  761. array_push($changes, 'endpoint');
  762. }
  763. if ($changes && count($changes)) {
  764. $isValid = FilesystemService::getVerifyCredentials(
  765. $ckey ?? $key,
  766. $csecret ?? $secret,
  767. $region,
  768. $bucket,
  769. $endpoint,
  770. );
  771. if (! $isValid) {
  772. 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);
  773. }
  774. ConfigCacheService::put($dkey . 'key', $key);
  775. ConfigCacheService::put($dkey . 'secret', $secret);
  776. ConfigCacheService::put($dkey . 'region', $region);
  777. ConfigCacheService::put($dkey . 'bucket', $bucket);
  778. ConfigCacheService::put($dkey . 'endpoint', $endpoint);
  779. ConfigCacheService::put($dkey . 'url', $url);
  780. }
  781. $res['changes'] = json_encode($changes);
  782. }
  783. Cache::forget('api:v1:instance-data:rules');
  784. Cache::forget('api:v1:instance-data-response-v1');
  785. Cache::forget('api:v2:instance-data-response-v2');
  786. Config::refresh();
  787. return $res;
  788. }
  789. }