ApiV1Controller.php 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233
  1. <?php
  2. namespace App\Http\Controllers\Api;
  3. use Illuminate\Http\Request;
  4. use App\Http\Controllers\Controller;
  5. use Illuminate\Support\Str;
  6. use App\Jobs\StatusPipeline\StatusDelete;
  7. use Laravel\Passport\Passport;
  8. use Auth, Cache, DB;
  9. use App\{
  10. Like,
  11. Media,
  12. Profile,
  13. Status
  14. };
  15. use League\Fractal;
  16. use App\Transformer\Api\{
  17. AccountTransformer,
  18. RelationshipTransformer,
  19. StatusTransformer,
  20. };
  21. use League\Fractal\Serializer\ArraySerializer;
  22. use League\Fractal\Pagination\IlluminatePaginatorAdapter;
  23. use App\Services\NotificationService;
  24. class ApiV1Controller extends Controller
  25. {
  26. protected $fractal;
  27. public function __construct()
  28. {
  29. $this->fractal = new Fractal\Manager();
  30. $this->fractal->setSerializer(new ArraySerializer());
  31. }
  32. public function apps(Request $request)
  33. {
  34. abort_if(!config('pixelfed.oauth_enabled'), 404);
  35. $this->validate($request, [
  36. 'client_name' => 'required',
  37. 'redirect_uris' => 'required',
  38. 'scopes' => 'nullable',
  39. 'website' => 'nullable'
  40. ]);
  41. $client = Passport::client()->forceFill([
  42. 'user_id' => null,
  43. 'name' => e($request->client_name),
  44. 'secret' => Str::random(40),
  45. 'redirect' => $request->redirect_uris,
  46. 'personal_access_client' => false,
  47. 'password_client' => false,
  48. 'revoked' => false,
  49. ]);
  50. $client->save();
  51. $res = [
  52. 'id' => $client->id,
  53. 'name' => $client->name,
  54. 'website' => null,
  55. 'redirect_uri' => $client->redirect,
  56. 'client_id' => $client->id,
  57. 'client_secret' => $client->secret,
  58. 'vapid_key' => null
  59. ];
  60. return $res;
  61. }
  62. /**
  63. * GET /api/v1/accounts/{id}
  64. *
  65. * @param integer $id
  66. *
  67. * @return \App\Transformer\Api\AccountTransformer
  68. */
  69. public function accountById(Request $request, $id)
  70. {
  71. $profile = Profile::whereNull('status')->findOrFail($id);
  72. $resource = new Fractal\Resource\Item($profile, new AccountTransformer());
  73. $res = $this->fractal->createData($resource)->toArray();
  74. return response()->json($res);
  75. }
  76. /**
  77. * PATCH /api/v1/accounts/update_credentials
  78. *
  79. * @return \App\Transformer\Api\AccountTransformer
  80. */
  81. public function accountUpdateCredentials(Request, $request)
  82. {
  83. abort_if(!$request->user(), 403);
  84. $this->validate($request, [
  85. 'display_name' => 'nullable|string',
  86. 'note' => 'nullable|string',
  87. 'locked' => 'nullable|boolean',
  88. // 'source.privacy' => 'nullable|in:unlisted,public,private',
  89. // 'source.sensitive' => 'nullable|boolean'
  90. ]);
  91. $user = $request->user();
  92. $profile = $user->profile;
  93. $displayName = $request->input('display_name');
  94. $note = $request->input('note');
  95. $locked = $request->input('locked');
  96. // $privacy = $request->input('source.privacy');
  97. // $sensitive = $request->input('source.sensitive');
  98. $changes = false;
  99. if($displayName !== $user->name) {
  100. $user->name = $displayName;
  101. $profile->name = $displayName;
  102. $changes = true;
  103. }
  104. if($note !== $profile->bio) {
  105. $profile->bio = e($note);
  106. $changes = true;
  107. }
  108. if(!is_null($locked)) {
  109. $profile->is_private = $locked;
  110. $changes = true;
  111. }
  112. if($changes) {
  113. $user->save();
  114. $profile->save()
  115. }
  116. $resource = new Fractal\Resource\Item($profile, new AccountTransformer());
  117. $res = $this->fractal->createData($resource)->toArray();
  118. return response()->json($res);
  119. }
  120. /**
  121. * GET /api/v1/accounts/{id}/followers
  122. *
  123. * @param integer $id
  124. *
  125. * @return \App\Transformer\Api\AccountTransformer
  126. */
  127. public function accountFollowersById(Request $request, $id)
  128. {
  129. abort_if(!$request->user(), 403);
  130. $profile = Profile::whereNull('status')->findOrFail($id);
  131. $settings = $profile->user->settings;
  132. if($settings->show_profile_followers == true) {
  133. $limit = $request->input('limit') ?? 40;
  134. $followers = $profile->followers()->paginate($limit);
  135. $resource = new Fractal\Resource\Collection($followers, new AccountTransformer());
  136. $res = $this->fractal->createData($resource)->toArray();
  137. } else {
  138. $res = [];
  139. }
  140. return response()->json($res);
  141. }
  142. public function statusById(Request $request, $id)
  143. {
  144. $status = Status::whereVisibility('public')->findOrFail($id);
  145. $resource = new Fractal\Resource\Item($status, new StatusTransformer());
  146. $res = $this->fractal->createData($resource)->toArray();
  147. return response()->json($res);
  148. }
  149. public function instance(Request $request)
  150. {
  151. $res = [
  152. 'description' => 'Pixelfed - Photo sharing for everyone',
  153. 'email' => config('instance.email'),
  154. 'languages' => ['en'],
  155. 'max_toot_chars' => config('pixelfed.max_caption_length'),
  156. 'registrations' => config('pixelfed.open_registration'),
  157. 'stats' => [
  158. 'user_count' => 0,
  159. 'status_count' => 0,
  160. 'domain_count' => 0
  161. ],
  162. 'thumbnail' => config('app.url') . '/img/pixelfed-icon-color.png',
  163. 'title' => 'Pixelfed (' . config('pixelfed.domain.app') . ')',
  164. 'uri' => config('app.url'),
  165. 'urls' => [],
  166. 'version' => '2.7.2 (compatible; Pixelfed ' . config('pixelfed.version') . ')'
  167. ];
  168. return response()->json($res, 200, [], JSON_PRETTY_PRINT);
  169. }
  170. public function filters(Request $request)
  171. {
  172. // Pixelfed does not yet support keyword filters
  173. return response()->json([]);
  174. }
  175. public function context(Request $request)
  176. {
  177. // todo
  178. $res = [
  179. 'ancestors' => [],
  180. 'descendants' => []
  181. ];
  182. return response()->json($res);
  183. }
  184. public function createStatus(Request $request)
  185. {
  186. abort_if(!$request->user(), 403);
  187. $this->validate($request, [
  188. 'status' => 'string',
  189. 'media_ids' => 'array',
  190. 'media_ids.*' => 'integer|min:1',
  191. 'sensitive' => 'nullable|boolean',
  192. 'visibility' => 'string|in:private,unlisted,public',
  193. 'in_reply_to_id' => 'integer'
  194. ]);
  195. if(!$request->filled('media_ids') && !$request->filled('in_reply_to_id')) {
  196. abort(403, 'Empty statuses are not allowed');
  197. }
  198. }
  199. }