3 namespace BookStack\Uploads\Controllers;
5 use BookStack\Entities\Queries\PageQueries;
6 use BookStack\Entities\Repos\PageRepo;
7 use BookStack\Exceptions\FileUploadException;
8 use BookStack\Exceptions\NotFoundException;
9 use BookStack\Http\Controller;
10 use BookStack\Uploads\Attachment;
11 use BookStack\Uploads\AttachmentService;
13 use Illuminate\Contracts\Filesystem\FileNotFoundException;
14 use Illuminate\Http\Request;
15 use Illuminate\Support\MessageBag;
16 use Illuminate\Validation\ValidationException;
18 class AttachmentController extends Controller
20 public function __construct(
21 protected AttachmentService $attachmentService,
22 protected PageQueries $pageQueries,
23 protected PageRepo $pageRepo
28 * Endpoint at which attachments are uploaded to.
30 * @throws ValidationException
31 * @throws NotFoundException
33 public function upload(Request $request)
35 $this->validate($request, [
36 'uploaded_to' => ['required', 'integer', 'exists:pages,id'],
37 'file' => array_merge(['required'], $this->attachmentService->getFileValidationRules()),
40 $pageId = $request->get('uploaded_to');
41 $page = $this->pageQueries->findVisibleByIdOrFail($pageId);
43 $this->checkPermission('attachment-create-all');
44 $this->checkOwnablePermission('page-update', $page);
46 $uploadedFile = $request->file('file');
49 $attachment = $this->attachmentService->saveNewUpload($uploadedFile, $pageId);
50 } catch (FileUploadException $e) {
51 return response($e->getMessage(), 500);
54 return response()->json($attachment);
58 * Update an uploaded attachment.
60 * @throws ValidationException
62 public function uploadUpdate(Request $request, $attachmentId)
64 $this->validate($request, [
65 'file' => array_merge(['required'], $this->attachmentService->getFileValidationRules()),
68 /** @var Attachment $attachment */
69 $attachment = Attachment::query()->findOrFail($attachmentId);
70 $this->checkOwnablePermission('view', $attachment->page);
71 $this->checkOwnablePermission('page-update', $attachment->page);
72 $this->checkOwnablePermission('attachment-create', $attachment);
74 $uploadedFile = $request->file('file');
77 $attachment = $this->attachmentService->saveUpdatedUpload($uploadedFile, $attachment);
78 } catch (FileUploadException $e) {
79 return response($e->getMessage(), 500);
82 return response()->json($attachment);
86 * Get the update form for an attachment.
88 public function getUpdateForm(string $attachmentId)
90 /** @var Attachment $attachment */
91 $attachment = Attachment::query()->findOrFail($attachmentId);
93 $this->checkOwnablePermission('page-update', $attachment->page);
94 $this->checkOwnablePermission('attachment-create', $attachment);
96 return view('attachments.manager-edit-form', [
97 'attachment' => $attachment,
102 * Update the details of an existing file.
104 public function update(Request $request, string $attachmentId)
106 /** @var Attachment $attachment */
107 $attachment = Attachment::query()->findOrFail($attachmentId);
110 $this->validate($request, [
111 'attachment_edit_name' => ['required', 'string', 'min:1', 'max:255'],
112 'attachment_edit_url' => ['string', 'min:1', 'max:2000', 'safe_url'],
114 } catch (ValidationException $exception) {
115 return response()->view('attachments.manager-edit-form', array_merge($request->only(['attachment_edit_name', 'attachment_edit_url']), [
116 'attachment' => $attachment,
117 'errors' => new MessageBag($exception->errors()),
121 $this->checkOwnablePermission('page-view', $attachment->page);
122 $this->checkOwnablePermission('page-update', $attachment->page);
123 $this->checkOwnablePermission('attachment-update', $attachment);
125 $attachment = $this->attachmentService->updateFile($attachment, [
126 'name' => $request->get('attachment_edit_name'),
127 'link' => $request->get('attachment_edit_url'),
130 return view('attachments.manager-edit-form', [
131 'attachment' => $attachment,
136 * Attach a link to a page.
138 * @throws NotFoundException
140 public function attachLink(Request $request)
142 $pageId = $request->get('attachment_link_uploaded_to');
145 $this->validate($request, [
146 'attachment_link_uploaded_to' => ['required', 'integer', 'exists:pages,id'],
147 'attachment_link_name' => ['required', 'string', 'min:1', 'max:255'],
148 'attachment_link_url' => ['required', 'string', 'min:1', 'max:2000', 'safe_url'],
150 } catch (ValidationException $exception) {
151 return response()->view('attachments.manager-link-form', array_merge($request->only(['attachment_link_name', 'attachment_link_url']), [
153 'errors' => new MessageBag($exception->errors()),
157 $page = $this->pageQueries->findVisibleByIdOrFail($pageId);
159 $this->checkPermission('attachment-create-all');
160 $this->checkOwnablePermission('page-update', $page);
162 $attachmentName = $request->get('attachment_link_name');
163 $link = $request->get('attachment_link_url');
164 $this->attachmentService->saveNewFromLink($attachmentName, $link, intval($pageId));
166 return view('attachments.manager-link-form', [
172 * Get the attachments for a specific page.
174 * @throws NotFoundException
176 public function listForPage(int $pageId)
178 $page = $this->pageQueries->findVisibleByIdOrFail($pageId);
179 $this->checkOwnablePermission('page-view', $page);
181 return view('attachments.manager-list', [
182 'attachments' => $page->attachments->all(),
187 * Update the attachment sorting.
189 * @throws ValidationException
190 * @throws NotFoundException
192 public function sortForPage(Request $request, int $pageId)
194 $this->validate($request, [
195 'order' => ['required', 'array'],
197 $page = $this->pageQueries->findVisibleByIdOrFail($pageId);
198 $this->checkOwnablePermission('page-update', $page);
200 $attachmentOrder = $request->get('order');
201 $this->attachmentService->updateFileOrderWithinPage($attachmentOrder, $pageId);
203 return response()->json(['message' => trans('entities.attachments_order_updated')]);
207 * Get an attachment from storage.
209 * @throws FileNotFoundException
210 * @throws NotFoundException
212 public function get(Request $request, string $attachmentId)
214 /** @var Attachment $attachment */
215 $attachment = Attachment::query()->findOrFail($attachmentId);
218 $page = $this->pageQueries->findVisibleByIdOrFail($attachment->uploaded_to);
219 } catch (NotFoundException $exception) {
220 throw new NotFoundException(trans('errors.attachment_not_found'));
223 $this->checkOwnablePermission('page-view', $page);
225 if ($attachment->external) {
226 return redirect($attachment->path);
229 $fileName = $attachment->getFileName();
230 $attachmentStream = $this->attachmentService->streamAttachmentFromStorage($attachment);
231 $attachmentSize = $this->attachmentService->getAttachmentFileSize($attachment);
233 if ($request->get('open') === 'true') {
234 return $this->download()->streamedInline($attachmentStream, $fileName, $attachmentSize);
237 return $this->download()->streamedDirectly($attachmentStream, $fileName, $attachmentSize);
241 * Delete a specific attachment in the system.
245 public function delete(string $attachmentId)
247 /** @var Attachment $attachment */
248 $attachment = Attachment::query()->findOrFail($attachmentId);
249 $this->checkOwnablePermission('attachment-delete', $attachment);
250 $this->attachmentService->deleteFile($attachment);
252 return response()->json(['message' => trans('entities.attachments_deleted')]);