BaseApiController.php 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  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. };
  18. use App\Transformer\Api\{
  19. AccountTransformer,
  20. NotificationTransformer,
  21. MediaTransformer,
  22. MediaDraftTransformer,
  23. StatusTransformer,
  24. StatusStatelessTransformer
  25. };
  26. use League\Fractal;
  27. use App\Util\Media\Filter;
  28. use League\Fractal\Serializer\ArraySerializer;
  29. use League\Fractal\Pagination\IlluminatePaginatorAdapter;
  30. use App\Jobs\AvatarPipeline\AvatarOptimize;
  31. use App\Jobs\ImageOptimizePipeline\ImageOptimize;
  32. use App\Jobs\VideoPipeline\{
  33. VideoOptimize,
  34. VideoPostProcess,
  35. VideoThumbnail
  36. };
  37. use App\Services\NotificationService;
  38. use App\Services\MediaPathService;
  39. use App\Services\MediaBlocklistService;
  40. class BaseApiController extends Controller
  41. {
  42. protected $fractal;
  43. public function __construct()
  44. {
  45. // $this->middleware('auth');
  46. $this->fractal = new Fractal\Manager();
  47. $this->fractal->setSerializer(new ArraySerializer());
  48. }
  49. public function notifications(Request $request)
  50. {
  51. abort_if(!$request->user(), 403);
  52. $pid = $request->user()->profile_id;
  53. $limit = $request->input('limit', 20);
  54. $since = $request->input('since_id');
  55. $min = $request->input('min_id');
  56. $max = $request->input('max_id');
  57. if(!$since && !$min && !$max) {
  58. $min = 1;
  59. }
  60. $maxId = null;
  61. $minId = null;
  62. if($max) {
  63. $res = NotificationService::getMax($pid, $max, $limit);
  64. $ids = NotificationService::getRankedMaxId($pid, $max, $limit);
  65. if(!empty($ids)) {
  66. $maxId = max($ids);
  67. $minId = min($ids);
  68. }
  69. } else {
  70. $res = NotificationService::getMin($pid, $min ?? $since, $limit);
  71. $ids = NotificationService::getRankedMinId($pid, $min ?? $since, $limit);
  72. if(!empty($ids)) {
  73. $maxId = max($ids);
  74. $minId = min($ids);
  75. }
  76. }
  77. if(empty($res) && !Cache::has('pf:services:notifications:hasSynced:'.$pid)) {
  78. Cache::put('pf:services:notifications:hasSynced:'.$pid, 1, 1209600);
  79. NotificationService::warmCache($pid, 400, true);
  80. }
  81. return response()->json($res);
  82. }
  83. public function accounts(Request $request, $id)
  84. {
  85. abort_if(!$request->user(), 403);
  86. $profile = Profile::findOrFail($id);
  87. $resource = new Fractal\Resource\Item($profile, new AccountTransformer());
  88. $res = $this->fractal->createData($resource)->toArray();
  89. return response()->json($res);
  90. }
  91. public function accountFollowers(Request $request, $id)
  92. {
  93. abort_if(!$request->user(), 403);
  94. $profile = Profile::findOrFail($id);
  95. $followers = $profile->followers;
  96. $resource = new Fractal\Resource\Collection($followers, new AccountTransformer());
  97. $res = $this->fractal->createData($resource)->toArray();
  98. return response()->json($res);
  99. }
  100. public function accountFollowing(Request $request, $id)
  101. {
  102. abort_if(!$request->user(), 403);
  103. $profile = Profile::findOrFail($id);
  104. $following = $profile->following;
  105. $resource = new Fractal\Resource\Collection($following, new AccountTransformer());
  106. $res = $this->fractal->createData($resource)->toArray();
  107. return response()->json($res);
  108. }
  109. public function accountStatuses(Request $request, $id)
  110. {
  111. abort_if(!$request->user(), 403);
  112. $this->validate($request, [
  113. 'only_media' => 'nullable',
  114. 'pinned' => 'nullable',
  115. 'exclude_replies' => 'nullable',
  116. 'max_id' => 'nullable|integer|min:1',
  117. 'since_id' => 'nullable|integer|min:1',
  118. 'min_id' => 'nullable|integer|min:1',
  119. 'limit' => 'nullable|integer|min:1|max:24'
  120. ]);
  121. $limit = $request->limit ?? 20;
  122. $max_id = $request->max_id ?? false;
  123. $min_id = $request->min_id ?? false;
  124. $since_id = $request->since_id ?? false;
  125. $only_media = $request->only_media ?? false;
  126. $user = Auth::user();
  127. $account = Profile::whereNull('status')->findOrFail($id);
  128. $statuses = $account->statuses()->getQuery();
  129. if($only_media == true) {
  130. $statuses = $statuses
  131. ->whereIn('scope', ['public','unlisted'])
  132. ->whereHas('media')
  133. ->whereNull('in_reply_to_id')
  134. ->whereNull('reblog_of_id');
  135. }
  136. if($id == $account->id && !$max_id && !$min_id && !$since_id) {
  137. $statuses = $statuses->orderBy('id', 'desc')
  138. ->paginate($limit);
  139. } else if($since_id) {
  140. $statuses = $statuses->where('id', '>', $since_id)
  141. ->orderBy('id', 'DESC')
  142. ->paginate($limit);
  143. } else if($min_id) {
  144. $statuses = $statuses->where('id', '>', $min_id)
  145. ->orderBy('id', 'ASC')
  146. ->paginate($limit);
  147. } else if($max_id) {
  148. $statuses = $statuses->where('id', '<', $max_id)
  149. ->orderBy('id', 'DESC')
  150. ->paginate($limit);
  151. } else {
  152. $statuses = $statuses->whereScope('public')->orderBy('id', 'desc')->paginate($limit);
  153. }
  154. $resource = new Fractal\Resource\Collection($statuses, new StatusTransformer());
  155. $res = $this->fractal->createData($resource)->toArray();
  156. return response()->json($res);
  157. }
  158. public function avatarUpdate(Request $request)
  159. {
  160. abort_if(!$request->user(), 403);
  161. $this->validate($request, [
  162. 'upload' => 'required|mimes:jpeg,png,gif|max:'.config('pixelfed.max_avatar_size'),
  163. ]);
  164. try {
  165. $user = Auth::user();
  166. $profile = $user->profile;
  167. $file = $request->file('upload');
  168. $path = (new AvatarController())->getPath($user, $file);
  169. $dir = $path['root'];
  170. $name = $path['name'];
  171. $public = $path['storage'];
  172. $currentAvatar = storage_path('app/'.$profile->avatar->media_path);
  173. $loc = $request->file('upload')->storeAs($public, $name);
  174. $avatar = Avatar::whereProfileId($profile->id)->firstOrFail();
  175. $opath = $avatar->media_path;
  176. $avatar->media_path = "$public/$name";
  177. $avatar->change_count = ++$avatar->change_count;
  178. $avatar->last_processed_at = null;
  179. $avatar->save();
  180. Cache::forget("avatar:{$profile->id}");
  181. AvatarOptimize::dispatch($user->profile, $currentAvatar);
  182. } catch (Exception $e) {
  183. }
  184. return response()->json([
  185. 'code' => 200,
  186. 'msg' => 'Avatar successfully updated',
  187. ]);
  188. }
  189. public function showTempMedia(Request $request, $profileId, $mediaId, $timestamp)
  190. {
  191. abort(400, 'Endpoint deprecated');
  192. }
  193. public function uploadMedia(Request $request)
  194. {
  195. abort(400, 'Endpoint deprecated');
  196. }
  197. public function deleteMedia(Request $request)
  198. {
  199. abort(400, 'Endpoint deprecated');
  200. }
  201. public function verifyCredentials(Request $request)
  202. {
  203. $user = $request->user();
  204. abort_if(!$user, 403);
  205. if($user->status != null) {
  206. Auth::logout();
  207. return redirect('/login');
  208. }
  209. $key = 'user:last_active_at:id:'.$user->id;
  210. $ttl = now()->addMinutes(5);
  211. Cache::remember($key, $ttl, function() use($user) {
  212. $user->last_active_at = now();
  213. $user->save();
  214. return;
  215. });
  216. $resource = new Fractal\Resource\Item($user->profile, new AccountTransformer());
  217. $res = $this->fractal->createData($resource)->toArray();
  218. return response()->json($res);
  219. }
  220. public function drafts(Request $request)
  221. {
  222. $user = $request->user();
  223. abort_if(!$request->user(), 403);
  224. $medias = Media::whereUserId($user->id)
  225. ->whereNull('status_id')
  226. ->latest()
  227. ->take(13)
  228. ->get();
  229. $resource = new Fractal\Resource\Collection($medias, new MediaDraftTransformer());
  230. $res = $this->fractal->createData($resource)->toArray();
  231. return response()->json($res, 200, [], JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES);
  232. }
  233. public function accountLikes(Request $request)
  234. {
  235. $user = $request->user();
  236. abort_if(!$request->user(), 403);
  237. $limit = 10;
  238. $page = (int) $request->input('page', 1);
  239. if($page > 20) {
  240. return [];
  241. }
  242. $favourites = $user->profile->likes()
  243. ->latest()
  244. ->simplePaginate($limit)
  245. ->pluck('status_id');
  246. $statuses = Status::find($favourites)->reverse();
  247. $resource = new Fractal\Resource\Collection($statuses, new StatusStatelessTransformer());
  248. $res = $this->fractal->createData($resource)->toArray();
  249. return response()->json($res, 200, [], JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES);
  250. }
  251. }