3 namespace BookStack\Http\Controllers;
5 use BookStack\Entities\Repos\PageRepo;
6 use BookStack\Exceptions\FileUploadException;
7 use BookStack\Exceptions\NotFoundException;
8 use BookStack\Uploads\Attachment;
9 use BookStack\Uploads\AttachmentService;
11 use Illuminate\Contracts\Filesystem\FileNotFoundException;
12 use Illuminate\Http\Request;
13 use Illuminate\Support\MessageBag;
14 use Illuminate\Validation\ValidationException;
16 class AttachmentController extends Controller
18 public function __construct(
19 protected AttachmentService $attachmentService,
20 protected PageRepo $pageRepo
25 * Endpoint at which attachments are uploaded to.
27 * @throws ValidationException
28 * @throws NotFoundException
30 public function upload(Request $request)
32 $this->validate($request, [
33 'uploaded_to' => ['required', 'integer', 'exists:pages,id'],
34 'file' => array_merge(['required'], $this->attachmentService->getFileValidationRules()),
37 $pageId = $request->get('uploaded_to');
38 $page = $this->pageRepo->getById($pageId);
40 $this->checkPermission('attachment-create-all');
41 $this->checkOwnablePermission('page-update', $page);
43 $uploadedFile = $request->file('file');
46 $attachment = $this->attachmentService->saveNewUpload($uploadedFile, $pageId);
47 } catch (FileUploadException $e) {
48 return response($e->getMessage(), 500);
51 return response()->json($attachment);
55 * Update an uploaded attachment.
57 * @throws ValidationException
59 public function uploadUpdate(Request $request, $attachmentId)
61 $this->validate($request, [
62 'file' => array_merge(['required'], $this->attachmentService->getFileValidationRules()),
65 /** @var Attachment $attachment */
66 $attachment = Attachment::query()->findOrFail($attachmentId);
67 $this->checkOwnablePermission('view', $attachment->page);
68 $this->checkOwnablePermission('page-update', $attachment->page);
69 $this->checkOwnablePermission('attachment-create', $attachment);
71 $uploadedFile = $request->file('file');
74 $attachment = $this->attachmentService->saveUpdatedUpload($uploadedFile, $attachment);
75 } catch (FileUploadException $e) {
76 return response($e->getMessage(), 500);
79 return response()->json($attachment);
83 * Get the update form for an attachment.
85 public function getUpdateForm(string $attachmentId)
87 /** @var Attachment $attachment */
88 $attachment = Attachment::query()->findOrFail($attachmentId);
90 $this->checkOwnablePermission('page-update', $attachment->page);
91 $this->checkOwnablePermission('attachment-create', $attachment);
93 return view('attachments.manager-edit-form', [
94 'attachment' => $attachment,
99 * Update the details of an existing file.
101 public function update(Request $request, string $attachmentId)
103 /** @var Attachment $attachment */
104 $attachment = Attachment::query()->findOrFail($attachmentId);
107 $this->validate($request, [
108 'attachment_edit_name' => ['required', 'string', 'min:1', 'max:255'],
109 'attachment_edit_url' => ['string', 'min:1', 'max:2000', 'safe_url'],
111 } catch (ValidationException $exception) {
112 return response()->view('attachments.manager-edit-form', array_merge($request->only(['attachment_edit_name', 'attachment_edit_url']), [
113 'attachment' => $attachment,
114 'errors' => new MessageBag($exception->errors()),
118 $this->checkOwnablePermission('page-view', $attachment->page);
119 $this->checkOwnablePermission('page-update', $attachment->page);
120 $this->checkOwnablePermission('attachment-update', $attachment);
122 $attachment = $this->attachmentService->updateFile($attachment, [
123 'name' => $request->get('attachment_edit_name'),
124 'link' => $request->get('attachment_edit_url'),
127 return view('attachments.manager-edit-form', [
128 'attachment' => $attachment,
133 * Attach a link to a page.
135 * @throws NotFoundException
137 public function attachLink(Request $request)
139 $pageId = $request->get('attachment_link_uploaded_to');
142 $this->validate($request, [
143 'attachment_link_uploaded_to' => ['required', 'integer', 'exists:pages,id'],
144 'attachment_link_name' => ['required', 'string', 'min:1', 'max:255'],
145 'attachment_link_url' => ['required', 'string', 'min:1', 'max:2000', 'safe_url'],
147 } catch (ValidationException $exception) {
148 return response()->view('attachments.manager-link-form', array_merge($request->only(['attachment_link_name', 'attachment_link_url']), [
150 'errors' => new MessageBag($exception->errors()),
154 $page = $this->pageRepo->getById($pageId);
156 $this->checkPermission('attachment-create-all');
157 $this->checkOwnablePermission('page-update', $page);
159 $attachmentName = $request->get('attachment_link_name');
160 $link = $request->get('attachment_link_url');
161 $this->attachmentService->saveNewFromLink($attachmentName, $link, intval($pageId));
163 return view('attachments.manager-link-form', [
169 * Get the attachments for a specific page.
171 * @throws NotFoundException
173 public function listForPage(int $pageId)
175 $page = $this->pageRepo->getById($pageId);
176 $this->checkOwnablePermission('page-view', $page);
178 return view('attachments.manager-list', [
179 'attachments' => $page->attachments->all(),
184 * Update the attachment sorting.
186 * @throws ValidationException
187 * @throws NotFoundException
189 public function sortForPage(Request $request, int $pageId)
191 $this->validate($request, [
192 'order' => ['required', 'array'],
194 $page = $this->pageRepo->getById($pageId);
195 $this->checkOwnablePermission('page-update', $page);
197 $attachmentOrder = $request->get('order');
198 $this->attachmentService->updateFileOrderWithinPage($attachmentOrder, $pageId);
200 return response()->json(['message' => trans('entities.attachments_order_updated')]);
204 * Get an attachment from storage.
206 * @throws FileNotFoundException
207 * @throws NotFoundException
209 public function get(Request $request, string $attachmentId)
211 /** @var Attachment $attachment */
212 $attachment = Attachment::query()->findOrFail($attachmentId);
215 $page = $this->pageRepo->getById($attachment->uploaded_to);
216 } catch (NotFoundException $exception) {
217 throw new NotFoundException(trans('errors.attachment_not_found'));
220 $this->checkOwnablePermission('page-view', $page);
222 if ($attachment->external) {
223 return redirect($attachment->path);
226 $fileName = $attachment->getFileName();
227 $attachmentStream = $this->attachmentService->streamAttachmentFromStorage($attachment);
229 if ($request->get('open') === 'true') {
230 return $this->download()->streamedInline($attachmentStream, $fileName);
233 return $this->download()->streamedDirectly($attachmentStream, $fileName);
237 * Delete a specific attachment in the system.
241 public function delete(string $attachmentId)
243 /** @var Attachment $attachment */
244 $attachment = Attachment::query()->findOrFail($attachmentId);
245 $this->checkOwnablePermission('attachment-delete', $attachment);
246 $this->attachmentService->deleteFile($attachment);
248 return response()->json(['message' => trans('entities.attachments_deleted')]);