AdminApiController.php 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818
  1. <?php
  2. namespace App\Http\Controllers\Api;
  3. use Illuminate\Http\Request;
  4. use App\Http\Controllers\Controller;
  5. use App\Jobs\StatusPipeline\StatusDelete;
  6. use Auth, Cache, DB;
  7. use Carbon\Carbon;
  8. use App\{
  9. AccountInterstitial,
  10. Instance,
  11. Like,
  12. Notification,
  13. Media,
  14. Profile,
  15. Report,
  16. Status,
  17. User
  18. };
  19. use App\Models\Conversation;
  20. use App\Models\RemoteReport;
  21. use App\Services\AccountService;
  22. use App\Services\AdminStatsService;
  23. use App\Services\ConfigCacheService;
  24. use App\Services\InstanceService;
  25. use App\Services\ModLogService;
  26. use App\Services\SnowflakeService;
  27. use App\Services\StatusService;
  28. use App\Services\PublicTimelineService;
  29. use App\Services\NetworkTimelineService;
  30. use App\Services\NotificationService;
  31. use App\Http\Resources\AdminInstance;
  32. use App\Http\Resources\AdminUser;
  33. use App\Jobs\DeletePipeline\DeleteAccountPipeline;
  34. use App\Jobs\DeletePipeline\DeleteRemoteProfilePipeline;
  35. use App\Jobs\DeletePipeline\DeleteRemoteStatusPipeline;
  36. class AdminApiController extends Controller
  37. {
  38. public function supported(Request $request)
  39. {
  40. abort_if(!$request->user(), 404);
  41. abort_unless($request->user()->is_admin == 1, 404);
  42. return response()->json(['supported' => true]);
  43. }
  44. public function getStats(Request $request)
  45. {
  46. abort_if(!$request->user(), 404);
  47. abort_unless($request->user()->is_admin == 1, 404);
  48. $res = AdminStatsService::summary();
  49. $res['autospam_count'] = AccountInterstitial::whereType('post.autospam')
  50. ->whereNull('appeal_handled_at')
  51. ->count();
  52. return $res;
  53. }
  54. public function autospam(Request $request)
  55. {
  56. abort_if(!$request->user(), 404);
  57. abort_unless($request->user()->is_admin == 1, 404);
  58. $appeals = AccountInterstitial::whereType('post.autospam')
  59. ->whereNull('appeal_handled_at')
  60. ->latest()
  61. ->simplePaginate(6)
  62. ->map(function($report) {
  63. $r = [
  64. 'id' => $report->id,
  65. 'type' => $report->type,
  66. 'item_id' => $report->item_id,
  67. 'item_type' => $report->item_type,
  68. 'created_at' => $report->created_at
  69. ];
  70. if($report->item_type === 'App\\Status') {
  71. $status = StatusService::get($report->item_id, false);
  72. if(!$status) {
  73. return;
  74. }
  75. $r['status'] = $status;
  76. if($status['in_reply_to_id']) {
  77. $r['parent'] = StatusService::get($status['in_reply_to_id'], false);
  78. }
  79. }
  80. return $r;
  81. });
  82. return $appeals;
  83. }
  84. public function autospamHandle(Request $request)
  85. {
  86. abort_if(!$request->user(), 404);
  87. abort_unless($request->user()->is_admin == 1, 404);
  88. $this->validate($request, [
  89. 'action' => 'required|in:dismiss,approve,dismiss-all,approve-all,delete-post,delete-account',
  90. 'id' => 'required'
  91. ]);
  92. $action = $request->input('action');
  93. $id = $request->input('id');
  94. $appeal = AccountInterstitial::whereType('post.autospam')
  95. ->whereNull('appeal_handled_at')
  96. ->findOrFail($id);
  97. $now = now();
  98. $res = ['status' => 'success'];
  99. $meta = json_decode($appeal->meta);
  100. $user = $appeal->user;
  101. $profile = $user->profile;
  102. if($action == 'dismiss') {
  103. $appeal->is_spam = true;
  104. $appeal->appeal_handled_at = $now;
  105. $appeal->save();
  106. Cache::forget('pf:bouncer_v0:exemption_by_pid:' . $profile->id);
  107. Cache::forget('pf:bouncer_v0:recent_by_pid:' . $profile->id);
  108. Cache::forget('admin-dash:reports:spam-count');
  109. return $res;
  110. }
  111. if($action == 'delete-post') {
  112. $appeal->appeal_handled_at = now();
  113. $appeal->is_spam = true;
  114. $appeal->save();
  115. ModLogService::boot()
  116. ->objectUid($profile->id)
  117. ->objectId($appeal->status->id)
  118. ->objectType('App\Status::class')
  119. ->user($request->user())
  120. ->action('admin.status.delete')
  121. ->accessLevel('admin')
  122. ->save();
  123. PublicTimelineService::deleteByProfileId($profile->id);
  124. StatusDelete::dispatch($appeal->status)->onQueue('high');
  125. Cache::forget('admin-dash:reports:spam-count');
  126. return $res;
  127. }
  128. if($action == 'delete-account') {
  129. abort_if($user->is_admin, 400, 'Cannot delete an admin account.');
  130. $appeal->appeal_handled_at = now();
  131. $appeal->is_spam = true;
  132. $appeal->save();
  133. ModLogService::boot()
  134. ->objectUid($profile->id)
  135. ->objectId($profile->id)
  136. ->objectType('App\User::class')
  137. ->user($request->user())
  138. ->action('admin.user.delete')
  139. ->accessLevel('admin')
  140. ->save();
  141. PublicTimelineService::deleteByProfileId($profile->id);
  142. DeleteAccountPipeline::dispatch($appeal->user)->onQueue('high');
  143. Cache::forget('admin-dash:reports:spam-count');
  144. return $res;
  145. }
  146. if($action == 'dismiss-all') {
  147. AccountInterstitial::whereType('post.autospam')
  148. ->whereItemType('App\Status')
  149. ->whereNull('appeal_handled_at')
  150. ->whereUserId($appeal->user_id)
  151. ->update(['appeal_handled_at' => $now, 'is_spam' => true]);
  152. Cache::forget('pf:bouncer_v0:exemption_by_pid:' . $appeal->user->profile_id);
  153. Cache::forget('pf:bouncer_v0:recent_by_pid:' . $appeal->user->profile_id);
  154. Cache::forget('admin-dash:reports:spam-count');
  155. return $res;
  156. }
  157. if($action == 'approve') {
  158. $status = $appeal->status;
  159. $status->is_nsfw = $meta->is_nsfw;
  160. $status->scope = 'public';
  161. $status->visibility = 'public';
  162. $status->save();
  163. $appeal->is_spam = false;
  164. $appeal->appeal_handled_at = now();
  165. $appeal->save();
  166. StatusService::del($status->id);
  167. Notification::whereAction('autospam.warning')
  168. ->whereProfileId($appeal->user->profile_id)
  169. ->get()
  170. ->each(function($n) use($appeal) {
  171. NotificationService::del($appeal->user->profile_id, $n->id);
  172. $n->forceDelete();
  173. });
  174. Cache::forget('pf:bouncer_v0:exemption_by_pid:' . $appeal->user->profile_id);
  175. Cache::forget('pf:bouncer_v0:recent_by_pid:' . $appeal->user->profile_id);
  176. Cache::forget('admin-dash:reports:spam-count');
  177. return $res;
  178. }
  179. if($action == 'approve-all') {
  180. AccountInterstitial::whereType('post.autospam')
  181. ->whereItemType('App\Status')
  182. ->whereNull('appeal_handled_at')
  183. ->whereUserId($appeal->user_id)
  184. ->get()
  185. ->each(function($report) use($meta) {
  186. $report->is_spam = false;
  187. $report->appeal_handled_at = now();
  188. $report->save();
  189. $status = Status::find($report->item_id);
  190. if($status) {
  191. $status->is_nsfw = $meta->is_nsfw;
  192. $status->scope = 'public';
  193. $status->visibility = 'public';
  194. $status->save();
  195. StatusService::del($status->id, true);
  196. }
  197. Notification::whereAction('autospam.warning')
  198. ->whereProfileId($report->user->profile_id)
  199. ->get()
  200. ->each(function($n) use($report) {
  201. NotificationService::del($report->user->profile_id, $n->id);
  202. $n->forceDelete();
  203. });
  204. });
  205. Cache::forget('pf:bouncer_v0:exemption_by_pid:' . $appeal->user->profile_id);
  206. Cache::forget('pf:bouncer_v0:recent_by_pid:' . $appeal->user->profile_id);
  207. Cache::forget('admin-dash:reports:spam-count');
  208. return $res;
  209. }
  210. return $res;
  211. }
  212. public function modReports(Request $request)
  213. {
  214. abort_if(!$request->user(), 404);
  215. abort_unless($request->user()->is_admin == 1, 404);
  216. $reports = Report::whereNull('admin_seen')
  217. ->orderBy('created_at','desc')
  218. ->paginate(6)
  219. ->map(function($report) {
  220. $r = [
  221. 'id' => $report->id,
  222. 'type' => $report->type,
  223. 'message' => $report->message,
  224. 'object_id' => $report->object_id,
  225. 'object_type' => $report->object_type,
  226. 'created_at' => $report->created_at
  227. ];
  228. if($report->profile_id) {
  229. $r['reported_by_account'] = AccountService::get($report->profile_id, true);
  230. }
  231. if($report->object_type === 'App\\Status') {
  232. $status = StatusService::get($report->object_id, false);
  233. if(!$status) {
  234. return;
  235. }
  236. $r['status'] = $status;
  237. if($status['in_reply_to_id']) {
  238. $r['parent'] = StatusService::get($status['in_reply_to_id'], false);
  239. }
  240. }
  241. if($report->object_type === 'App\\Profile') {
  242. $r['account'] = AccountService::get($report->object_id, false);
  243. }
  244. return $r;
  245. })
  246. ->filter()
  247. ->values();
  248. return $reports;
  249. }
  250. public function modReportHandle(Request $request)
  251. {
  252. abort_if(!$request->user(), 404);
  253. abort_unless($request->user()->is_admin == 1, 404);
  254. $this->validate($request, [
  255. 'action' => 'required|string',
  256. 'id' => 'required'
  257. ]);
  258. $action = $request->input('action');
  259. $id = $request->input('id');
  260. $actions = [
  261. 'ignore',
  262. 'cw',
  263. 'unlist'
  264. ];
  265. if (!in_array($action, $actions)) {
  266. return abort(403);
  267. }
  268. $report = Report::findOrFail($id);
  269. $item = $report->reported();
  270. $report->admin_seen = now();
  271. switch ($action) {
  272. case 'ignore':
  273. $report->not_interested = true;
  274. break;
  275. case 'cw':
  276. Cache::forget('status:thumb:'.$item->id);
  277. $item->is_nsfw = true;
  278. $item->save();
  279. $report->nsfw = true;
  280. StatusService::del($item->id, true);
  281. break;
  282. case 'unlist':
  283. $item->visibility = 'unlisted';
  284. $item->save();
  285. StatusService::del($item->id, true);
  286. break;
  287. default:
  288. $report->admin_seen = null;
  289. break;
  290. }
  291. $report->save();
  292. Cache::forget('admin-dash:reports:list-cache');
  293. Cache::forget('admin:dashboard:home:data:v0:15min');
  294. return ['success' => true];
  295. }
  296. public function getConfiguration(Request $request)
  297. {
  298. abort_if(!$request->user(), 404);
  299. abort_unless($request->user()->is_admin == 1, 404);
  300. abort_unless(config('instance.enable_cc'), 400);
  301. return collect([
  302. [
  303. 'name' => 'ActivityPub Federation',
  304. 'description' => 'Enable activitypub federation support, compatible with Pixelfed, Mastodon and other platforms.',
  305. 'key' => 'federation.activitypub.enabled'
  306. ],
  307. [
  308. 'name' => 'Open Registration',
  309. 'description' => 'Allow new account registrations.',
  310. 'key' => 'pixelfed.open_registration'
  311. ],
  312. [
  313. 'name' => 'Stories',
  314. 'description' => 'Enable the ephemeral Stories feature.',
  315. 'key' => 'instance.stories.enabled'
  316. ],
  317. [
  318. 'name' => 'Require Email Verification',
  319. 'description' => 'Require new accounts to verify their email address.',
  320. 'key' => 'pixelfed.enforce_email_verification'
  321. ],
  322. [
  323. 'name' => 'AutoSpam Detection',
  324. 'description' => 'Detect and remove spam from public timelines.',
  325. 'key' => 'pixelfed.bouncer.enabled'
  326. ],
  327. ])
  328. ->map(function($s) {
  329. $s['state'] = (bool) config_cache($s['key']);
  330. return $s;
  331. });
  332. }
  333. public function updateConfiguration(Request $request)
  334. {
  335. abort_if(!$request->user(), 404);
  336. abort_unless($request->user()->is_admin == 1, 404);
  337. abort_unless(config('instance.enable_cc'), 400);
  338. $this->validate($request, [
  339. 'key' => 'required',
  340. 'value' => 'required'
  341. ]);
  342. $allowedKeys = [
  343. 'federation.activitypub.enabled',
  344. 'pixelfed.open_registration',
  345. 'instance.stories.enabled',
  346. 'pixelfed.enforce_email_verification',
  347. 'pixelfed.bouncer.enabled',
  348. ];
  349. $key = $request->input('key');
  350. $value = (bool) filter_var($request->input('value'), FILTER_VALIDATE_BOOLEAN);
  351. abort_if(!in_array($key, $allowedKeys), 400, 'Invalid cache key.');
  352. ConfigCacheService::put($key, $value);
  353. return collect([
  354. [
  355. 'name' => 'ActivityPub Federation',
  356. 'description' => 'Enable activitypub federation support, compatible with Pixelfed, Mastodon and other platforms.',
  357. 'key' => 'federation.activitypub.enabled'
  358. ],
  359. [
  360. 'name' => 'Open Registration',
  361. 'description' => 'Allow new account registrations.',
  362. 'key' => 'pixelfed.open_registration'
  363. ],
  364. [
  365. 'name' => 'Stories',
  366. 'description' => 'Enable the ephemeral Stories feature.',
  367. 'key' => 'instance.stories.enabled'
  368. ],
  369. [
  370. 'name' => 'Require Email Verification',
  371. 'description' => 'Require new accounts to verify their email address.',
  372. 'key' => 'pixelfed.enforce_email_verification'
  373. ],
  374. [
  375. 'name' => 'AutoSpam Detection',
  376. 'description' => 'Detect and remove spam from public timelines.',
  377. 'key' => 'pixelfed.bouncer.enabled'
  378. ],
  379. ])
  380. ->map(function($s) {
  381. $s['state'] = (bool) config_cache($s['key']);
  382. return $s;
  383. });
  384. }
  385. public function getUsers(Request $request)
  386. {
  387. abort_if(!$request->user(), 404);
  388. abort_unless($request->user()->is_admin == 1, 404);
  389. $this->validate($request, [
  390. 'sort' => 'sometimes|in:asc,desc',
  391. ]);
  392. $q = $request->input('q');
  393. $sort = $request->input('sort', 'desc') === 'asc' ? 'asc' : 'desc';
  394. $res = User::whereNull('status')
  395. ->when($q, function($query, $q) {
  396. return $query->where('username', 'like', '%' . $q . '%');
  397. })
  398. ->orderBy('id', $sort)
  399. ->cursorPaginate(10);
  400. return AdminUser::collection($res);
  401. }
  402. public function getUser(Request $request)
  403. {
  404. abort_if(!$request->user(), 404);
  405. abort_unless($request->user()->is_admin == 1, 404);
  406. $id = $request->input('user_id');
  407. $key = 'pf-admin-api:getUser:byId:' . $id;
  408. if($request->has('refresh')) {
  409. Cache::forget($key);
  410. }
  411. return Cache::remember($key, 86400, function() use($id) {
  412. $user = User::findOrFail($id);
  413. $profile = $user->profile;
  414. $account = AccountService::get($user->profile_id, true);
  415. $res = (new AdminUser($user))->additional(['meta' => [
  416. 'cached_at' => str_replace('+00:00', 'Z', now()->format(DATE_RFC3339_EXTENDED)),
  417. 'account' => $account,
  418. 'dms_sent' => Conversation::whereFromId($profile->id)->count(),
  419. 'report_count' => Report::where('object_id', $profile->id)->orWhere('reported_profile_id', $profile->id)->count(),
  420. 'remote_report_count' => RemoteReport::whereAccountId($profile->id)->count(),
  421. 'moderation' => [
  422. 'unlisted' => (bool) $profile->unlisted,
  423. 'cw' => (bool) $profile->cw,
  424. 'no_autolink' => (bool) $profile->no_autolink
  425. ]
  426. ]]);
  427. return $res;
  428. });
  429. }
  430. public function userAdminAction(Request $request)
  431. {
  432. abort_if(!$request->user(), 404);
  433. abort_unless($request->user()->is_admin == 1, 404);
  434. $this->validate($request, [
  435. 'id' => 'required',
  436. 'action' => 'required|in:unlisted,cw,no_autolink,refresh_stats,verify_email,delete',
  437. 'value' => 'sometimes'
  438. ]);
  439. $id = $request->input('id');
  440. $user = User::findOrFail($id);
  441. $profile = Profile::findOrFail($user->profile_id);
  442. $action = $request->input('action');
  443. abort_if($user->is_admin == true && $action !== 'refresh_stats', 400, 'Cannot moderate admin accounts');
  444. if($action === 'delete') {
  445. if(config('pixelfed.account_deletion') == false) {
  446. abort(404);
  447. }
  448. abort_if($user->is_admin, 400, 'Cannot delete an admin account.');
  449. $ts = now()->addMonth();
  450. $user->status = 'delete';
  451. $user->delete_after = $ts;
  452. $user->save();
  453. $profile->status = 'delete';
  454. $profile->delete_after = $ts;
  455. $profile->save();
  456. ModLogService::boot()
  457. ->objectUid($profile->id)
  458. ->objectId($profile->id)
  459. ->objectType('App\Profile::class')
  460. ->user($request->user())
  461. ->action('admin.user.delete')
  462. ->accessLevel('admin')
  463. ->save();
  464. PublicTimelineService::deleteByProfileId($profile->id);
  465. NetworkTimelineService::deleteByProfileId($profile->id);
  466. if($profile->user_id) {
  467. DB::table('oauth_access_tokens')->whereUserId($user->id)->delete();
  468. DB::table('oauth_auth_codes')->whereUserId($user->id)->delete();
  469. $user->email = $user->id;
  470. $user->password = '';
  471. $user->status = 'delete';
  472. $user->save();
  473. $profile->status = 'delete';
  474. $profile->delete_after = now()->addMonth();
  475. $profile->save();
  476. AccountService::del($profile->id);
  477. DeleteAccountPipeline::dispatch($user)->onQueue('high');
  478. } else {
  479. $profile->status = 'delete';
  480. $profile->delete_after = now()->addMonth();
  481. $profile->save();
  482. AccountService::del($profile->id);
  483. DeleteRemoteProfilePipeline::dispatch($profile)->onQueue('high');
  484. }
  485. return [
  486. 'status' => 200,
  487. 'msg' => 'deleted',
  488. ];
  489. } else if($action === 'refresh_stats') {
  490. $profile->following_count = DB::table('followers')->whereProfileId($user->profile_id)->count();
  491. $profile->followers_count = DB::table('followers')->whereFollowingId($user->profile_id)->count();
  492. $statusCount = Status::whereProfileId($user->profile_id)
  493. ->whereNull('in_reply_to_id')
  494. ->whereNull('reblog_of_id')
  495. ->whereIn('scope', ['public', 'unlisted', 'private'])
  496. ->count();
  497. $profile->status_count = $statusCount;
  498. $profile->save();
  499. } else if($action === 'verify_email') {
  500. $user->email_verified_at = now();
  501. $user->save();
  502. ModLogService::boot()
  503. ->objectUid($user->id)
  504. ->objectId($user->id)
  505. ->objectType('App\User::class')
  506. ->user($request->user())
  507. ->action('admin.user.moderate')
  508. ->metadata([
  509. 'action' => 'Manually verified email address',
  510. 'message' => 'Success!'
  511. ])
  512. ->accessLevel('admin')
  513. ->save();
  514. } else if($action === 'unlisted') {
  515. ModLogService::boot()
  516. ->objectUid($profile->id)
  517. ->objectId($profile->id)
  518. ->objectType('App\Profile::class')
  519. ->user($request->user())
  520. ->action('admin.user.moderate')
  521. ->metadata([
  522. 'action' => $action,
  523. 'message' => 'Success!'
  524. ])
  525. ->accessLevel('admin')
  526. ->save();
  527. $profile->unlisted = !$profile->unlisted;
  528. $profile->save();
  529. } else if($action === 'cw') {
  530. ModLogService::boot()
  531. ->objectUid($profile->id)
  532. ->objectId($profile->id)
  533. ->objectType('App\Profile::class')
  534. ->user($request->user())
  535. ->action('admin.user.moderate')
  536. ->metadata([
  537. 'action' => $action,
  538. 'message' => 'Success!'
  539. ])
  540. ->accessLevel('admin')
  541. ->save();
  542. $profile->cw = !$profile->cw;
  543. $profile->save();
  544. } else if($action === 'no_autolink') {
  545. ModLogService::boot()
  546. ->objectUid($profile->id)
  547. ->objectId($profile->id)
  548. ->objectType('App\Profile::class')
  549. ->user($request->user())
  550. ->action('admin.user.moderate')
  551. ->metadata([
  552. 'action' => $action,
  553. 'message' => 'Success!'
  554. ])
  555. ->accessLevel('admin')
  556. ->save();
  557. $profile->no_autolink = !$profile->no_autolink;
  558. $profile->save();
  559. } else {
  560. $profile->{$action} = filter_var($request->input('value'), FILTER_VALIDATE_BOOLEAN);
  561. $profile->save();
  562. ModLogService::boot()
  563. ->objectUid($user->id)
  564. ->objectId($user->id)
  565. ->objectType('App\User::class')
  566. ->user($request->user())
  567. ->action('admin.user.moderate')
  568. ->metadata([
  569. 'action' => $action,
  570. 'message' => 'Success!'
  571. ])
  572. ->accessLevel('admin')
  573. ->save();
  574. }
  575. AccountService::del($user->profile_id);
  576. $account = AccountService::get($user->profile_id, true);
  577. return (new AdminUser($user))->additional(['meta' => [
  578. 'account' => $account,
  579. 'moderation' => [
  580. 'unlisted' => (bool) $profile->unlisted,
  581. 'cw' => (bool) $profile->cw,
  582. 'no_autolink' => (bool) $profile->no_autolink
  583. ]
  584. ]]);
  585. }
  586. public function instances(Request $request)
  587. {
  588. abort_if(!$request->user(), 404);
  589. abort_unless($request->user()->is_admin == 1, 404);
  590. $this->validate($request, [
  591. 'q' => 'sometimes',
  592. 'sort' => 'sometimes|in:asc,desc',
  593. 'sort_by' => 'sometimes|in:id,status_count,user_count,domain',
  594. 'filter' => 'sometimes|in:all,unlisted,auto_cw,banned',
  595. ]);
  596. $q = $request->input('q');
  597. $sort = $request->input('sort', 'desc') === 'asc' ? 'asc' : 'desc';
  598. $sortBy = $request->input('sort_by', 'id');
  599. $filter = $request->input('filter');
  600. $res = Instance::when($q, function($query, $q) {
  601. return $query->where('domain', 'like', '%' . $q . '%');
  602. })
  603. ->when($filter, function($query, $filter) {
  604. if($filter === 'all') {
  605. return $query;
  606. } else {
  607. return $query->where($filter, true);
  608. }
  609. })
  610. ->when($sortBy, function($query, $sortBy) use($sort) {
  611. return $query->orderBy($sortBy, $sort);
  612. }, function($query) {
  613. return $query->orderBy('id', 'desc');
  614. })
  615. ->cursorPaginate(10)
  616. ->withQueryString();
  617. return AdminInstance::collection($res);
  618. }
  619. public function getInstance(Request $request)
  620. {
  621. abort_if(!$request->user(), 404);
  622. abort_unless($request->user()->is_admin == 1, 404);
  623. $id = $request->input('id');
  624. $res = Instance::findOrFail($id);
  625. return new AdminInstance($res);
  626. }
  627. public function moderateInstance(Request $request)
  628. {
  629. abort_if(!$request->user(), 404);
  630. abort_unless($request->user()->is_admin == 1, 404);
  631. $this->validate($request, [
  632. 'id' => 'required',
  633. 'key' => 'required|in:unlisted,auto_cw,banned',
  634. 'value' => 'required'
  635. ]);
  636. $id = $request->input('id');
  637. $key = $request->input('key');
  638. $value = (bool) filter_var($request->input('value'), FILTER_VALIDATE_BOOLEAN);
  639. $res = Instance::findOrFail($id);
  640. $res->{$key} = $value;
  641. $res->save();
  642. InstanceService::refresh();
  643. NetworkTimelineService::warmCache(true);
  644. return new AdminInstance($res);
  645. }
  646. public function refreshInstanceStats(Request $request)
  647. {
  648. abort_if(!$request->user(), 404);
  649. abort_unless($request->user()->is_admin == 1, 404);
  650. $this->validate($request, [
  651. 'id' => 'required',
  652. ]);
  653. $id = $request->input('id');
  654. $instance = Instance::findOrFail($id);
  655. $instance->user_count = Profile::whereDomain($instance->domain)->count();
  656. $instance->status_count = Profile::whereDomain($instance->domain)->leftJoin('statuses', 'profiles.id', '=', 'statuses.profile_id')->count();
  657. $instance->save();
  658. return new AdminInstance($instance);
  659. }
  660. public function getAllStats(Request $request)
  661. {
  662. abort_if(!$request->user(), 404);
  663. abort_unless($request->user()->is_admin === 1, 404);
  664. if($request->has('refresh')) {
  665. Cache::forget('admin-api:instance-all-stats-v1');
  666. }
  667. return Cache::remember('admin-api:instance-all-stats-v1', 1209600, function() {
  668. $days = range(1, 7);
  669. $res = [
  670. 'cached_at' => now()->format('c'),
  671. ];
  672. $minStatusId = SnowflakeService::byDate(now()->subDays(7));
  673. foreach($days as $day) {
  674. $label = now()->subDays($day)->format('D');
  675. $labelShort = substr($label, 0, 1);
  676. $res['users']['days'][] = [
  677. 'date' => now()->subDays($day)->format('M j Y'),
  678. 'label_full' => $label,
  679. 'label' => $labelShort,
  680. 'count' => User::whereDate('created_at', now()->subDays($day))->count()
  681. ];
  682. $res['posts']['days'][] = [
  683. 'date' => now()->subDays($day)->format('M j Y'),
  684. 'label_full' => $label,
  685. 'label' => $labelShort,
  686. 'count' => Status::whereNull('uri')->where('id', '>', $minStatusId)->whereDate('created_at', now()->subDays($day))->count()
  687. ];
  688. $res['instances']['days'][] = [
  689. 'date' => now()->subDays($day)->format('M j Y'),
  690. 'label_full' => $label,
  691. 'label' => $labelShort,
  692. 'count' => Instance::whereDate('created_at', now()->subDays($day))->count()
  693. ];
  694. }
  695. $res['users']['total'] = DB::table('users')->count();
  696. $res['users']['min'] = collect($res['users']['days'])->min('count');
  697. $res['users']['max'] = collect($res['users']['days'])->max('count');
  698. $res['users']['change'] = collect($res['users']['days'])->sum('count');;
  699. $res['posts']['total'] = DB::table('statuses')->whereNull('uri')->count();
  700. $res['posts']['min'] = collect($res['posts']['days'])->min('count');
  701. $res['posts']['max'] = collect($res['posts']['days'])->max('count');
  702. $res['posts']['change'] = collect($res['posts']['days'])->sum('count');
  703. $res['instances']['total'] = DB::table('instances')->count();
  704. $res['instances']['min'] = collect($res['instances']['days'])->min('count');
  705. $res['instances']['max'] = collect($res['instances']['days'])->max('count');
  706. $res['instances']['change'] = collect($res['instances']['days'])->sum('count');
  707. return $res;
  708. });
  709. }
  710. }