BaseApiController.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  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(), 403);
  55. $pid = $request->user()->profile_id;
  56. $limit = $request->input('limit', 20);
  57. $since = $request->input('since_id');
  58. $min = $request->input('min_id');
  59. $max = $request->input('max_id');
  60. if(!$since && !$min && !$max) {
  61. $min = 1;
  62. }
  63. $maxId = null;
  64. $minId = null;
  65. if($max) {
  66. $res = NotificationService::getMax($pid, $max, $limit);
  67. $ids = NotificationService::getRankedMaxId($pid, $max, $limit);
  68. if(!empty($ids)) {
  69. $maxId = max($ids);
  70. $minId = min($ids);
  71. }
  72. } else {
  73. $res = NotificationService::getMin($pid, $min ?? $since, $limit);
  74. $ids = NotificationService::getRankedMinId($pid, $min ?? $since, $limit);
  75. if(!empty($ids)) {
  76. $maxId = max($ids);
  77. $minId = min($ids);
  78. }
  79. }
  80. if(empty($res) && !Cache::has('pf:services:notifications:hasSynced:'.$pid)) {
  81. Cache::put('pf:services:notifications:hasSynced:'.$pid, 1, 1209600);
  82. NotificationService::warmCache($pid, 400, true);
  83. }
  84. return response()->json($res);
  85. }
  86. public function accounts(Request $request, $id)
  87. {
  88. abort_if(!$request->user(), 403);
  89. $profile = Profile::findOrFail($id);
  90. $resource = new Fractal\Resource\Item($profile, new AccountTransformer());
  91. $res = $this->fractal->createData($resource)->toArray();
  92. return response()->json($res);
  93. }
  94. public function accountFollowers(Request $request, $id)
  95. {
  96. abort_if(!$request->user(), 403);
  97. $profile = Profile::findOrFail($id);
  98. $followers = $profile->followers;
  99. $resource = new Fractal\Resource\Collection($followers, new AccountTransformer());
  100. $res = $this->fractal->createData($resource)->toArray();
  101. return response()->json($res);
  102. }
  103. public function accountFollowing(Request $request, $id)
  104. {
  105. abort_if(!$request->user(), 403);
  106. $profile = Profile::findOrFail($id);
  107. $following = $profile->following;
  108. $resource = new Fractal\Resource\Collection($following, new AccountTransformer());
  109. $res = $this->fractal->createData($resource)->toArray();
  110. return response()->json($res);
  111. }
  112. public function accountStatuses(Request $request, $id)
  113. {
  114. abort_if(!$request->user(), 403);
  115. $this->validate($request, [
  116. 'only_media' => 'nullable',
  117. 'pinned' => 'nullable',
  118. 'exclude_replies' => 'nullable',
  119. 'max_id' => 'nullable|integer|min:1',
  120. 'since_id' => 'nullable|integer|min:1',
  121. 'min_id' => 'nullable|integer|min:1',
  122. 'limit' => 'nullable|integer|min:1|max:24'
  123. ]);
  124. $limit = $request->limit ?? 20;
  125. $max_id = $request->max_id ?? false;
  126. $min_id = $request->min_id ?? false;
  127. $since_id = $request->since_id ?? false;
  128. $only_media = $request->only_media ?? false;
  129. $user = Auth::user();
  130. $account = Profile::whereNull('status')->findOrFail($id);
  131. $statuses = $account->statuses()->getQuery();
  132. if($only_media == true) {
  133. $statuses = $statuses
  134. ->whereIn('scope', ['public','unlisted'])
  135. ->whereHas('media')
  136. ->whereNull('in_reply_to_id')
  137. ->whereNull('reblog_of_id');
  138. }
  139. if($id == $account->id && !$max_id && !$min_id && !$since_id) {
  140. $statuses = $statuses->orderBy('id', 'desc')
  141. ->paginate($limit);
  142. } else if($since_id) {
  143. $statuses = $statuses->where('id', '>', $since_id)
  144. ->orderBy('id', 'DESC')
  145. ->paginate($limit);
  146. } else if($min_id) {
  147. $statuses = $statuses->where('id', '>', $min_id)
  148. ->orderBy('id', 'ASC')
  149. ->paginate($limit);
  150. } else if($max_id) {
  151. $statuses = $statuses->where('id', '<', $max_id)
  152. ->orderBy('id', 'DESC')
  153. ->paginate($limit);
  154. } else {
  155. $statuses = $statuses->whereScope('public')->orderBy('id', 'desc')->paginate($limit);
  156. }
  157. $resource = new Fractal\Resource\Collection($statuses, new StatusTransformer());
  158. $res = $this->fractal->createData($resource)->toArray();
  159. return response()->json($res);
  160. }
  161. public function avatarUpdate(Request $request)
  162. {
  163. abort_if(!$request->user(), 403);
  164. $this->validate($request, [
  165. 'upload' => 'required|mimetypes:image/jpeg,image/jpg,image/png|max:'.config('pixelfed.max_avatar_size'),
  166. ]);
  167. try {
  168. $user = Auth::user();
  169. $profile = $user->profile;
  170. $file = $request->file('upload');
  171. $path = (new AvatarController())->getPath($user, $file);
  172. $dir = $path['root'];
  173. $name = $path['name'];
  174. $public = $path['storage'];
  175. $currentAvatar = storage_path('app/'.$profile->avatar->media_path);
  176. $loc = $request->file('upload')->storeAs($public, $name);
  177. $avatar = Avatar::whereProfileId($profile->id)->firstOrFail();
  178. $opath = $avatar->media_path;
  179. $avatar->media_path = "$public/$name";
  180. $avatar->change_count = ++$avatar->change_count;
  181. $avatar->last_processed_at = null;
  182. $avatar->save();
  183. Cache::forget("avatar:{$profile->id}");
  184. AvatarOptimize::dispatch($user->profile, $currentAvatar);
  185. } catch (Exception $e) {
  186. }
  187. return response()->json([
  188. 'code' => 200,
  189. 'msg' => 'Avatar successfully updated',
  190. ]);
  191. }
  192. public function showTempMedia(Request $request, $profileId, $mediaId, $timestamp)
  193. {
  194. abort(400, 'Endpoint deprecated');
  195. }
  196. public function uploadMedia(Request $request)
  197. {
  198. abort(400, 'Endpoint deprecated');
  199. }
  200. public function deleteMedia(Request $request)
  201. {
  202. abort(400, 'Endpoint deprecated');
  203. }
  204. public function verifyCredentials(Request $request)
  205. {
  206. $user = $request->user();
  207. abort_if(!$user, 403);
  208. if($user->status != null) {
  209. Auth::logout();
  210. abort(403);
  211. }
  212. $res = AccountService::get($user->profile_id);
  213. return response()->json($res);
  214. }
  215. public function drafts(Request $request)
  216. {
  217. $user = $request->user();
  218. abort_if(!$request->user(), 403);
  219. $medias = Media::whereUserId($user->id)
  220. ->whereNull('status_id')
  221. ->latest()
  222. ->take(13)
  223. ->get();
  224. $resource = new Fractal\Resource\Collection($medias, new MediaDraftTransformer());
  225. $res = $this->fractal->createData($resource)->toArray();
  226. return response()->json($res, 200, [], JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES);
  227. }
  228. public function accountLikes(Request $request)
  229. {
  230. abort_if(!$request->user(), 403);
  231. $this->validate($request, [
  232. 'page' => 'sometimes|int|min:1|max:20',
  233. 'limit' => 'sometimes|int|min:1|max:10'
  234. ]);
  235. $user = $request->user();
  236. $limit = $request->input('limit', 10);
  237. $res = \DB::table('likes')
  238. ->whereProfileId($user->profile_id)
  239. ->latest()
  240. ->simplePaginate($limit)
  241. ->map(function($id) {
  242. $status = StatusService::get($id->status_id, false);
  243. $status['favourited'] = true;
  244. return $status;
  245. })
  246. ->filter(function($post) {
  247. return $post && isset($post['account']);
  248. })
  249. ->values();
  250. return response()->json($res);
  251. }
  252. public function archive(Request $request, $id)
  253. {
  254. abort_if(!$request->user(), 403);
  255. $status = Status::whereNull('in_reply_to_id')
  256. ->whereNull('reblog_of_id')
  257. ->whereProfileId($request->user()->profile_id)
  258. ->findOrFail($id);
  259. if($status->scope === 'archived') {
  260. return [200];
  261. }
  262. $archive = new StatusArchived;
  263. $archive->status_id = $status->id;
  264. $archive->profile_id = $status->profile_id;
  265. $archive->original_scope = $status->scope;
  266. $archive->save();
  267. $status->scope = 'archived';
  268. $status->visibility = 'draft';
  269. $status->save();
  270. StatusService::del($status->id, true);
  271. AccountService::syncPostCount($status->profile_id);
  272. return [200];
  273. }
  274. public function unarchive(Request $request, $id)
  275. {
  276. abort_if(!$request->user(), 403);
  277. $status = Status::whereNull('in_reply_to_id')
  278. ->whereNull('reblog_of_id')
  279. ->whereProfileId($request->user()->profile_id)
  280. ->findOrFail($id);
  281. if($status->scope !== 'archived') {
  282. return [200];
  283. }
  284. $archive = StatusArchived::whereStatusId($status->id)
  285. ->whereProfileId($status->profile_id)
  286. ->firstOrFail();
  287. $status->scope = $archive->original_scope;
  288. $status->visibility = $archive->original_scope;
  289. $status->save();
  290. $archive->delete();
  291. StatusService::del($status->id, true);
  292. AccountService::syncPostCount($status->profile_id);
  293. return [200];
  294. }
  295. public function archivedPosts(Request $request)
  296. {
  297. abort_if(!$request->user(), 403);
  298. $statuses = Status::whereProfileId($request->user()->profile_id)
  299. ->whereScope('archived')
  300. ->orderByDesc('id')
  301. ->simplePaginate(10);
  302. $fractal = new Fractal\Manager();
  303. $fractal->setSerializer(new ArraySerializer());
  304. $resource = new Fractal\Resource\Collection($statuses, new StatusStatelessTransformer());
  305. return $fractal->createData($resource)->toArray();
  306. }
  307. }