Ben Thomson 805b95b490
Allow decompilation of local assets that match local URL
https://github.com/wintercms/winter/pull/552 introduced a change where all backend assets are converted to URLs to provide support for the `app.asset_url` configuration variable. This change allows asset decompilation to still work by converting a (local) URL back to a relative path. If a URL doesn't match the root URL, then it is passed through as is with no decompilation taking place.

Fixes https://github.com/wintercms/winter/issues/562.
2022-05-25 16:18:55 +08:00

283 lines
8.5 KiB
PHP

<?php namespace Backend\Helpers;
use Url;
use File;
use Html;
use Config;
use Request;
use Redirect;
use Exception;
use Winter\Storm\Router\Helper as RouterHelper;
use System\Helpers\DateTime as DateTimeHelper;
use Backend\Classes\Skin;
use Backend\Helpers\Exception\DecompileException;
/**
* Backend Helper
*
* @package winter\wn-backend-module
* @see \Backend\Facades\Backend
* @author Alexey Bobkov, Samuel Georges
*/
class Backend
{
/**
* Returns the backend URI segment.
*/
public function uri()
{
return Config::get('cms.backendUri', 'backend');
}
/**
* Returns a URL in context of the Backend
*/
public function url($path = null, $parameters = [], $secure = null)
{
return Url::to($this->uri() . '/' . $path, $parameters, $secure);
}
/**
* Returns the base backend URL
*/
public function baseUrl($path = null)
{
$backendUri = $this->uri();
$baseUrl = Request::getBaseUrl();
if ($path === null) {
return $baseUrl . '/' . $backendUri;
}
$path = RouterHelper::normalizeUrl($path);
return $baseUrl . '/' . $backendUri . $path;
}
/**
* Returns a URL in context of the active Backend skin
*/
public function skinAsset($path = null)
{
$skinPath = Skin::getActive()->getPath($path, true);
return Url::asset($skinPath);
}
/**
* Create a new redirect response to a given backend path.
*/
public function redirect($path, $status = 302, $headers = [], $secure = null)
{
return Redirect::to($this->uri() . '/' . $path, $status, $headers, $secure);
}
/**
* Create a new backend redirect response, while putting the current URL in the session.
*/
public function redirectGuest($path, $status = 302, $headers = [], $secure = null)
{
return Redirect::guest($this->uri() . '/' . $path, $status, $headers, $secure);
}
/**
* Create a new redirect response to the previously intended backend location.
*/
public function redirectIntended($path, $status = 302, $headers = [], $secure = null)
{
return Redirect::intended($this->uri() . '/' . $path, $status, $headers, $secure);
}
/**
* Convert mixed inputs to a Carbon object and sets the backend timezone on that object
*
* @return \Carbon\Carbon
*/
public static function makeCarbon($value, $throwException = true)
{
$carbon = DateTimeHelper::makeCarbon($value, $throwException);
try {
// Find user preference
$carbon->setTimezone(\Backend\Models\Preference::get('timezone'));
} catch (Exception $ex) {
// Use system default
$carbon->setTimezone(Config::get('cms.backendTimezone', Config::get('app.timezone')));
}
return $carbon;
}
/**
* Proxy method for dateTime() using "date" format alias.
* @return string
*/
public function date($dateTime, $options = [])
{
return $this->dateTime($dateTime, $options + ['formatAlias' => 'date']);
}
/**
* Returns the HTML for a date formatted in the backend.
* Supported for formatAlias:
* time -> 6:28 AM
* timeLong -> 6:28:01 AM
* date -> 04/23/2016
* dateMin -> 4/23/2016
* dateLong -> April 23, 2016
* dateLongMin -> Apr 23, 2016
* dateTime -> April 23, 2016 6:28 AM
* dateTimeMin -> Apr 23, 2016 6:28 AM
* dateTimeLong -> Saturday, April 23, 2016 6:28 AM
* dateTimeLongMin -> Sat, Apr 23, 2016 6:29 AM
* @return string
*/
public function dateTime($dateTime, $options = [])
{
extract(array_merge([
'defaultValue' => '',
'format' => null,
'formatAlias' => null,
'jsFormat' => null,
'timeTense' => false,
'timeSince' => false,
'ignoreTimezone' => false,
], $options));
if (!$dateTime) {
return '';
}
$carbon = DateTimeHelper::makeCarbon($dateTime);
if ($jsFormat !== null) {
$format = $jsFormat;
}
else {
$format = DateTimeHelper::momentFormat($format);
}
$attributes = [
'datetime' => $carbon,
'data-datetime-control' => 1,
];
if ($ignoreTimezone) {
$attributes['data-ignore-timezone'] = true;
}
if ($timeTense) {
$attributes['data-time-tense'] = 1;
}
elseif ($timeSince) {
$attributes['data-time-since'] = 1;
}
elseif ($format) {
$attributes['data-format'] = $format;
}
elseif ($formatAlias) {
$attributes['data-format-alias'] = $formatAlias;
}
return '<time'.Html::attributes($attributes).'>'.e($defaultValue).'</time>'.PHP_EOL;
}
/**
* Decompiles the compilation asset files
*
* This is used to load each individual asset file, as opposed to using the compilation assets. This is useful only
* for development, to allow developers to test changes without having to re-compile assets.
*
* @param string $file The compilation asset file to decompile
* @param boolean $skinAsset If true, will load decompiled assets from the "skins" directory.
* @throws DecompileException If the compilation file cannot be decompiled
* @return array
*/
public function decompileAsset(string $file, bool $skinAsset = false)
{
$assets = $this->parseAsset($file, $skinAsset);
if (!$assets) {
// Return URL-based assets as is
if (starts_with($file, ['https://', 'http://'])) {
return [$file];
}
// Resolve relative asset paths
if ($skinAsset) {
$assetPath = base_path(substr(Skin::getActive()->getPath($file, true), 1));
} else {
$assetPath = base_path($file);
}
$relativePath = File::localToPublic(realpath($assetPath));
return [Url::asset($relativePath)];
}
return array_map(function ($asset) use ($skinAsset) {
// Resolve relative asset paths
if ($skinAsset) {
$assetPath = base_path(substr(Skin::getActive()->getPath($asset, true), 1));
} else {
$assetPath = base_path($asset);
}
$relativePath = File::localToPublic(realpath($assetPath));
return Url::asset($relativePath);
}, $assets);
}
/**
* Parse the provided asset file to get the files that it includes
*
* @param string $file The compilation asset file to parse
* @param boolean $skinAsset If true, will load decompiled assets from the "skins" directory.
* @return array
*/
protected function parseAsset($file, $skinAsset)
{
if (starts_with($file, ['https://', 'http://'])) {
$rootUrl = Url::to('/');
if (!starts_with($file, $rootUrl)) {
return false;
}
$file = str_replace($rootUrl, '', $file);
}
if ($skinAsset) {
$assetFile = base_path(substr(Skin::getActive()->getPath($file, true), 1));
} else {
$assetFile = base_path($file);
}
$results = [$file];
if (!file_exists($assetFile)) {
throw new DecompileException('File ' . $file . ' does not exist to be decompiled.');
}
if (!is_readable($assetFile)) {
throw new DecompileException('File ' . $file . ' cannot be decompiled. Please allow read access to the file.');
}
$contents = file_get_contents($assetFile);
// Find all assets that are compiled in this file
preg_match_all('/^=require\s+([A-z0-9-_+\.\/]+)[\n|\r\n|$]/m', $contents, $matches, PREG_SET_ORDER);
// Determine correct asset path
$directory = str_replace(basename($file), '', $file);
if (count($matches)) {
$results = array_map(function ($match) use ($directory) {
return str_replace('/', DIRECTORY_SEPARATOR, $directory . $match[1]);
}, $matches);
foreach ($results as $i => $result) {
$nested = $this->parseAsset($result, $skinAsset);
array_splice($results, $i, 1, $nested);
}
}
return $results;
}
}