. /** * This file is responsible for serving the one huge CSS of each theme. * * @package core * @copyright 2009 Petr Skoda (skodak) {@link http://skodak.org} * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ // Disable moodle specific debug messages and any errors in output, // comment out when debugging or better look into error log! define('NO_DEBUG_DISPLAY', true); define('ABORT_AFTER_CONFIG', true); require('../config.php'); require_once($CFG->dirroot.'/lib/csslib.php'); if ($slashargument = min_get_slash_argument()) { $slashargument = ltrim($slashargument, '/'); if (substr_count($slashargument, '/') < 2) { css_send_css_not_found(); } if (strpos($slashargument, '_s/') === 0) { // Can't use SVG. $slashargument = substr($slashargument, 3); $usesvg = false; } else { $usesvg = true; } $chunk = null; if (preg_match('#/(chunk(\d+)(/|$))#', $slashargument, $matches)) { $chunk = (int)$matches[2]; $slashargument = str_replace($matches[1], '', $slashargument); } list($themename, $rev, $type) = explode('/', $slashargument, 3); $themename = min_clean_param($themename, 'SAFEDIR'); $rev = min_clean_param($rev, 'INT'); $type = min_clean_param($type, 'SAFEDIR'); } else { $themename = min_optional_param('theme', 'standard', 'SAFEDIR'); $rev = min_optional_param('rev', 0, 'INT'); $type = min_optional_param('type', 'all', 'SAFEDIR'); $chunk = min_optional_param('chunk', null, 'INT'); $usesvg = (bool)min_optional_param('svg', '1', 'INT'); } if ($type === 'editor') { // The editor CSS is never chunked. $chunk = null; } else if ($type === 'all' || $type === 'all-rtl') { // We're fine. } else { css_send_css_not_found(); } if (file_exists("$CFG->dirroot/theme/$themename/config.php")) { // The theme exists in standard location - ok. } else if (!empty($CFG->themedir) and file_exists("$CFG->themedir/$themename/config.php")) { // Alternative theme location contains this theme - ok. } else { header('HTTP/1.0 404 not found'); die('Theme was not found, sorry.'); } $candidatedir = "$CFG->localcachedir/theme/$rev/$themename/css"; $etag = "$rev/$themename/$type"; $candidatename = $type; if (!$usesvg) { // Add to the sheet name, one day we'll be able to just drop this. $candidatedir .= '/nosvg'; $etag .= '/nosvg'; } if ($chunk !== null) { $etag .= '/chunk'.$chunk; $candidatename .= '.'.$chunk; } $candidatesheet = "$candidatedir/$candidatename.css"; $etag = sha1($etag); if (file_exists($candidatesheet)) { if (!empty($_SERVER['HTTP_IF_NONE_MATCH']) || !empty($_SERVER['HTTP_IF_MODIFIED_SINCE'])) { // We do not actually need to verify the etag value because our files // never change in cache because we increment the rev counter. css_send_unmodified(filemtime($candidatesheet), $etag); } css_send_cached_css($candidatesheet, $etag); } // Ok, now we need to start normal moodle script, we need to load all libs and $DB. define('ABORT_AFTER_CONFIG_CANCEL', true); define('NO_MOODLE_COOKIES', true); // Session not used here. define('NO_UPGRADE_CHECK', true); // Ignore upgrade check. require("$CFG->dirroot/lib/setup.php"); $theme = theme_config::load($themename); $theme->force_svg_use($usesvg); $theme->set_rtl_mode($type === 'all-rtl' ? true : false); $themerev = theme_get_revision(); $cache = true; if ($themerev <= 0 or $themerev != $rev) { $rev = $themerev; $cache = false; $candidatedir = "$CFG->localcachedir/theme/$rev/$themename/css"; $etag = "$rev/$themename/$type"; $candidatename = $type; if (!$usesvg) { // Add to the sheet name, one day we'll be able to just drop this. $candidatedir .= '/nosvg'; $etag .= '/nosvg'; } if ($chunk !== null) { $etag .= '/chunk'.$chunk; $candidatename .= '.'.$chunk; } $candidatesheet = "$candidatedir/$candidatename.css"; $etag = sha1($etag); } make_localcache_directory('theme', false); if ($type === 'editor') { $csscontent = $theme->get_css_content_editor(); css_store_css($theme, "$candidatedir/editor.css", $csscontent, false); } else { $lock = null; // Lock system to prevent concurrent requests to compile LESS/SCSS, which is really slow and CPU intensive. // Each client should wait for one to finish the compilation before starting a new compiling process. // We only do this when the file will be cached... if (in_array($type, ['less', 'scss']) && $cache) { $lockfactory = \core\lock\lock_config::get_lock_factory('core_theme_get_css_content'); // We wait for the lock to be acquired, the timeout does not need to be strict here. $lock = $lockfactory->get_lock($themename, rand(15, 30)); if (file_exists($candidatesheet)) { // The file was built while we waited for the lock, we release the lock and serve the file. if ($lock) { $lock->release(); } css_send_cached_css($candidatesheet, $etag); } } // Older IEs require smaller chunks. $csscontent = $theme->get_css_content(); $relroot = preg_replace('|^http.?://[^/]+|', '', $CFG->wwwroot); if (!empty($slashargument)) { if ($usesvg) { $chunkurl = "{$relroot}/theme/styles.php/{$themename}/{$rev}/$type"; } else { $chunkurl = "{$relroot}/theme/styles.php/_s/{$themename}/{$rev}/$type"; } } else { if ($usesvg) { $chunkurl = "{$relroot}/theme/styles.php?theme={$themename}&rev={$rev}&type=$type"; } else { $chunkurl = "{$relroot}/theme/styles.php?theme={$themename}&rev={$rev}&type=$type&svg=0"; } } css_store_css($theme, "$candidatedir/$type.css", $csscontent, true, $chunkurl); // Release the lock. if ($lock) { $lock->release(); } } if (!$cache) { // Do not pollute browser caches if invalid revision requested, // let's ignore legacy IE breakage here too. css_send_uncached_css($csscontent); } else if ($chunk !== null and file_exists($candidatesheet)) { // Greetings stupid legacy IEs! css_send_cached_css($candidatesheet, $etag); } else { // Real browsers - this is the expected result! css_send_cached_css_content($csscontent, $etag); }