123 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			PHP
		
	
	
	
			
		
		
	
	
			123 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			PHP
		
	
	
	
| <?php
 | |
| 
 | |
| namespace BookStack\References;
 | |
| 
 | |
| use BookStack\Entities\Models\Book;
 | |
| use BookStack\Entities\Models\Entity;
 | |
| use BookStack\Entities\Models\Page;
 | |
| use BookStack\Entities\Repos\RevisionRepo;
 | |
| use DOMDocument;
 | |
| use DOMXPath;
 | |
| 
 | |
| class ReferenceUpdater
 | |
| {
 | |
|     protected ReferenceFetcher $referenceFetcher;
 | |
|     protected RevisionRepo $revisionRepo;
 | |
| 
 | |
|     public function __construct(ReferenceFetcher $referenceFetcher, RevisionRepo $revisionRepo)
 | |
|     {
 | |
|         $this->referenceFetcher = $referenceFetcher;
 | |
|         $this->revisionRepo = $revisionRepo;
 | |
|     }
 | |
| 
 | |
|     public function updateEntityPageReferences(Entity $entity, string $oldLink)
 | |
|     {
 | |
|         $references = $this->getReferencesToUpdate($entity);
 | |
|         $newLink = $entity->getUrl();
 | |
| 
 | |
|         /** @var Reference $reference */
 | |
|         foreach ($references as $reference) {
 | |
|             /** @var Page $page */
 | |
|             $page = $reference->from;
 | |
|             $this->updateReferencesWithinPage($page, $oldLink, $newLink);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return Reference[]
 | |
|      */
 | |
|     protected function getReferencesToUpdate(Entity $entity): array
 | |
|     {
 | |
|         /** @var Reference[] $references */
 | |
|         $references = $this->referenceFetcher->getPageReferencesToEntity($entity)->values()->all();
 | |
| 
 | |
|         if ($entity instanceof Book) {
 | |
|             $pages = $entity->pages()->get(['id']);
 | |
|             $chapters = $entity->chapters()->get(['id']);
 | |
|             $children = $pages->concat($chapters);
 | |
|             foreach ($children as $bookChild) {
 | |
|                 $childRefs = $this->referenceFetcher->getPageReferencesToEntity($bookChild)->values()->all();
 | |
|                 array_push($references, ...$childRefs);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         $deduped = [];
 | |
|         foreach ($references as $reference) {
 | |
|             $key = $reference->from_id . ':' . $reference->from_type;
 | |
|             $deduped[$key] = $reference;
 | |
|         }
 | |
| 
 | |
|         return array_values($deduped);
 | |
|     }
 | |
| 
 | |
|     protected function updateReferencesWithinPage(Page $page, string $oldLink, string $newLink)
 | |
|     {
 | |
|         $page = (clone $page)->refresh();
 | |
|         $html = $this->updateLinksInHtml($page->html, $oldLink, $newLink);
 | |
|         $markdown = $this->updateLinksInMarkdown($page->markdown, $oldLink, $newLink);
 | |
| 
 | |
|         $page->html = $html;
 | |
|         $page->markdown = $markdown;
 | |
|         $page->revision_count++;
 | |
|         $page->save();
 | |
| 
 | |
|         $summary = trans('entities.pages_references_update_revision');
 | |
|         $this->revisionRepo->storeNewForPage($page, $summary);
 | |
|     }
 | |
| 
 | |
|     protected function updateLinksInMarkdown(string $markdown, string $oldLink, string $newLink): string
 | |
|     {
 | |
|         if (empty($markdown)) {
 | |
|             return $markdown;
 | |
|         }
 | |
| 
 | |
|         $commonLinkRegex = '/(\[.*?\]\()' . preg_quote($oldLink, '/') . '(.*?\))/i';
 | |
|         $markdown = preg_replace($commonLinkRegex, '$1' . $newLink . '$2', $markdown);
 | |
| 
 | |
|         $referenceLinkRegex = '/(\[.*?\]:\s?)' . preg_quote($oldLink, '/') . '(.*?)($|\s)/i';
 | |
|         $markdown = preg_replace($referenceLinkRegex, '$1' . $newLink . '$2$3', $markdown);
 | |
| 
 | |
|         return $markdown;
 | |
|     }
 | |
| 
 | |
|     protected function updateLinksInHtml(string $html, string $oldLink, string $newLink): string
 | |
|     {
 | |
|         if (empty($html)) {
 | |
|             return $html;
 | |
|         }
 | |
| 
 | |
|         $html = '<body>' . $html . '</body>';
 | |
|         libxml_use_internal_errors(true);
 | |
|         $doc = new DOMDocument();
 | |
|         $doc->loadHTML(mb_convert_encoding($html, 'HTML-ENTITIES', 'UTF-8'));
 | |
| 
 | |
|         $xPath = new DOMXPath($doc);
 | |
|         $anchors = $xPath->query('//a[@href]');
 | |
| 
 | |
|         /** @var \DOMElement $anchor */
 | |
|         foreach ($anchors as $anchor) {
 | |
|             $link = $anchor->getAttribute('href');
 | |
|             $updated = str_ireplace($oldLink, $newLink, $link);
 | |
|             $anchor->setAttribute('href', $updated);
 | |
|         }
 | |
| 
 | |
|         $html = '';
 | |
|         $topElems = $doc->documentElement->childNodes->item(0)->childNodes;
 | |
|         foreach ($topElems as $child) {
 | |
|             $html .= $doc->saveHTML($child);
 | |
|         }
 | |
| 
 | |
|         return $html;
 | |
|     }
 | |
| }
 |