From c5e68dd2c0c3bbc9742b4411738581741c004ede Mon Sep 17 00:00:00 2001 From: Sergey Shevchenko Date: Mon, 1 Nov 2021 14:16:00 +1300 Subject: [PATCH] ENH: resolve relative references in CSS files when combining --- src/View/Requirements_Backend.php | 32 +++++++++++++++++++++++++++++++ 1 file changed, 32 insertions(+) diff --git a/src/View/Requirements_Backend.php b/src/View/Requirements_Backend.php index 5c5c9ddda..9d6c79494 100644 --- a/src/View/Requirements_Backend.php +++ b/src/View/Requirements_Backend.php @@ -1395,6 +1395,10 @@ MESSAGE throw new InvalidArgumentException("Combined file {$file} does not exist"); } $fileContent = file_get_contents($filePath ?? ''); + if ($type == 'css') { + // resolve relative paths for css files + $fileContent = $this->resolveCSSReferences($fileContent, $file); + } // Use configured minifier if ($minify) { $fileContent = $this->minifier->minify($fileContent, $type, $file); @@ -1421,6 +1425,34 @@ MESSAGE return $combinedURL; } + /** + * Resolves relative paths in CSS files which are lost when combining them + * + * @param string $content + * @param string $filePath + * @return string New content with paths resolved + */ + protected function resolveCSSReferences($content, $filePath) + { + $fileUrl = Injector::inst()->get(ResourceURLGenerator::class)->urlForResource($filePath); + $fileUrlDir = dirname($fileUrl); + $content = preg_replace_callback('#([\.]{1,2}/)+#', function ($a) use ($fileUrlDir) { + $full = $fileUrlDir . '/' . $a[0]; + $full = preg_replace('#/{2,}#', '/', $full); // ensure there's no repeated slashes + while (strpos($full, './') > 0) { + $post = $full; + $post = preg_replace('#([^/\.]+)/\.\./#', '', $post); // erase 'something/../' with the predecessor + $post = preg_replace('#([^/\.]+)/\./#', '\\1/', $post); // erase './' + if ($post == $full) { + break; // nothing changed + } + $full = $post; + } + return $full; + }, $content); + return $content; + } + /** * Given a filename and list of files, generate a new filename unique to these files *