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 protected AttachmentService $attachmentService;
19 protected PageRepo $pageRepo;
22 * AttachmentController constructor.
24 public function __construct(AttachmentService $attachmentService, PageRepo $pageRepo)
26 $this->attachmentService = $attachmentService;
27 $this->pageRepo = $pageRepo;
31 * Endpoint at which attachments are uploaded to.
33 * @throws ValidationException
34 * @throws NotFoundException
36 public function upload(Request $request)
38 $this->validate($request, [
39 'uploaded_to' => ['required', 'integer', 'exists:pages,id'],
40 'file' => array_merge(['required'], $this->attachmentService->getFileValidationRules()),
43 $pageId = $request->get('uploaded_to');
44 $page = $this->pageRepo->getById($pageId);
46 $this->checkPermission('attachment-create-all');
47 $this->checkOwnablePermission('page-update', $page);
49 $uploadedFile = $request->file('file');
52 $attachment = $this->attachmentService->saveNewUpload($uploadedFile, $pageId);
53 } catch (FileUploadException $e) {
54 return response($e->getMessage(), 500);
57 return response()->json($attachment);
61 * Update an uploaded attachment.
63 * @throws ValidationException
65 public function uploadUpdate(Request $request, $attachmentId)
67 $this->validate($request, [
68 'file' => array_merge(['required'], $this->attachmentService->getFileValidationRules()),
71 /** @var Attachment $attachment */
72 $attachment = Attachment::query()->findOrFail($attachmentId);
73 $this->checkOwnablePermission('view', $attachment->page);
74 $this->checkOwnablePermission('page-update', $attachment->page);
75 $this->checkOwnablePermission('attachment-create', $attachment);
77 $uploadedFile = $request->file('file');
80 $attachment = $this->attachmentService->saveUpdatedUpload($uploadedFile, $attachment);
81 } catch (FileUploadException $e) {
82 return response($e->getMessage(), 500);
85 return response()->json($attachment);
89 * Get the update form for an attachment.
91 public function getUpdateForm(string $attachmentId)
93 /** @var Attachment $attachment */
94 $attachment = Attachment::query()->findOrFail($attachmentId);
96 $this->checkOwnablePermission('page-update', $attachment->page);
97 $this->checkOwnablePermission('attachment-create', $attachment);
99 return view('attachments.manager-edit-form', [
100 'attachment' => $attachment,
105 * Update the details of an existing file.
107 public function update(Request $request, string $attachmentId)
109 /** @var Attachment $attachment */
110 $attachment = Attachment::query()->findOrFail($attachmentId);
113 $this->validate($request, [
114 'attachment_edit_name' => ['required', 'string', 'min:1', 'max:255'],
115 'attachment_edit_url' => ['string', 'min:1', 'max:255', 'safe_url'],
117 } catch (ValidationException $exception) {
118 return response()->view('attachments.manager-edit-form', array_merge($request->only(['attachment_edit_name', 'attachment_edit_url']), [
119 'attachment' => $attachment,
120 'errors' => new MessageBag($exception->errors()),
124 $this->checkOwnablePermission('page-view', $attachment->page);
125 $this->checkOwnablePermission('page-update', $attachment->page);
126 $this->checkOwnablePermission('attachment-update', $attachment);
128 $attachment = $this->attachmentService->updateFile($attachment, [
129 'name' => $request->get('attachment_edit_name'),
130 'link' => $request->get('attachment_edit_url'),
133 return view('attachments.manager-edit-form', [
134 'attachment' => $attachment,
139 * Attach a link to a page.
141 * @throws NotFoundException
143 public function attachLink(Request $request)
145 $pageId = $request->get('attachment_link_uploaded_to');
148 $this->validate($request, [
149 'attachment_link_uploaded_to' => ['required', 'integer', 'exists:pages,id'],
150 'attachment_link_name' => ['required', 'string', 'min:1', 'max:255'],
151 'attachment_link_url' => ['required', 'string', 'min:1', 'max:255', 'safe_url'],
153 } catch (ValidationException $exception) {
154 return response()->view('attachments.manager-link-form', array_merge($request->only(['attachment_link_name', 'attachment_link_url']), [
156 'errors' => new MessageBag($exception->errors()),
160 $page = $this->pageRepo->getById($pageId);
162 $this->checkPermission('attachment-create-all');
163 $this->checkOwnablePermission('page-update', $page);
165 $attachmentName = $request->get('attachment_link_name');
166 $link = $request->get('attachment_link_url');
167 $this->attachmentService->saveNewFromLink($attachmentName, $link, intval($pageId));
169 return view('attachments.manager-link-form', [
175 * Get the attachments for a specific page.
177 * @throws NotFoundException
179 public function listForPage(int $pageId)
181 $page = $this->pageRepo->getById($pageId);
182 $this->checkOwnablePermission('page-view', $page);
184 return view('attachments.manager-list', [
185 'attachments' => $page->attachments->all(),
190 * Update the attachment sorting.
192 * @throws ValidationException
193 * @throws NotFoundException
195 public function sortForPage(Request $request, int $pageId)
197 $this->validate($request, [
198 'order' => ['required', 'array'],
200 $page = $this->pageRepo->getById($pageId);
201 $this->checkOwnablePermission('page-update', $page);
203 $attachmentOrder = $request->get('order');
204 $this->attachmentService->updateFileOrderWithinPage($attachmentOrder, $pageId);
206 return response()->json(['message' => trans('entities.attachments_order_updated')]);
210 * Get an attachment from storage.
212 * @throws FileNotFoundException
213 * @throws NotFoundException
215 public function get(Request $request, string $attachmentId)
217 /** @var Attachment $attachment */
218 $attachment = Attachment::query()->findOrFail($attachmentId);
221 $page = $this->pageRepo->getById($attachment->uploaded_to);
222 } catch (NotFoundException $exception) {
223 throw new NotFoundException(trans('errors.attachment_not_found'));
226 $this->checkOwnablePermission('page-view', $page);
228 if ($attachment->external) {
229 return redirect($attachment->path);
232 $fileName = $attachment->getFileName();
233 $attachmentStream = $this->attachmentService->streamAttachmentFromStorage($attachment);
235 if ($request->get('open') === 'true') {
236 return $this->download()->streamedInline($attachmentStream, $fileName);
239 return $this->download()->streamedDirectly($attachmentStream, $fileName);
243 * Delete a specific attachment in the system.
247 public function delete(string $attachmentId)
249 /** @var Attachment $attachment */
250 $attachment = Attachment::query()->findOrFail($attachmentId);
251 $this->checkOwnablePermission('attachment-delete', $attachment);
252 $this->attachmentService->deleteFile($attachment);
254 return response()->json(['message' => trans('entities.attachments_deleted')]);