derivato da rebillar/site-accueil-insa
		
	
		
			
				
	
	
		
			521 righe
		
	
	
	
		
			16 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			521 righe
		
	
	
	
		
			16 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * Matomo - free/libre analytics platform
 | |
|  *
 | |
|  * @link https://matomo.org
 | |
|  * @license http://www.gnu.org/licenses/gpl-3.0.html GPL v3 or later
 | |
|  *
 | |
|  */
 | |
| namespace Piwik;
 | |
| 
 | |
| use Exception;
 | |
| use Piwik\AssetManager\UIAsset;
 | |
| use Piwik\AssetManager\UIAsset\InMemoryUIAsset;
 | |
| use Piwik\AssetManager\UIAsset\OnDiskUIAsset;
 | |
| use Piwik\AssetManager\UIAssetCacheBuster;
 | |
| use Piwik\AssetManager\UIAssetFetcher\JScriptUIAssetFetcher;
 | |
| use Piwik\AssetManager\UIAssetFetcher\StaticUIAssetFetcher;
 | |
| use Piwik\AssetManager\UIAssetFetcher\StylesheetUIAssetFetcher;
 | |
| use Piwik\AssetManager\UIAssetFetcher\PluginUmdAssetFetcher;
 | |
| use Piwik\AssetManager\UIAssetFetcher;
 | |
| use Piwik\AssetManager\UIAssetMerger\JScriptUIAssetMerger;
 | |
| use Piwik\AssetManager\UIAssetMerger\StylesheetUIAssetMerger;
 | |
| use Piwik\Container\StaticContainer;
 | |
| use Piwik\Plugin\Manager;
 | |
| 
 | |
| /**
 | |
|  * AssetManager is the class used to manage the inclusion of UI assets:
 | |
|  * JavaScript and CSS files.
 | |
|  *
 | |
|  * It performs the following actions:
 | |
|  *  - Identifies required assets
 | |
|  *  - Includes assets in the rendered HTML page
 | |
|  *  - Manages asset merging and minifying
 | |
|  *  - Manages server-side cache
 | |
|  *
 | |
|  * Whether assets are included individually or as merged files is defined by
 | |
|  * the global option 'disable_merged_assets'. See the documentation in the global
 | |
|  * config for more information.
 | |
|  */
 | |
| class AssetManager extends Singleton
 | |
| {
 | |
|     const MERGED_CSS_FILE = "asset_manager_global_css.css";
 | |
|     const MERGED_CORE_JS_FILE = "asset_manager_core_js.js";
 | |
|     const MERGED_NON_CORE_JS_FILE = "asset_manager_non_core_js.js";
 | |
| 
 | |
|     const CSS_IMPORT_DIRECTIVE = "<link rel=\"stylesheet\" type=\"text/css\" href=\"%s\" />\n";
 | |
|     const JS_IMPORT_DIRECTIVE = "<script type=\"text/javascript\" src=\"%s\"></script>\n";
 | |
|     const JS_DEFER_IMPORT_DIRECTIVE = "<script type=\"text/javascript\" src=\"%s\" defer></script>\n";
 | |
|     const GET_CSS_MODULE_ACTION = "index.php?module=Proxy&action=getCss";
 | |
|     const GET_CORE_JS_MODULE_ACTION = "index.php?module=Proxy&action=getCoreJs";
 | |
|     const GET_NON_CORE_JS_MODULE_ACTION = "index.php?module=Proxy&action=getNonCoreJs";
 | |
|     const GET_JS_UMD_MODULE_ACTION = "index.php?module=Proxy&action=getUmdJs&chunk=";
 | |
| 
 | |
|     /**
 | |
|      * @var UIAssetCacheBuster
 | |
|      */
 | |
|     private $cacheBuster;
 | |
| 
 | |
|     /**
 | |
|      * @var UIAssetFetcher
 | |
|      */
 | |
|     private $minimalStylesheetFetcher;
 | |
| 
 | |
|     /**
 | |
|      * @var Theme
 | |
|      */
 | |
|     private $theme;
 | |
| 
 | |
|     public function __construct()
 | |
|     {
 | |
|         $this->cacheBuster = UIAssetCacheBuster::getInstance();
 | |
| 
 | |
|         $this->minimalStylesheetFetcher = new StaticUIAssetFetcher(array(), array(), $this->theme);
 | |
| 
 | |
|         $theme = Manager::getInstance()->getThemeEnabled();
 | |
|         if (!empty($theme)) {
 | |
|             $this->theme = new Theme();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @inheritDoc
 | |
|      * @return AssetManager
 | |
|      */
 | |
|     public static function getInstance()
 | |
|     {
 | |
|         $assetManager = parent::getInstance();
 | |
| 
 | |
|         /**
 | |
|          * Triggered when creating an instance of the asset manager. Lets you overwrite the
 | |
|          * asset manager behavior.
 | |
|          *
 | |
|          * @param AssetManager &$assetManager
 | |
|          *
 | |
|          * @ignore
 | |
|          * This event is not a public event since we don't want to make the asset manager itself public
 | |
|          * API
 | |
|          */
 | |
|         Piwik::postEvent('AssetManager.makeNewAssetManagerObject', array(&$assetManager));
 | |
| 
 | |
|         return $assetManager;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param UIAssetCacheBuster $cacheBuster
 | |
|      */
 | |
|     public function setCacheBuster($cacheBuster)
 | |
|     {
 | |
|         $this->cacheBuster = $cacheBuster;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param UIAssetFetcher $minimalStylesheetFetcher
 | |
|      */
 | |
|     public function setMinimalStylesheetFetcher($minimalStylesheetFetcher)
 | |
|     {
 | |
|         $this->minimalStylesheetFetcher = $minimalStylesheetFetcher;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param Theme $theme
 | |
|      */
 | |
|     public function setTheme($theme)
 | |
|     {
 | |
|         $this->theme = $theme;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Return CSS file inclusion directive(s) using the markup <link>
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     public function getCssInclusionDirective()
 | |
|     {
 | |
|         return sprintf(self::CSS_IMPORT_DIRECTIVE, self::GET_CSS_MODULE_ACTION);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Return JS file inclusion directive(s) using the markup <script>
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     public function getJsInclusionDirective()
 | |
|     {
 | |
|         $result = "<script type=\"text/javascript\">\n" . StaticContainer::get('Piwik\Translation\Translator')->getJavascriptTranslations() . "\n</script>";
 | |
| 
 | |
|         if ($this->isMergedAssetsDisabled()) {
 | |
|             $this->getMergedCoreJSAsset()->delete();
 | |
|             $this->getMergedNonCoreJSAsset()->delete();
 | |
| 
 | |
|             $result .= $this->getIndividualCoreAndNonCoreJsIncludes();
 | |
|         } else {
 | |
|             $result .= sprintf(self::JS_IMPORT_DIRECTIVE, self::GET_CORE_JS_MODULE_ACTION);
 | |
|             $result .= sprintf(self::JS_IMPORT_DIRECTIVE, self::GET_NON_CORE_JS_MODULE_ACTION);
 | |
| 
 | |
|             $result .= $this->getPluginUmdChunks();
 | |
|         }
 | |
| 
 | |
|         return $result;
 | |
|     }
 | |
| 
 | |
|     protected function getPluginUmdChunks()
 | |
|     {
 | |
|         $fetcher = $this->getPluginUmdJScriptFetcher();
 | |
| 
 | |
|         $chunks = $fetcher->getChunkFiles();
 | |
| 
 | |
|         $result = '';
 | |
|         foreach ($chunks as $chunk) {
 | |
|             $src = self::GET_JS_UMD_MODULE_ACTION . urlencode($chunk->getChunkName());
 | |
|             $result .= sprintf(self::JS_DEFER_IMPORT_DIRECTIVE, $src);
 | |
|         }
 | |
|         return $result;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Return the base.less compiled to css
 | |
|      *
 | |
|      * @return UIAsset
 | |
|      */
 | |
|     public function getCompiledBaseCss()
 | |
|     {
 | |
|         $mergedAsset = new InMemoryUIAsset();
 | |
| 
 | |
|         $assetMerger = new StylesheetUIAssetMerger($mergedAsset, $this->minimalStylesheetFetcher, $this->cacheBuster);
 | |
| 
 | |
|         $assetMerger->generateFile();
 | |
| 
 | |
|         return $mergedAsset;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Return the css merged file absolute location.
 | |
|      * If there is none, the generation process will be triggered.
 | |
|      *
 | |
|      * @return UIAsset
 | |
|      */
 | |
|     public function getMergedStylesheet()
 | |
|     {
 | |
|         $mergedAsset = $this->getMergedStylesheetAsset();
 | |
| 
 | |
|         $assetFetcher = new StylesheetUIAssetFetcher(Manager::getInstance()->getLoadedPluginsName(), $this->theme);
 | |
| 
 | |
|         $assetMerger = new StylesheetUIAssetMerger($mergedAsset, $assetFetcher, $this->cacheBuster);
 | |
| 
 | |
|         $assetMerger->generateFile();
 | |
| 
 | |
|         return $mergedAsset;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Return the core js merged file absolute location.
 | |
|      * If there is none, the generation process will be triggered.
 | |
|      *
 | |
|      * @return UIAsset
 | |
|      */
 | |
|     public function getMergedCoreJavaScript()
 | |
|     {
 | |
|         return $this->getMergedJavascript($this->getCoreJScriptFetcher(), $this->getMergedCoreJSAsset());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Return the non core js merged file absolute location.
 | |
|      * If there is none, the generation process will be triggered.
 | |
|      *
 | |
|      * @return UIAsset
 | |
|      */
 | |
|     public function getMergedNonCoreJavaScript()
 | |
|     {
 | |
|         return $this->getMergedJavascript($this->getNonCoreJScriptFetcher(), $this->getMergedNonCoreJSAsset());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Return a chunk JS merged file absolute location.
 | |
|      * If there is none, the generation process will be triggered.
 | |
|      *
 | |
|      * @param string $chunk The name of the chunk. Will either be a plugin name or an integer.
 | |
|      * @return UIAsset
 | |
|      */
 | |
|     public function getMergedJavaScriptChunk($chunk)
 | |
|     {
 | |
|         $assetFetcher = $this->getPluginUmdJScriptFetcher($chunk);
 | |
|         $outputFile = $assetFetcher->getRequestedChunkOutputFile();
 | |
| 
 | |
|         return $this->getMergedJavascript($assetFetcher, $this->getMergedUIAsset($outputFile));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param boolean|"all" $core
 | |
|      * @return string[]
 | |
|      */
 | |
|     public function getLoadedPlugins($core)
 | |
|     {
 | |
|         $loadedPlugins = array();
 | |
| 
 | |
|         foreach (Manager::getInstance()->getPluginsLoadedAndActivated() as $plugin) {
 | |
|             $pluginName = $plugin->getPluginName();
 | |
|             $pluginIsCore = Manager::getInstance()->isPluginBundledWithCore($pluginName);
 | |
| 
 | |
|             if ($core === 'all' || ($pluginIsCore && $core) || (!$pluginIsCore && !$core)) {
 | |
|                 $loadedPlugins[] = $pluginName;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return $loadedPlugins;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Remove previous merged assets
 | |
|      */
 | |
|     public function removeMergedAssets($pluginName = false)
 | |
|     {
 | |
|         $assetsToRemove = array($this->getMergedStylesheetAsset());
 | |
| 
 | |
|         if ($pluginName) {
 | |
|             if ($this->pluginContainsJScriptAssets($pluginName)) {
 | |
|                 if (Manager::getInstance()->isPluginBundledWithCore($pluginName)) {
 | |
|                     $assetsToRemove[] = $this->getMergedCoreJSAsset();
 | |
|                 } else {
 | |
|                     $assetsToRemove[] = $this->getMergedNonCoreJSAsset();
 | |
|                 }
 | |
| 
 | |
|                 $assetFetcher = $this->getPluginUmdJScriptFetcher();
 | |
|                 foreach ($assetFetcher->getChunkFiles() as $chunk) {
 | |
|                     $files = $chunk->getFiles();
 | |
| 
 | |
|                     $foundInChunk = false;
 | |
|                     foreach ($files as $file) {
 | |
|                         if (strpos($file, "/$pluginName.umd.") !== false) {
 | |
|                             $foundInChunk = true;
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     if ($foundInChunk) {
 | |
|                         $outputFile = $chunk->getOutputFile();
 | |
|                         $asset = $this->getMergedUIAsset($outputFile);
 | |
|                         if ($asset->exists()) {
 | |
|                             $assetsToRemove[] = $asset;
 | |
|                         }
 | |
|                         break;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         } else {
 | |
|             $assetsToRemove[] = $this->getMergedCoreJSAsset();
 | |
|             $assetsToRemove[] = $this->getMergedNonCoreJSAsset();
 | |
| 
 | |
|             $assetFetcher = $this->getPluginUmdJScriptFetcher();
 | |
|             foreach ($assetFetcher->getChunkFiles() as $chunk) {
 | |
|                 $outputFile = $chunk->getOutputFile();
 | |
|                 $asset = $this->getMergedUIAsset($outputFile);
 | |
|                 if ($asset->exists()) {
 | |
|                     $assetsToRemove[] = $asset;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         $this->removeAssets($assetsToRemove);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Check if the merged file directory exists and is writable.
 | |
|      *
 | |
|      * @return string The directory location
 | |
|      * @throws Exception if directory is not writable.
 | |
|      */
 | |
|     public function getAssetDirectory()
 | |
|     {
 | |
|         $mergedFileDirectory = StaticContainer::get('path.tmp') . '/assets';
 | |
| 
 | |
|         if (!is_dir($mergedFileDirectory)) {
 | |
|             Filesystem::mkdir($mergedFileDirectory);
 | |
|         }
 | |
| 
 | |
|         if (!is_writable($mergedFileDirectory)) {
 | |
|             throw new Exception("Directory " . $mergedFileDirectory . " has to be writable.");
 | |
|         }
 | |
| 
 | |
|         return $mergedFileDirectory;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Return the global option disable_merged_assets
 | |
|      *
 | |
|      * @return boolean
 | |
|      */
 | |
|     public function isMergedAssetsDisabled()
 | |
|     {
 | |
|         if (Config::getInstance()->Development['disable_merged_assets'] == 1) {
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         if (isset($_GET['disable_merged_assets']) && $_GET['disable_merged_assets'] == 1) {
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param UIAssetFetcher $assetFetcher
 | |
|      * @param UIAsset $mergedAsset
 | |
|      * @return UIAsset
 | |
|      */
 | |
|     private function getMergedJavascript($assetFetcher, $mergedAsset)
 | |
|     {
 | |
|         $assetMerger = new JScriptUIAssetMerger($mergedAsset, $assetFetcher, $this->cacheBuster);
 | |
| 
 | |
|         $assetMerger->generateFile();
 | |
| 
 | |
|         return $mergedAsset;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Return individual JS file inclusion directive(s) using the markup <script>
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     protected function getIndividualCoreAndNonCoreJsIncludes()
 | |
|     {
 | |
|         return
 | |
|             $this->getIndividualJsIncludesFromAssetFetcher($this->getCoreJScriptFetcher()) .
 | |
|             $this->getIndividualJsIncludesFromAssetFetcher($this->getNonCoreJScriptFetcher()) .
 | |
|             $this->getIndividualJsIncludesFromAssetFetcher($this->getPluginUmdJScriptFetcher());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param UIAssetFetcher $assetFetcher
 | |
|      * @return string
 | |
|      */
 | |
|     protected function getIndividualJsIncludesFromAssetFetcher($assetFetcher)
 | |
|     {
 | |
|         $jsIncludeString = '';
 | |
| 
 | |
|         $assets = $assetFetcher->getCatalog()->getAssets();
 | |
| 
 | |
|         foreach ($assets as $jsFile) {
 | |
|             $jsFile->validateFile();
 | |
|             $jsIncludeString = $jsIncludeString . sprintf(self::JS_IMPORT_DIRECTIVE, $jsFile->getRelativeLocation());
 | |
|         }
 | |
| 
 | |
|         return $jsIncludeString;
 | |
|     }
 | |
| 
 | |
|     private function getCoreJScriptFetcher()
 | |
|     {
 | |
|         return new JScriptUIAssetFetcher($this->getLoadedPlugins(true), $this->theme);
 | |
|     }
 | |
| 
 | |
|     protected function getNonCoreJScriptFetcher()
 | |
|     {
 | |
|         return new JScriptUIAssetFetcher($this->getLoadedPlugins(false), $this->theme);
 | |
|     }
 | |
| 
 | |
|     protected function getPluginUmdJScriptFetcher($chunk = null)
 | |
|     {
 | |
|         return new PluginUmdAssetFetcher($this->getLoadedPlugins('all'), $this->theme, $chunk);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param string $pluginName
 | |
|      * @return boolean
 | |
|      */
 | |
|     private function pluginContainsJScriptAssets($pluginName)
 | |
|     {
 | |
|         $fetcher = new JScriptUIAssetFetcher(array($pluginName), $this->theme);
 | |
| 
 | |
|         try {
 | |
|             $assets = $fetcher->getCatalog()->getAssets();
 | |
|         } catch (\Exception $e) {
 | |
|             // This can happen when a plugin is not valid (eg. Piwik 1.x format)
 | |
|             // When posting the event to the plugin, it returns an exception "Plugin has not been loaded"
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         $pluginManager = Manager::getInstance();
 | |
|         $plugin = null;
 | |
|         if ($pluginManager->isPluginLoaded($pluginName)) {
 | |
|             $plugin = $pluginManager->getLoadedPlugin($pluginName);
 | |
|         }
 | |
| 
 | |
|         if ($plugin && $plugin->isTheme()) {
 | |
|             $theme = $pluginManager->getTheme($pluginName);
 | |
| 
 | |
|             $javaScriptFiles = $theme->getJavaScriptFiles();
 | |
| 
 | |
|             if (!empty($javaScriptFiles)) {
 | |
|                 $assets = array_merge($assets, $javaScriptFiles);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return !empty($assets);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param UIAsset[] $uiAssets
 | |
|      */
 | |
|     public function removeAssets($uiAssets)
 | |
|     {
 | |
|         foreach ($uiAssets as $uiAsset) {
 | |
|             $uiAsset->delete();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return UIAsset
 | |
|      */
 | |
|     public function getMergedStylesheetAsset()
 | |
|     {
 | |
|         return $this->getMergedUIAsset(self::MERGED_CSS_FILE);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return UIAsset
 | |
|      */
 | |
|     private function getMergedCoreJSAsset()
 | |
|     {
 | |
|         return $this->getMergedUIAsset(self::MERGED_CORE_JS_FILE);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return UIAsset
 | |
|      */
 | |
|     protected function getMergedNonCoreJSAsset()
 | |
|     {
 | |
|         return $this->getMergedUIAsset(self::MERGED_NON_CORE_JS_FILE);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param string $fileName
 | |
|      * @return UIAsset
 | |
|      */
 | |
|     private function getMergedUIAsset($fileName)
 | |
|     {
 | |
|         return new OnDiskUIAsset($this->getAssetDirectory(), $fileName);
 | |
|     }
 | |
| 
 | |
|     public static function compileCustomStylesheets($files)
 | |
|     {
 | |
|         $assetManager = new AssetManager();
 | |
| 
 | |
|         $fetcher = new StaticUIAssetFetcher($files, $priorityOrder = array(), $theme = null);
 | |
| 
 | |
|         $assetManager->setMinimalStylesheetFetcher($fetcher);
 | |
| 
 | |
|         return $assetManager->getCompiledBaseCss()->getContent();
 | |
|     }
 | |
| 
 | |
|     public static function compileCustomJs($files)
 | |
|     {
 | |
|         $mergedAsset = new InMemoryUIAsset();
 | |
|         $fetcher = new StaticUIAssetFetcher($files, $priorityOrder = array(), $theme = null);
 | |
| 
 | |
|         $cacheBuster = UIAssetCacheBuster::getInstance();
 | |
| 
 | |
|         $assetMerger = new JScriptUIAssetMerger($mergedAsset, $fetcher, $cacheBuster);
 | |
|         $assetMerger->generateFile();
 | |
| 
 | |
|         return $mergedAsset->getContent();
 | |
|     }
 | |
| }
 |