BaseApiController.php 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253
  1. <?php
  2. namespace App\Http\Controllers\Api;
  3. use Illuminate\Http\Request;
  4. use App\Http\Controllers\{
  5. Controller,
  6. AvatarController
  7. };
  8. use Auth, Cache, Storage, URL;
  9. use Carbon\Carbon;
  10. use App\{
  11. Avatar,
  12. Like,
  13. Media,
  14. Notification,
  15. Profile,
  16. Status,
  17. StatusArchived
  18. };
  19. use App\Transformer\Api\{
  20. AccountTransformer,
  21. NotificationTransformer,
  22. MediaTransformer,
  23. MediaDraftTransformer,
  24. StatusTransformer,
  25. StatusStatelessTransformer
  26. };
  27. use League\Fractal;
  28. use App\Util\Media\Filter;
  29. use League\Fractal\Serializer\ArraySerializer;
  30. use League\Fractal\Pagination\IlluminatePaginatorAdapter;
  31. use App\Jobs\AvatarPipeline\AvatarOptimize;
  32. use App\Jobs\ImageOptimizePipeline\ImageOptimize;
  33. use App\Jobs\VideoPipeline\{
  34. VideoOptimize,
  35. VideoPostProcess,
  36. VideoThumbnail
  37. };
  38. use App\Services\AccountService;
  39. use App\Services\NotificationService;
  40. use App\Services\MediaPathService;
  41. use App\Services\MediaBlocklistService;
  42. use App\Services\StatusService;
  43. class BaseApiController extends Controller
  44. {
  45. protected $fractal;
  46. public function __construct()
  47. {
  48. // $this->middleware('auth');
  49. $this->fractal = new Fractal\Manager();
  50. $this->fractal->setSerializer(new ArraySerializer());
  51. }
  52. public function notifications(Request $request)
  53. {
  54. abort_if(!$request->user() || !$request->user()->token(), 403);
  55. abort_unless($request->user()->tokenCan('read'), 403);
  56. $pid = $request->user()->profile_id;
  57. $limit = $request->input('limit', 20);
  58. $since = $request->input('since_id');
  59. $min = $request->input('min_id');
  60. $max = $request->input('max_id');
  61. if(!$since && !$min && !$max) {
  62. $min = 1;
  63. }
  64. $maxId = null;
  65. $minId = null;
  66. if($max) {
  67. $res = NotificationService::getMax($pid, $max, $limit);
  68. $ids = NotificationService::getRankedMaxId($pid, $max, $limit);
  69. if(!empty($ids)) {
  70. $maxId = max($ids);
  71. $minId = min($ids);
  72. }
  73. } else {
  74. $res = NotificationService::getMin($pid, $min ?? $since, $limit);
  75. $ids = NotificationService::getRankedMinId($pid, $min ?? $since, $limit);
  76. if(!empty($ids)) {
  77. $maxId = max($ids);
  78. $minId = min($ids);
  79. }
  80. }
  81. if(empty($res) && !Cache::has('pf:services:notifications:hasSynced:'.$pid)) {
  82. Cache::put('pf:services:notifications:hasSynced:'.$pid, 1, 1209600);
  83. NotificationService::warmCache($pid, 100, true);
  84. }
  85. return response()->json($res);
  86. }
  87. public function avatarUpdate(Request $request)
  88. {
  89. abort_if(!$request->user() || !$request->user()->token(), 403);
  90. abort_unless($request->user()->tokenCan('write'), 403);
  91. $this->validate($request, [
  92. 'upload' => 'required|mimetypes:image/jpeg,image/jpg,image/png|max:'.config('pixelfed.max_avatar_size'),
  93. ]);
  94. try {
  95. $user = Auth::user();
  96. $profile = $user->profile;
  97. $file = $request->file('upload');
  98. $path = (new AvatarController())->getPath($user, $file);
  99. $dir = $path['root'];
  100. $name = $path['name'];
  101. $public = $path['storage'];
  102. $currentAvatar = storage_path('app/'.$profile->avatar->media_path);
  103. $loc = $request->file('upload')->storePubliclyAs($public, $name);
  104. $avatar = Avatar::whereProfileId($profile->id)->firstOrFail();
  105. $opath = $avatar->media_path;
  106. $avatar->media_path = "$public/$name";
  107. $avatar->change_count = ++$avatar->change_count;
  108. $avatar->last_processed_at = null;
  109. $avatar->save();
  110. Cache::forget("avatar:{$profile->id}");
  111. AvatarOptimize::dispatch($user->profile, $currentAvatar);
  112. } catch (Exception $e) {
  113. }
  114. return response()->json([
  115. 'code' => 200,
  116. 'msg' => 'Avatar successfully updated',
  117. ]);
  118. }
  119. public function verifyCredentials(Request $request)
  120. {
  121. abort_if(!$request->user() || !$request->user()->token(), 403);
  122. abort_unless($request->user()->tokenCan('read'), 403);
  123. $user = $request->user();
  124. if ($user->status != null) {
  125. Auth::logout();
  126. abort(403);
  127. }
  128. $res = AccountService::get($user->profile_id);
  129. return response()->json($res);
  130. }
  131. public function accountLikes(Request $request)
  132. {
  133. abort_if(!$request->user() || !$request->user()->token(), 403);
  134. abort_unless($request->user()->tokenCan('read'), 403);
  135. $this->validate($request, [
  136. 'page' => 'sometimes|int|min:1|max:20',
  137. 'limit' => 'sometimes|int|min:1|max:10'
  138. ]);
  139. $user = $request->user();
  140. $limit = $request->input('limit', 10);
  141. $res = \DB::table('likes')
  142. ->whereProfileId($user->profile_id)
  143. ->latest()
  144. ->simplePaginate($limit)
  145. ->map(function($id) {
  146. $status = StatusService::get($id->status_id, false);
  147. $status['favourited'] = true;
  148. return $status;
  149. })
  150. ->filter(function($post) {
  151. return $post && isset($post['account']);
  152. })
  153. ->values();
  154. return response()->json($res);
  155. }
  156. public function archive(Request $request, $id)
  157. {
  158. abort_if(!$request->user() || !$request->user()->token(), 403);
  159. abort_unless($request->user()->tokenCan('write'), 403);
  160. $status = Status::whereNull('in_reply_to_id')
  161. ->whereNull('reblog_of_id')
  162. ->whereProfileId($request->user()->profile_id)
  163. ->findOrFail($id);
  164. if($status->scope === 'archived') {
  165. return [200];
  166. }
  167. $archive = new StatusArchived;
  168. $archive->status_id = $status->id;
  169. $archive->profile_id = $status->profile_id;
  170. $archive->original_scope = $status->scope;
  171. $archive->save();
  172. $status->scope = 'archived';
  173. $status->visibility = 'draft';
  174. $status->save();
  175. StatusService::del($status->id, true);
  176. AccountService::syncPostCount($status->profile_id);
  177. return [200];
  178. }
  179. public function unarchive(Request $request, $id)
  180. {
  181. abort_if(!$request->user() || !$request->user()->token(), 403);
  182. abort_unless($request->user()->tokenCan('write'), 403);
  183. $status = Status::whereNull('in_reply_to_id')
  184. ->whereNull('reblog_of_id')
  185. ->whereProfileId($request->user()->profile_id)
  186. ->findOrFail($id);
  187. if($status->scope !== 'archived') {
  188. return [200];
  189. }
  190. $archive = StatusArchived::whereStatusId($status->id)
  191. ->whereProfileId($status->profile_id)
  192. ->firstOrFail();
  193. $status->scope = $archive->original_scope;
  194. $status->visibility = $archive->original_scope;
  195. $status->save();
  196. $archive->delete();
  197. StatusService::del($status->id, true);
  198. AccountService::syncPostCount($status->profile_id);
  199. return [200];
  200. }
  201. public function archivedPosts(Request $request)
  202. {
  203. abort_if(!$request->user() || !$request->user()->token(), 403);
  204. abort_unless($request->user()->tokenCan('read'), 403);
  205. $statuses = Status::whereProfileId($request->user()->profile_id)
  206. ->whereScope('archived')
  207. ->orderByDesc('id')
  208. ->simplePaginate(10);
  209. $fractal = new Fractal\Manager();
  210. $fractal->setSerializer(new ArraySerializer());
  211. $resource = new Fractal\Resource\Collection($statuses, new StatusStatelessTransformer());
  212. return $fractal->createData($resource)->toArray();
  213. }
  214. }