AdminSettingsController.php 36 KB

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