AdminSettingsController.php 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285
  1. <?php
  2. namespace App\Http\Controllers\Admin;
  3. use Artisan, Cache, DB;
  4. use Illuminate\Http\Request;
  5. use Carbon\Carbon;
  6. use App\{Comment, Like, Media, Page, Profile, Report, Status, User};
  7. use App\Models\InstanceActor;
  8. use App\Http\Controllers\Controller;
  9. use App\Util\Lexer\PrettyNumber;
  10. use App\Models\ConfigCache;
  11. use App\Services\AccountService;
  12. use App\Services\ConfigCacheService;
  13. use App\Util\Site\Config;
  14. use Illuminate\Support\Str;
  15. trait AdminSettingsController
  16. {
  17. public function settings(Request $request)
  18. {
  19. $cloud_storage = ConfigCacheService::get('pixelfed.cloud_storage');
  20. $cloud_disk = config('filesystems.cloud');
  21. $cloud_ready = !empty(config('filesystems.disks.' . $cloud_disk . '.key')) && !empty(config('filesystems.disks.' . $cloud_disk . '.secret'));
  22. $types = explode(',', ConfigCacheService::get('pixelfed.media_types'));
  23. $rules = ConfigCacheService::get('app.rules') ? json_decode(ConfigCacheService::get('app.rules'), true) : null;
  24. $jpeg = in_array('image/jpg', $types) || in_array('image/jpeg', $types);
  25. $png = in_array('image/png', $types);
  26. $gif = in_array('image/gif', $types);
  27. $mp4 = in_array('video/mp4', $types);
  28. $webp = in_array('image/webp', $types);
  29. $availableAdmins = User::whereIsAdmin(true)->get();
  30. $currentAdmin = config_cache('instance.admin.pid') ? AccountService::get(config_cache('instance.admin.pid'), true) : null;
  31. // $system = [
  32. // 'permissions' => is_writable(base_path('storage')) && is_writable(base_path('bootstrap')),
  33. // 'max_upload_size' => ini_get('post_max_size'),
  34. // 'image_driver' => config('image.driver'),
  35. // 'image_driver_loaded' => extension_loaded(config('image.driver'))
  36. // ];
  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. // 'system'
  50. ));
  51. }
  52. public function settingsHomeStore(Request $request)
  53. {
  54. $this->validate($request, [
  55. 'name' => 'nullable|string',
  56. 'short_description' => 'nullable',
  57. 'long_description' => 'nullable',
  58. 'max_photo_size' => 'nullable|integer|min:1',
  59. 'max_album_length' => 'nullable|integer|min:1|max:100',
  60. 'image_quality' => 'nullable|integer|min:1|max:100',
  61. 'type_jpeg' => 'nullable',
  62. 'type_png' => 'nullable',
  63. 'type_gif' => 'nullable',
  64. 'type_mp4' => 'nullable',
  65. 'type_webp' => 'nullable',
  66. 'admin_account_id' => 'nullable',
  67. ]);
  68. if($request->filled('admin_account_id')) {
  69. ConfigCacheService::put('instance.admin.pid', $request->admin_account_id);
  70. Cache::forget('api:v1:instance-data:contact');
  71. Cache::forget('api:v1:instance-data-response-v1');
  72. }
  73. if($request->filled('rule_delete')) {
  74. $index = (int) $request->input('rule_delete');
  75. $rules = ConfigCacheService::get('app.rules');
  76. $json = json_decode($rules, true);
  77. if(!$rules || empty($json)) {
  78. return;
  79. }
  80. unset($json[$index]);
  81. $json = json_encode(array_values($json));
  82. ConfigCacheService::put('app.rules', $json);
  83. Cache::forget('api:v1:instance-data:rules');
  84. Cache::forget('api:v1:instance-data-response-v1');
  85. return 200;
  86. }
  87. $media_types = explode(',', config_cache('pixelfed.media_types'));
  88. $media_types_original = $media_types;
  89. $mimes = [
  90. 'type_jpeg' => 'image/jpeg',
  91. 'type_png' => 'image/png',
  92. 'type_gif' => 'image/gif',
  93. 'type_mp4' => 'video/mp4',
  94. 'type_webp' => 'image/webp',
  95. ];
  96. foreach ($mimes as $key => $value) {
  97. if($request->input($key) == 'on') {
  98. if(!in_array($value, $media_types)) {
  99. array_push($media_types, $value);
  100. }
  101. } else {
  102. $media_types = array_diff($media_types, [$value]);
  103. }
  104. }
  105. if($media_types !== $media_types_original) {
  106. ConfigCacheService::put('pixelfed.media_types', implode(',', array_unique($media_types)));
  107. }
  108. $keys = [
  109. 'name' => 'app.name',
  110. 'short_description' => 'app.short_description',
  111. 'long_description' => 'app.description',
  112. 'max_photo_size' => 'pixelfed.max_photo_size',
  113. 'max_album_length' => 'pixelfed.max_album_length',
  114. 'image_quality' => 'pixelfed.image_quality',
  115. 'account_limit' => 'pixelfed.max_account_size',
  116. 'custom_css' => 'uikit.custom.css',
  117. 'custom_js' => 'uikit.custom.js',
  118. 'about_title' => 'about.title'
  119. ];
  120. foreach ($keys as $key => $value) {
  121. $cc = ConfigCache::whereK($value)->first();
  122. $val = $request->input($key);
  123. if($cc && $cc->v != $val) {
  124. ConfigCacheService::put($value, $val);
  125. } else if(!empty($val)) {
  126. ConfigCacheService::put($value, $val);
  127. }
  128. }
  129. $bools = [
  130. 'activitypub' => 'federation.activitypub.enabled',
  131. 'open_registration' => 'pixelfed.open_registration',
  132. 'mobile_apis' => 'pixelfed.oauth_enabled',
  133. 'stories' => 'instance.stories.enabled',
  134. 'ig_import' => 'pixelfed.import.instagram.enabled',
  135. 'spam_detection' => 'pixelfed.bouncer.enabled',
  136. 'require_email_verification' => 'pixelfed.enforce_email_verification',
  137. 'enforce_account_limit' => 'pixelfed.enforce_account_limit',
  138. 'show_custom_css' => 'uikit.show_custom.css',
  139. 'show_custom_js' => 'uikit.show_custom.js',
  140. 'cloud_storage' => 'pixelfed.cloud_storage',
  141. 'account_autofollow' => 'account.autofollow',
  142. 'show_directory' => 'instance.landing.show_directory',
  143. 'show_explore_feed' => 'instance.landing.show_explore',
  144. ];
  145. foreach ($bools as $key => $value) {
  146. $active = $request->input($key) == 'on';
  147. if($key == 'activitypub' && $active && !InstanceActor::exists()) {
  148. Artisan::call('instance:actor');
  149. }
  150. if( $key == 'mobile_apis' &&
  151. $active &&
  152. !file_exists(storage_path('oauth-public.key')) &&
  153. !file_exists(storage_path('oauth-private.key'))
  154. ) {
  155. Artisan::call('passport:keys');
  156. Artisan::call('route:cache');
  157. }
  158. if(config_cache($value) !== $active) {
  159. ConfigCacheService::put($value, (bool) $active);
  160. }
  161. }
  162. if($request->filled('new_rule')) {
  163. $rules = ConfigCacheService::get('app.rules');
  164. $val = $request->input('new_rule');
  165. if(!$rules) {
  166. ConfigCacheService::put('app.rules', json_encode([$val]));
  167. } else {
  168. $json = json_decode($rules, true);
  169. $json[] = $val;
  170. ConfigCacheService::put('app.rules', json_encode(array_values($json)));
  171. }
  172. Cache::forget('api:v1:instance-data:rules');
  173. Cache::forget('api:v1:instance-data-response-v1');
  174. }
  175. if($request->filled('account_autofollow_usernames')) {
  176. $usernames = explode(',', $request->input('account_autofollow_usernames'));
  177. $names = [];
  178. foreach($usernames as $n) {
  179. $p = Profile::whereUsername($n)->first();
  180. if(!$p) {
  181. continue;
  182. }
  183. array_push($names, $p->username);
  184. }
  185. ConfigCacheService::put('account.autofollow_usernames', implode(',', $names));
  186. }
  187. Cache::forget(Config::CACHE_KEY);
  188. return redirect('/i/admin/settings')->with('status', 'Successfully updated settings!');
  189. }
  190. public function settingsBackups(Request $request)
  191. {
  192. $path = storage_path('app/'.config('app.name'));
  193. $files = is_dir($path) ? new \DirectoryIterator($path) : [];
  194. return view('admin.settings.backups', compact('files'));
  195. }
  196. public function settingsMaintenance(Request $request)
  197. {
  198. return view('admin.settings.maintenance');
  199. }
  200. public function settingsStorage(Request $request)
  201. {
  202. $storage = [];
  203. return view('admin.settings.storage', compact('storage'));
  204. }
  205. public function settingsFeatures(Request $request)
  206. {
  207. return view('admin.settings.features');
  208. }
  209. public function settingsPages(Request $request)
  210. {
  211. $pages = Page::orderByDesc('updated_at')->paginate(10);
  212. return view('admin.pages.home', compact('pages'));
  213. }
  214. public function settingsPageEdit(Request $request)
  215. {
  216. return view('admin.pages.edit');
  217. }
  218. public function settingsSystem(Request $request)
  219. {
  220. $sys = [
  221. 'pixelfed' => config('pixelfed.version'),
  222. 'php' => phpversion(),
  223. 'laravel' => app()->version(),
  224. ];
  225. switch (config('database.default')) {
  226. case 'pgsql':
  227. $exp = DB::raw('select version();');
  228. $expQuery = $exp->getValue(DB::connection()->getQueryGrammar());
  229. $sys['database'] = [
  230. 'name' => 'Postgres',
  231. 'version' => explode(' ', DB::select($expQuery)[0]->version)[1]
  232. ];
  233. break;
  234. case 'mysql':
  235. $exp = DB::raw('select version()');
  236. $expQuery = $exp->getValue(DB::connection()->getQueryGrammar());
  237. $sys['database'] = [
  238. 'name' => 'MySQL',
  239. 'version' => DB::select($expQuery)[0]->{'version()'}
  240. ];
  241. break;
  242. default:
  243. $sys['database'] = [
  244. 'name' => 'Unknown',
  245. 'version' => '?'
  246. ];
  247. break;
  248. }
  249. return view('admin.settings.system', compact('sys'));
  250. }
  251. }