mirror of
				https://github.com/BookStackApp/BookStack.git
				synced 2025-10-25 06:37:36 +03:00 
			
		
		
		
	Creates a new organsied formatting system for webhook data, with interfaces for extending with custom model formatting rules. Allows easy usage & extension of the default bookstack formatting behaviour when customizing webhook events via theme system, and keeps default data customizations organised. This also makes the following webhook data changes: - owned_by/created_by/updated_by user details are loaded for events with Entity details. (POTENTIALLY BREAKING CHANGE). - current_revision details are loaded for page update/create events. Added testing to cover added model formatting rules. For #3279 and #3218
		
			
				
	
	
		
			140 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			140 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| namespace BookStack\Http\Controllers;
 | |
| 
 | |
| use BookStack\Entities\Repos\PageRepo;
 | |
| use BookStack\Entities\Tools\PageContent;
 | |
| use BookStack\Exceptions\NotFoundException;
 | |
| use Ssddanbrown\HtmlDiff\Diff;
 | |
| 
 | |
| class PageRevisionController extends Controller
 | |
| {
 | |
|     protected $pageRepo;
 | |
| 
 | |
|     /**
 | |
|      * PageRevisionController constructor.
 | |
|      */
 | |
|     public function __construct(PageRepo $pageRepo)
 | |
|     {
 | |
|         $this->pageRepo = $pageRepo;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Shows the last revisions for this page.
 | |
|      *
 | |
|      * @throws NotFoundException
 | |
|      */
 | |
|     public function index(string $bookSlug, string $pageSlug)
 | |
|     {
 | |
|         $page = $this->pageRepo->getBySlug($bookSlug, $pageSlug);
 | |
|         $this->setPageTitle(trans('entities.pages_revisions_named', ['pageName'=>$page->getShortName()]));
 | |
| 
 | |
|         return view('pages.revisions', [
 | |
|             'page'    => $page,
 | |
|             'current' => $page,
 | |
|         ]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Shows a preview of a single revision.
 | |
|      *
 | |
|      * @throws NotFoundException
 | |
|      */
 | |
|     public function show(string $bookSlug, string $pageSlug, int $revisionId)
 | |
|     {
 | |
|         $page = $this->pageRepo->getBySlug($bookSlug, $pageSlug);
 | |
|         $revision = $page->revisions()->where('id', '=', $revisionId)->first();
 | |
|         if ($revision === null) {
 | |
|             throw new NotFoundException();
 | |
|         }
 | |
| 
 | |
|         $page->fill($revision->toArray());
 | |
|         // TODO - Refactor PageContent so we don't need to juggle this
 | |
|         $page->html = $revision->html;
 | |
|         $page->html = (new PageContent($page))->render();
 | |
| 
 | |
|         $this->setPageTitle(trans('entities.pages_revision_named', ['pageName' => $page->getShortName()]));
 | |
| 
 | |
|         return view('pages.revision', [
 | |
|             'page'     => $page,
 | |
|             'book'     => $page->book,
 | |
|             'diff'     => null,
 | |
|             'revision' => $revision,
 | |
|         ]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Shows the changes of a single revision.
 | |
|      *
 | |
|      * @throws NotFoundException
 | |
|      */
 | |
|     public function changes(string $bookSlug, string $pageSlug, int $revisionId)
 | |
|     {
 | |
|         $page = $this->pageRepo->getBySlug($bookSlug, $pageSlug);
 | |
|         $revision = $page->revisions()->where('id', '=', $revisionId)->first();
 | |
|         if ($revision === null) {
 | |
|             throw new NotFoundException();
 | |
|         }
 | |
| 
 | |
|         $prev = $revision->getPrevious();
 | |
|         $prevContent = $prev->html ?? '';
 | |
|         $diff = Diff::excecute($prevContent, $revision->html);
 | |
| 
 | |
|         $page->fill($revision->toArray());
 | |
|         // TODO - Refactor PageContent so we don't need to juggle this
 | |
|         $page->html = $revision->html;
 | |
|         $page->html = (new PageContent($page))->render();
 | |
|         $this->setPageTitle(trans('entities.pages_revision_named', ['pageName'=>$page->getShortName()]));
 | |
| 
 | |
|         return view('pages.revision', [
 | |
|             'page'     => $page,
 | |
|             'book'     => $page->book,
 | |
|             'diff'     => $diff,
 | |
|             'revision' => $revision,
 | |
|         ]);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Restores a page using the content of the specified revision.
 | |
|      *
 | |
|      * @throws NotFoundException
 | |
|      */
 | |
|     public function restore(string $bookSlug, string $pageSlug, int $revisionId)
 | |
|     {
 | |
|         $page = $this->pageRepo->getBySlug($bookSlug, $pageSlug);
 | |
|         $this->checkOwnablePermission('page-update', $page);
 | |
| 
 | |
|         $page = $this->pageRepo->restoreRevision($page, $revisionId);
 | |
| 
 | |
|         return redirect($page->getUrl());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Deletes a revision using the id of the specified revision.
 | |
|      *
 | |
|      * @throws NotFoundException
 | |
|      */
 | |
|     public function destroy(string $bookSlug, string $pageSlug, int $revId)
 | |
|     {
 | |
|         $page = $this->pageRepo->getBySlug($bookSlug, $pageSlug);
 | |
|         $this->checkOwnablePermission('page-delete', $page);
 | |
| 
 | |
|         $revision = $page->revisions()->where('id', '=', $revId)->first();
 | |
|         if ($revision === null) {
 | |
|             throw new NotFoundException("Revision #{$revId} not found");
 | |
|         }
 | |
| 
 | |
|         // Check if it's the latest revision, cannot delete the latest revision.
 | |
|         if (intval($page->currentRevision->id ?? null) === intval($revId)) {
 | |
|             $this->showErrorNotification(trans('entities.revision_cannot_delete_latest'));
 | |
| 
 | |
|             return redirect($page->getUrl('/revisions'));
 | |
|         }
 | |
| 
 | |
|         $revision->delete();
 | |
|         $this->showSuccessNotification(trans('entities.revision_delete_success'));
 | |
| 
 | |
|         return redirect($page->getUrl('/revisions'));
 | |
|     }
 | |
| }
 |