diff --git a/core/includes/theme.inc b/core/includes/theme.inc index b71e437dcbab51af7031f1de04a5793a3877f225..04b41ce4bf4d5311c671f61022c768ee22a8b2f3 100644 --- a/core/includes/theme.inc +++ b/core/includes/theme.inc @@ -343,10 +343,10 @@ function theme_get_setting($setting_name, $theme = NULL) { // Generate the path to the logo image. if ($cache[$theme]->get('logo.use_default')) { - $cache[$theme]->set('logo.url', file_create_url($theme_object->getPath() . '/logo.svg')); + $cache[$theme]->set('logo.url', file_url_transform_relative(file_create_url($theme_object->getPath() . '/logo.svg'))); } elseif ($logo_path = $cache[$theme]->get('logo.path')) { - $cache[$theme]->set('logo.url', file_create_url($logo_path)); + $cache[$theme]->set('logo.url', file_url_transform_relative(file_create_url($logo_path))); } // Generate the path to the favicon. @@ -354,14 +354,14 @@ function theme_get_setting($setting_name, $theme = NULL) { $favicon_path = $cache[$theme]->get('favicon.path'); if ($cache[$theme]->get('favicon.use_default')) { if (file_exists($favicon = $theme_object->getPath() . '/favicon.ico')) { - $cache[$theme]->set('favicon.url', file_create_url($favicon)); + $cache[$theme]->set('favicon.url', file_url_transform_relative(file_create_url($favicon))); } else { - $cache[$theme]->set('favicon.url', file_create_url('core/misc/favicon.ico')); + $cache[$theme]->set('favicon.url', file_url_transform_relative(file_create_url('core/misc/favicon.ico'))); } } elseif ($favicon_path) { - $cache[$theme]->set('favicon.url', file_create_url($favicon_path)); + $cache[$theme]->set('favicon.url', file_url_transform_relative(file_create_url($favicon_path))); } else { $cache[$theme]->set('features.favicon', FALSE); @@ -756,7 +756,7 @@ function template_preprocess_links(&$variables) { */ function template_preprocess_image(&$variables) { if (!empty($variables['uri'])) { - $variables['attributes']['src'] = file_create_url($variables['uri']); + $variables['attributes']['src'] = file_url_transform_relative(file_create_url($variables['uri'])); } // Generate a srcset attribute conforming to the spec at // http://www.w3.org/html/wg/drafts/html/master/embedded-content.html#attr-img-srcset @@ -764,7 +764,7 @@ function template_preprocess_image(&$variables) { $srcset = array(); foreach ($variables['srcset'] as $src) { // URI is mandatory. - $source = file_create_url($src['uri']); + $source = file_url_transform_relative(file_create_url($src['uri'])); if (isset($src['width']) && !empty($src['width'])) { $source .= ' ' . $src['width']; } diff --git a/core/lib/Drupal/Core/Asset/CssCollectionRenderer.php b/core/lib/Drupal/Core/Asset/CssCollectionRenderer.php index 49e4675e1b7d83e55a3111d168ac6056990d76b5..0f70dbf593eaba961c36d2d3ceb4000efb6786ae 100644 --- a/core/lib/Drupal/Core/Asset/CssCollectionRenderer.php +++ b/core/lib/Drupal/Core/Asset/CssCollectionRenderer.php @@ -135,7 +135,7 @@ public function render(array $css_assets) { // assets: output a LINK tag for a file CSS asset. if (count($css_assets) <= 31) { $element = $link_element_defaults; - $element['#attributes']['href'] = file_create_url($css_asset['data']) . $query_string_separator . $query_string; + $element['#attributes']['href'] = file_url_transform_relative(file_create_url($css_asset['data'])) . $query_string_separator . $query_string; $element['#attributes']['media'] = $css_asset['media']; $element['#browsers'] = $css_asset['browsers']; $elements[] = $element; @@ -148,7 +148,7 @@ public function render(array $css_assets) { // LINK tag. if (!$css_asset['preprocess']) { $element = $link_element_defaults; - $element['#attributes']['href'] = file_create_url($css_asset['data']) . $query_string_separator . $query_string; + $element['#attributes']['href'] = file_url_transform_relative(file_create_url($css_asset['data'])) . $query_string_separator . $query_string; $element['#attributes']['media'] = $css_asset['media']; $element['#browsers'] = $css_asset['browsers']; $elements[] = $element; @@ -168,7 +168,7 @@ public function render(array $css_assets) { // control browser-caching. IE7 does not support a media type on // the @import statement, so we instead specify the media for // the group on the STYLE tag. - $import[] = '@import url("' . Html::escape(file_create_url($next_css_asset['data']) . '?' . $query_string) . '");'; + $import[] = '@import url("' . Html::escape(file_url_transform_relative(file_create_url($next_css_asset['data'])) . '?' . $query_string) . '");'; // Move the outer for loop skip the next item, since we // processed it here. $i = $j; diff --git a/core/lib/Drupal/Core/Asset/CssOptimizer.php b/core/lib/Drupal/Core/Asset/CssOptimizer.php index dc34a237abdf008b588d1ca3298bc7eec539b1f2..704ee571e4e87fe5c0961d912dc9125c1fc8ab42 100644 --- a/core/lib/Drupal/Core/Asset/CssOptimizer.php +++ b/core/lib/Drupal/Core/Asset/CssOptimizer.php @@ -265,7 +265,7 @@ public function rewriteFileURI($matches) { $last = $path; $path = preg_replace('`(^|/)(?!\.\./)([^/]+)/\.\./`', '$1', $path); } - return 'url(' . file_create_url($path) . ')'; + return 'url(' . file_url_transform_relative(file_create_url($path)) . ')'; } } diff --git a/core/lib/Drupal/Core/Asset/JsCollectionRenderer.php b/core/lib/Drupal/Core/Asset/JsCollectionRenderer.php index 70832e9d9d1027b5240043666fed1bde39e58ad0..b0d22f15cf94ca4f95937934da7bd289b35b084e 100644 --- a/core/lib/Drupal/Core/Asset/JsCollectionRenderer.php +++ b/core/lib/Drupal/Core/Asset/JsCollectionRenderer.php @@ -79,7 +79,7 @@ public function render(array $js_assets) { case 'file': $query_string = $js_asset['version'] == -1 ? $default_query_string : 'v=' . $js_asset['version']; $query_string_separator = (strpos($js_asset['data'], '?') !== FALSE) ? '&' : '?'; - $element['#attributes']['src'] = file_create_url($js_asset['data']); + $element['#attributes']['src'] = file_url_transform_relative(file_create_url($js_asset['data'])); // Only add the cache-busting query string if this isn't an aggregate // file. if (!isset($js_asset['preprocessed'])) { diff --git a/core/lib/Drupal/Core/Render/Element/ImageButton.php b/core/lib/Drupal/Core/Render/Element/ImageButton.php index 4ee39a6340f32bb96a737238b7e5c21e6aa79119..f22ae1decf8cd221dfaeb82aa108e64c8f71096a 100644 --- a/core/lib/Drupal/Core/Render/Element/ImageButton.php +++ b/core/lib/Drupal/Core/Render/Element/ImageButton.php @@ -75,7 +75,7 @@ public static function preRenderButton($element) { $element['#attributes']['type'] = 'image'; Element::setAttributes($element, array('id', 'name', 'value')); - $element['#attributes']['src'] = file_create_url($element['#src']); + $element['#attributes']['src'] = file_url_transform_relative(file_create_url($element['#src'])); if (!empty($element['#title'])) { $element['#attributes']['alt'] = $element['#title']; $element['#attributes']['title'] = $element['#title']; diff --git a/core/lib/Drupal/Core/Template/TwigExtension.php b/core/lib/Drupal/Core/Template/TwigExtension.php index 5b71c5290bdc10e1c267ddfe1ce7547a13b2b53c..73b2ca9c6554167d4bc8ba225e78b9b2208ccc29 100644 --- a/core/lib/Drupal/Core/Template/TwigExtension.php +++ b/core/lib/Drupal/Core/Template/TwigExtension.php @@ -135,7 +135,9 @@ public function getFunctions() { new \Twig_SimpleFunction('url', array($this, 'getUrl'), array('is_safe_callback' => array($this, 'isUrlGenerationSafe'))), new \Twig_SimpleFunction('path', array($this, 'getPath'), array('is_safe_callback' => array($this, 'isUrlGenerationSafe'))), new \Twig_SimpleFunction('link', array($this, 'getLink')), - new \Twig_SimpleFunction('file_url', 'file_create_url'), + new \Twig_SimpleFunction('file_url', function ($uri) { + return file_url_transform_relative(file_create_url($uri)); + }), new \Twig_SimpleFunction('attach_library', [$this, 'attachLibrary']), new \Twig_SimpleFunction('active_theme_path', [$this, 'getActiveThemePath']), new \Twig_SimpleFunction('active_theme', [$this, 'getActiveTheme']), diff --git a/core/modules/ckeditor/src/Plugin/Editor/CKEditor.php b/core/modules/ckeditor/src/Plugin/Editor/CKEditor.php index 5ae5e85b21059bb4200766e41c7a4a535c86da5f..8072060697003963ee23c6ec8f6c3117108ac7f7 100644 --- a/core/modules/ckeditor/src/Plugin/Editor/CKEditor.php +++ b/core/modules/ckeditor/src/Plugin/Editor/CKEditor.php @@ -298,8 +298,11 @@ public function getJSSettings(EditorEntity $editor) { ); // Finally, set Drupal-specific CKEditor settings. + $root_relative_file_url = function ($uri) { + return file_url_transform_relative(file_create_url($uri)); + }; $settings += array( - 'drupalExternalPlugins' => array_map('file_create_url', $external_plugin_files), + 'drupalExternalPlugins' => array_map($root_relative_file_url, $external_plugin_files), ); // Parse all CKEditor plugin JavaScript files for translations. @@ -421,6 +424,7 @@ public function buildContentsCssJSSetting(EditorEntity $editor) { $this->moduleHandler->alter('ckeditor_css', $css, $editor); $css = array_merge($css, _ckeditor_theme_css()); $css = array_map('file_create_url', $css); + $css = array_map('file_url_transform_relative', $css); return array_values($css); } diff --git a/core/modules/ckeditor/src/Tests/CKEditorTest.php b/core/modules/ckeditor/src/Tests/CKEditorTest.php index 4df0113333105b45792ab1d3748519aa96848676..6a2fd426495ddff320ec23a08e44ec429d1ff0cf 100644 --- a/core/modules/ckeditor/src/Tests/CKEditorTest.php +++ b/core/modules/ckeditor/src/Tests/CKEditorTest.php @@ -90,8 +90,8 @@ function testGetJSSettings() { 'language' => 'en', 'stylesSet' => FALSE, 'drupalExternalPlugins' => array( - 'drupalimage' => file_create_url('core/modules/ckeditor/js/plugins/drupalimage/plugin.js'), - 'drupallink' => file_create_url('core/modules/ckeditor/js/plugins/drupallink/plugin.js'), + 'drupalimage' => file_url_transform_relative(file_create_url('core/modules/ckeditor/js/plugins/drupalimage/plugin.js')), + 'drupallink' => file_url_transform_relative(file_create_url('core/modules/ckeditor/js/plugins/drupallink/plugin.js')), ), ); $expected_config = $this->castSafeStrings($expected_config); @@ -114,9 +114,9 @@ function testGetJSSettings() { $expected_config['toolbar'][0]['items'][] = 'Format'; $expected_config['format_tags'] = 'p;h2;h3;h4;h5;h6'; $expected_config['extraPlugins'] .= ',llama_contextual,llama_contextual_and_button'; - $expected_config['drupalExternalPlugins']['llama_contextual'] = file_create_url('core/modules/ckeditor/tests/modules/js/llama_contextual.js'); - $expected_config['drupalExternalPlugins']['llama_contextual_and_button'] = file_create_url('core/modules/ckeditor/tests/modules/js/llama_contextual_and_button.js'); - $expected_config['contentsCss'][] = file_create_url('core/modules/ckeditor/tests/modules/ckeditor_test.css'); + $expected_config['drupalExternalPlugins']['llama_contextual'] = file_url_transform_relative(file_create_url('core/modules/ckeditor/tests/modules/js/llama_contextual.js')); + $expected_config['drupalExternalPlugins']['llama_contextual_and_button'] = file_url_transform_relative(file_create_url('core/modules/ckeditor/tests/modules/js/llama_contextual_and_button.js')); + $expected_config['contentsCss'][] = file_url_transform_relative(file_create_url('core/modules/ckeditor/tests/modules/ckeditor_test.css')); ksort($expected_config); $this->assertIdentical($expected_config, $this->castSafeStrings($this->ckeditor->getJSSettings($editor)), 'Generated JS settings are correct for customized configuration.'); @@ -261,16 +261,16 @@ function testBuildContentsCssJSSetting() { // Enable the editor_test module, which implements hook_ckeditor_css_alter(). $this->enableModules(array('ckeditor_test')); - $expected[] = file_create_url('core/modules/ckeditor/tests/modules/ckeditor_test.css'); + $expected[] = file_url_transform_relative(file_create_url('core/modules/ckeditor/tests/modules/ckeditor_test.css')); $this->assertIdentical($expected, $this->ckeditor->buildContentsCssJSSetting($editor), '"contentsCss" configuration part of JS settings built correctly while a hook_ckeditor_css_alter() implementation exists.'); // Enable the Bartik theme, which specifies a CKEditor stylesheet. \Drupal::service('theme_handler')->install(['bartik']); \Drupal::service('theme_handler')->setDefault('bartik'); - $expected[] = file_create_url('core/themes/bartik/css/base/elements.css'); - $expected[] = file_create_url('core/themes/bartik/css/components/captions.css'); - $expected[] = file_create_url('core/themes/bartik/css/components/table.css'); - $expected[] = file_create_url('core/themes/bartik/css/components/text-formatted.css'); + $expected[] = file_url_transform_relative(file_create_url('core/themes/bartik/css/base/elements.css')); + $expected[] = file_url_transform_relative(file_create_url('core/themes/bartik/css/components/captions.css')); + $expected[] = file_url_transform_relative(file_create_url('core/themes/bartik/css/components/table.css')); + $expected[] = file_url_transform_relative(file_create_url('core/themes/bartik/css/components/text-formatted.css')); $this->assertIdentical($expected, $this->ckeditor->buildContentsCssJSSetting($editor), '"contentsCss" configuration part of JS settings built correctly while a theme providing a CKEditor stylesheet exists.'); } @@ -479,8 +479,8 @@ protected function getDefaultToolbarConfig() { protected function getDefaultContentsCssConfig() { return array( - file_create_url('core/modules/ckeditor/css/ckeditor-iframe.css'), - file_create_url('core/modules/system/css/components/align.module.css'), + file_url_transform_relative(file_create_url('core/modules/ckeditor/css/ckeditor-iframe.css')), + file_url_transform_relative(file_create_url('core/modules/system/css/components/align.module.css')), ); } diff --git a/core/modules/ckeditor/src/Tests/CKEditorToolbarButtonTest.php b/core/modules/ckeditor/src/Tests/CKEditorToolbarButtonTest.php index 5751f8629ad70792351b2ee569b35188b5d898fd..4b241d7997b1f856cba5b61c8ba06c96213029e7 100644 --- a/core/modules/ckeditor/src/Tests/CKEditorToolbarButtonTest.php +++ b/core/modules/ckeditor/src/Tests/CKEditorToolbarButtonTest.php @@ -57,7 +57,6 @@ protected function setUp() { * Method tests CKEditor image buttons. */ public function testImageButtonDisplay() { - global $base_url; $this->drupalLogin($this->admin_user); // Install the Arabic language (which is RTL) and configure as the default. @@ -75,7 +74,7 @@ public function testImageButtonDisplay() { $json_encode = function($html) { return trim(Json::encode($html), '"'); }; - $markup = $json_encode($base_url . '/core/modules/ckeditor/js/plugins/drupalimage/image.png'); + $markup = $json_encode(file_url_transform_relative(file_create_url('core/modules/ckeditor/js/plugins/drupalimage/image.png'))); $this->assertRaw($markup); } diff --git a/core/modules/color/color.module b/core/modules/color/color.module index 2fbef3034083babad17687e56bd464f39262a00a..055d750f7ddf4dced928b558ed8493ca8b4fa1ca 100644 --- a/core/modules/color/color.module +++ b/core/modules/color/color.module @@ -126,7 +126,7 @@ function color_block_view_pre_render(array $build) { // Override logo. $logo = $config->get('logo'); if ($logo && $build['content']['site_logo'] && preg_match('!' . $theme_key . '/logo.svg$!', $build['content']['site_logo']['#uri'])) { - $build['content']['site_logo']['#uri'] = file_create_url($logo); + $build['content']['site_logo']['#uri'] = file_url_transform_relative(file_create_url($logo)); } return $build; diff --git a/core/modules/color/src/Tests/ColorTest.php b/core/modules/color/src/Tests/ColorTest.php index 4a410644477009ea457a7a45825ba8b3a2af40d6..8e3d7714abe8f894ea3664f59bb998098f10ab26 100644 --- a/core/modules/color/src/Tests/ColorTest.php +++ b/core/modules/color/src/Tests/ColorTest.php @@ -121,7 +121,7 @@ function _testColor($theme, $test_values) { $this->drupalGet('<front>'); $stylesheets = $this->config('color.theme.' . $theme)->get('stylesheets'); foreach ($stylesheets as $stylesheet) { - $this->assertPattern('|' . file_create_url($stylesheet) . '|', 'Make sure the color stylesheet is included in the content. (' . $theme . ')'); + $this->assertPattern('|' . file_url_transform_relative(file_create_url($stylesheet)) . '|', 'Make sure the color stylesheet is included in the content. (' . $theme . ')'); $stylesheet_content = join("\n", file($stylesheet)); $this->assertTrue(strpos($stylesheet_content, 'color: #123456') !== FALSE, 'Make sure the color we changed is in the color stylesheet. (' . $theme . ')'); } @@ -191,7 +191,7 @@ function testLogoSettingOverride() { // Ensure that the overridden logo is present in Bartik, which is colorable. $this->drupalGet('admin/appearance/settings/bartik'); - $this->assertIdentical($GLOBALS['base_url'] . '/' . 'core/misc/druplicon.png', $this->getDrupalSettings()['color']['logo']); + $this->assertIdentical($GLOBALS['base_path'] . 'core/misc/druplicon.png', $this->getDrupalSettings()['color']['logo']); } /** diff --git a/core/modules/file/file.module b/core/modules/file/file.module index 08e94384bc70d12e03f2ea0e571d4e4b4cafb3ba..984773ace17c91673d5bd84c0c145f42efa0789e 100644 --- a/core/modules/file/file.module +++ b/core/modules/file/file.module @@ -977,7 +977,12 @@ function file_tokens($type, $tokens, array $data, array $options, BubbleableMeta break; case 'url': + // Ideally, this would use file_url_transform_relative(), but because + // tokens are also often used in e-mails, it's better to keep absolute + // file URLs. The 'url.site' cache context is associated to ensure the + // correct absolute URL is used in case of a multisite setup. $replacements[$original] = file_create_url($file->getFileUri()); + $bubbleable_metadata->addCacheContexts(['url.site']); break; // These tokens are default variations on the chained tokens handled below. @@ -1228,7 +1233,13 @@ function template_preprocess_file_link(&$variables) { $options = array(); $file_entity = ($file instanceof File) ? $file : File::load($file->fid); + // @todo Wrap in file_url_transform_relative(). This is currently + // impossible. As a work-around, we currently add the 'url.site' cache context + // to ensure different file URLs are generated for different sites in a + // multisite setup, including HTTP and HTTPS versions of the same site. + // Fix in https://www.drupal.org/node/2646744. $url = file_create_url($file_entity->getFileUri()); + $variables['#cache']['contexts'][] = 'url.site'; $mime_type = $file->getMimeType(); // Set options as per anchor format described at diff --git a/core/modules/file/src/Entity/File.php b/core/modules/file/src/Entity/File.php index ae5ca764b2183cc8cd891df3f753bf13c4a87a38..6417488e52b24a70b21737e24d583faeeb23cc9e 100644 --- a/core/modules/file/src/Entity/File.php +++ b/core/modules/file/src/Entity/File.php @@ -70,6 +70,8 @@ public function setFileUri($uri) { /** * {@inheritdoc} + * + * @see file_url_transform_relative() */ public function url($rel = 'canonical', $options = array()) { return file_create_url($this->getFileUri()); diff --git a/core/modules/file/src/Plugin/Field/FieldFormatter/BaseFieldFileFormatterBase.php b/core/modules/file/src/Plugin/Field/FieldFormatter/BaseFieldFileFormatterBase.php index fbe14f09ba95c50af42e6891958856de75e55aed..798babd73c13fcfc5aeb0e19731b8bbea44d1260 100644 --- a/core/modules/file/src/Plugin/Field/FieldFormatter/BaseFieldFileFormatterBase.php +++ b/core/modules/file/src/Plugin/Field/FieldFormatter/BaseFieldFileFormatterBase.php @@ -52,6 +52,8 @@ public function viewElements(FieldItemListInterface $items, $langcode) { $url = NULL; // Add support to link to the entity itself. if ($this->getSetting('link_to_file')) { + // @todo Wrap in file_url_transform_relative(). This is currently + // impossible. See below. $url = file_create_url($items->getEntity()->uri->value); } @@ -63,6 +65,16 @@ public function viewElements(FieldItemListInterface $items, $langcode) { '#type' => 'link', '#title' => $view_value, '#url' => Url::fromUri($url), + // @todo Remove the 'url.site' cache context by using a relative file + // URL (file_url_transform_relative()). This is currently impossible + // because #type => link requires a Url object, and Url objects do not + // support relative URLs: they require fully qualified URLs. Fix in + // https://www.drupal.org/node/2646744. + '#cache' => [ + 'contexts' => [ + 'url.site', + ], + ], ]; } else { diff --git a/core/modules/file/src/Plugin/Field/FieldFormatter/FileUriFormatter.php b/core/modules/file/src/Plugin/Field/FieldFormatter/FileUriFormatter.php index 0a2a855bcdfab5f4343080111cd19d408466d22c..c86cc8e0d99061f6e0c26047df90c4092ab49348 100644 --- a/core/modules/file/src/Plugin/Field/FieldFormatter/FileUriFormatter.php +++ b/core/modules/file/src/Plugin/Field/FieldFormatter/FileUriFormatter.php @@ -55,6 +55,9 @@ public function settingsForm(array $form, FormStateInterface $form_state) { protected function viewValue(FieldItemInterface $item) { $value = $item->value; if ($this->getSetting('file_download_path')) { + // @todo Wrap in file_url_transform_relative(). This is currently + // impossible. See BaseFieldFileFormatterBase::viewElements(). Fix in + // https://www.drupal.org/node/2646744. $value = file_create_url($value); } return $value; diff --git a/core/modules/file/src/Plugin/Field/FieldFormatter/RSSEnclosureFormatter.php b/core/modules/file/src/Plugin/Field/FieldFormatter/RSSEnclosureFormatter.php index ab479f32f3c61ad07e85d09f359479f40959af44..2f0a65f67be2506f7827a2422750ba56ae358921 100644 --- a/core/modules/file/src/Plugin/Field/FieldFormatter/RSSEnclosureFormatter.php +++ b/core/modules/file/src/Plugin/Field/FieldFormatter/RSSEnclosureFormatter.php @@ -33,6 +33,9 @@ public function viewElements(FieldItemListInterface $items, $langcode) { $entity->rss_elements[] = array( 'key' => 'enclosure', 'attributes' => array( + // In RSS feeds, it is necessary to use absolute URLs. The 'url.site' + // cache context is already associated with RSS feed responses, so it + // does not need to be specified here. 'url' => file_create_url($file->getFileUri()), 'length' => $file->getSize(), 'type' => $file->getMimeType(), diff --git a/core/modules/file/src/Plugin/Field/FieldFormatter/UrlPlainFormatter.php b/core/modules/file/src/Plugin/Field/FieldFormatter/UrlPlainFormatter.php index 6f2264e91d792d53fdbf3ac14bf179953df642b8..0795be2002ce0e4e66ef754e71d794b68123f25f 100644 --- a/core/modules/file/src/Plugin/Field/FieldFormatter/UrlPlainFormatter.php +++ b/core/modules/file/src/Plugin/Field/FieldFormatter/UrlPlainFormatter.php @@ -30,7 +30,7 @@ public function viewElements(FieldItemListInterface $items, $langcode) { foreach ($this->getEntitiesToView($items, $langcode) as $delta => $file) { $elements[$delta] = array( - '#markup' => file_create_url($file->getFileUri()), + '#markup' => file_url_transform_relative(file_create_url($file->getFileUri())), '#cache' => array( 'tags' => $file->getCacheTags(), ), diff --git a/core/modules/file/src/Plugin/views/field/File.php b/core/modules/file/src/Plugin/views/field/File.php index 615aa9f8e98c2cac0cd59553573ac5833c9ca8bb..0b9c6965ddc1e3da9bbceec71bf49e58cc516144 100644 --- a/core/modules/file/src/Plugin/views/field/File.php +++ b/core/modules/file/src/Plugin/views/field/File.php @@ -69,6 +69,12 @@ public function buildOptionsForm(&$form, FormStateInterface $form_state) { protected function renderLink($data, ResultRow $values) { if (!empty($this->options['link_to_file']) && $data !== NULL && $data !== '') { $this->options['alter']['make_link'] = TRUE; + // @todo Wrap in file_url_transform_relative(). This is currently + // impossible. As a work-around, we could add the 'url.site' cache context + // to ensure different file URLs are generated for different sites in a + // multisite setup, including HTTP and HTTPS versions of the same site. + // But unfortunately it's impossible to bubble a cache context here. + // Fix in https://www.drupal.org/node/2646744. $this->options['alter']['path'] = file_create_url($this->getValue($values, 'uri')); } diff --git a/core/modules/file/src/Tests/FileTokenReplaceTest.php b/core/modules/file/src/Tests/FileTokenReplaceTest.php index 93a6c67a87dc8a4abdc432721bb160e442ca7012..bfc85de3d4919d4c3141cc41f09dfe55b3de5620 100644 --- a/core/modules/file/src/Tests/FileTokenReplaceTest.php +++ b/core/modules/file/src/Tests/FileTokenReplaceTest.php @@ -66,7 +66,8 @@ function testFileTokenReplacement() { $metadata_tests['[file:path]'] = $base_bubbleable_metadata; $metadata_tests['[file:mime]'] = $base_bubbleable_metadata; $metadata_tests['[file:size]'] = $base_bubbleable_metadata; - $metadata_tests['[file:url]'] = $base_bubbleable_metadata; + $bubbleable_metadata = clone $base_bubbleable_metadata; + $metadata_tests['[file:url]'] = $bubbleable_metadata->addCacheContexts(['url.site']); $bubbleable_metadata = clone $base_bubbleable_metadata; $metadata_tests['[file:created]'] = $bubbleable_metadata->addCacheTags(['rendered']); $metadata_tests['[file:created:short]'] = $bubbleable_metadata; diff --git a/core/modules/image/image.admin.inc b/core/modules/image/image.admin.inc index b3d4a6e7df2b7b0a34c2d4b6dd2539906a49ffa7..777e43855a4124e29ed5ca0bf866c14e61470618 100644 --- a/core/modules/image/image.admin.inc +++ b/core/modules/image/image.admin.inc @@ -34,7 +34,7 @@ function template_preprocess_image_style_preview(&$variables) { $original_path = \Drupal::config('image.settings')->get('preview_image'); $original_image = $image_factory->get($original_path); $variables['original'] = array( - 'url' => file_create_url($original_path), + 'url' => file_url_transform_relative(file_create_url($original_path)), 'width' => $original_image->getWidth(), 'height' => $original_image->getHeight(), ); @@ -55,7 +55,7 @@ function template_preprocess_image_style_preview(&$variables) { } $preview_image = $image_factory->get($preview_file); $variables['derivative'] = array( - 'url' => file_create_url($preview_file), + 'url' => file_url_transform_relative(file_create_url($preview_file)), 'width' => $preview_image->getWidth(), 'height' => $preview_image->getHeight(), ); diff --git a/core/modules/image/src/ImageStyleInterface.php b/core/modules/image/src/ImageStyleInterface.php index 95072915148cdb3ba4d3aaeeb20188f366630a0a..e1cad03bbba6f683300934d0405b4a1608cd1794 100644 --- a/core/modules/image/src/ImageStyleInterface.php +++ b/core/modules/image/src/ImageStyleInterface.php @@ -70,6 +70,7 @@ public function buildUri($uri); * in an <img> tag. Requesting the URL will cause the image to be created. * * @see \Drupal\image\Controller\ImageStyleDownloadController::deliver() + * @see file_url_transform_relative() */ public function buildUrl($path, $clean_urls = NULL); diff --git a/core/modules/image/src/Plugin/Field/FieldFormatter/ImageFormatter.php b/core/modules/image/src/Plugin/Field/FieldFormatter/ImageFormatter.php index 57204152732ca43db034286af8b836fa8849b9b5..55b613a8ff50101fde1d6643fec42daece2f7a63 100644 --- a/core/modules/image/src/Plugin/Field/FieldFormatter/ImageFormatter.php +++ b/core/modules/image/src/Plugin/Field/FieldFormatter/ImageFormatter.php @@ -199,9 +199,16 @@ public function viewElements(FieldItemListInterface $items, $langcode) { } foreach ($files as $delta => $file) { + $cache_contexts = array(); if (isset($link_file)) { $image_uri = $file->getFileUri(); + // @todo Wrap in file_url_transform_relative(). This is currently + // impossible. As a work-around, we currently add the 'url.site' cache + // context to ensure different file URLs are generated for different + // sites in a multisite setup, including HTTP and HTTPS versions of the + // same site. Fix in https://www.drupal.org/node/2646744. $url = Url::fromUri(file_create_url($image_uri)); + $cache_contexts[] = 'url.site'; } $cache_tags = Cache::mergeTags($cache_tags, $file->getCacheTags()); @@ -219,6 +226,7 @@ public function viewElements(FieldItemListInterface $items, $langcode) { '#url' => $url, '#cache' => array( 'tags' => $cache_tags, + 'contexts' => $cache_contexts, ), ); } diff --git a/core/modules/image/src/Tests/ImageAdminStylesTest.php b/core/modules/image/src/Tests/ImageAdminStylesTest.php index 08f0adf9abf744270c45a9c474206ebcf1cb59b5..fb59a4809a0755a8103e5508124598af01becbef 100644 --- a/core/modules/image/src/Tests/ImageAdminStylesTest.php +++ b/core/modules/image/src/Tests/ImageAdminStylesTest.php @@ -307,7 +307,7 @@ function testStyleReplacement() { // Test that image is displayed using newly created style. $this->drupalGet('node/' . $nid); - $this->assertRaw($style->buildUrl($original_uri), format_string('Image displayed using style @style.', array('@style' => $style_name))); + $this->assertRaw(file_url_transform_relative($style->buildUrl($original_uri)), format_string('Image displayed using style @style.', array('@style' => $style_name))); // Rename the style and make sure the image field is updated. $new_style_name = strtolower($this->randomMachineName(10)); @@ -322,7 +322,7 @@ function testStyleReplacement() { // Reload the image style using the new name. $style = ImageStyle::load($new_style_name); - $this->assertRaw($style->buildUrl($original_uri), 'Image displayed using style replacement style.'); + $this->assertRaw(file_url_transform_relative($style->buildUrl($original_uri)), 'Image displayed using style replacement style.'); // Delete the style and choose a replacement style. $edit = array( @@ -334,7 +334,7 @@ function testStyleReplacement() { $replacement_style = ImageStyle::load('thumbnail'); $this->drupalGet('node/' . $nid); - $this->assertRaw($replacement_style->buildUrl($original_uri), 'Image displayed using style replacement style.'); + $this->assertRaw(file_url_transform_relative($replacement_style->buildUrl($original_uri)), 'Image displayed using style replacement style.'); } /** @@ -441,7 +441,7 @@ function testConfigImport() { // Test that image is displayed using newly created style. $this->drupalGet('node/' . $nid); - $this->assertRaw($style->buildUrl($original_uri), format_string('Image displayed using style @style.', array('@style' => $style_name))); + $this->assertRaw(file_url_transform_relative($style->buildUrl($original_uri)), format_string('Image displayed using style @style.', array('@style' => $style_name))); // Copy config to sync, and delete the image style. $sync = $this->container->get('config.storage.sync'); diff --git a/core/modules/image/src/Tests/ImageDimensionsTest.php b/core/modules/image/src/Tests/ImageDimensionsTest.php index 683756a089b3fb9def323430377a26d3c8997224..f232d71197e48201fc701d2ffbf43cc271373507 100644 --- a/core/modules/image/src/Tests/ImageDimensionsTest.php +++ b/core/modules/image/src/Tests/ImageDimensionsTest.php @@ -41,7 +41,7 @@ function testImageDimensions() { $style = ImageStyle::create(array('name' => 'test', 'label' => 'Test')); $style->save(); $generated_uri = 'public://styles/test/public/'. \Drupal::service('file_system')->basename($original_uri); - $url = $style->buildUrl($original_uri); + $url = file_url_transform_relative($style->buildUrl($original_uri)); $variables = array( '#theme' => 'image_style', @@ -70,7 +70,7 @@ function testImageDimensions() { $style->save(); $this->assertEqual($this->getImageTag($variables), '<img src="' . $url . '" width="120" height="60" alt="" class="image-style-test" />'); $this->assertFalse(file_exists($generated_uri), 'Generated file does not exist.'); - $this->drupalGet($url); + $this->drupalGet($this->getAbsoluteUrl($url)); $this->assertResponse(200, 'Image was generated at the URL.'); $this->assertTrue(file_exists($generated_uri), 'Generated file does exist after we accessed it.'); $image_file = $image_factory->get($generated_uri); @@ -91,7 +91,7 @@ function testImageDimensions() { $style->save(); $this->assertEqual($this->getImageTag($variables), '<img src="' . $url . '" width="60" height="120" alt="" class="image-style-test" />'); $this->assertFalse(file_exists($generated_uri), 'Generated file does not exist.'); - $this->drupalGet($url); + $this->drupalGet($this->getAbsoluteUrl($url)); $this->assertResponse(200, 'Image was generated at the URL.'); $this->assertTrue(file_exists($generated_uri), 'Generated file does exist after we accessed it.'); $image_file = $image_factory->get($generated_uri); @@ -113,7 +113,7 @@ function testImageDimensions() { $style->save(); $this->assertEqual($this->getImageTag($variables), '<img src="' . $url . '" width="45" height="90" alt="" class="image-style-test" />'); $this->assertFalse(file_exists($generated_uri), 'Generated file does not exist.'); - $this->drupalGet($url); + $this->drupalGet($this->getAbsoluteUrl($url)); $this->assertResponse(200, 'Image was generated at the URL.'); $this->assertTrue(file_exists($generated_uri), 'Generated file does exist after we accessed it.'); $image_file = $image_factory->get($generated_uri); @@ -135,7 +135,7 @@ function testImageDimensions() { $style->save(); $this->assertEqual($this->getImageTag($variables), '<img src="' . $url . '" width="45" height="90" alt="" class="image-style-test" />'); $this->assertFalse(file_exists($generated_uri), 'Generated file does not exist.'); - $this->drupalGet($url); + $this->drupalGet($this->getAbsoluteUrl($url)); $this->assertResponse(200, 'Image was generated at the URL.'); $this->assertTrue(file_exists($generated_uri), 'Generated file does exist after we accessed it.'); $image_file = $image_factory->get($generated_uri); @@ -153,7 +153,7 @@ function testImageDimensions() { $style->save(); $this->assertEqual($this->getImageTag($variables), '<img src="' . $url . '" width="45" height="90" alt="" class="image-style-test" />'); $this->assertFalse(file_exists($generated_uri), 'Generated file does not exist.'); - $this->drupalGet($url); + $this->drupalGet($this->getAbsoluteUrl($url)); $this->assertResponse(200, 'Image was generated at the URL.'); $this->assertTrue(file_exists($generated_uri), 'Generated file does exist after we accessed it.'); $image_file = $image_factory->get($generated_uri); @@ -174,7 +174,7 @@ function testImageDimensions() { $style->save(); $this->assertEqual($this->getImageTag($variables), '<img src="' . $url . '" alt="" class="image-style-test" />'); $this->assertFalse(file_exists($generated_uri), 'Generated file does not exist.'); - $this->drupalGet($url); + $this->drupalGet($this->getAbsoluteUrl($url)); $this->assertResponse(200, 'Image was generated at the URL.'); $this->assertTrue(file_exists($generated_uri), 'Generated file does exist after we accessed it.'); @@ -194,7 +194,7 @@ function testImageDimensions() { $style->save(); $this->assertEqual($this->getImageTag($variables), '<img src="' . $url . '" width="30" height="30" alt="" class="image-style-test" />'); $this->assertFalse(file_exists($generated_uri), 'Generated file does not exist.'); - $this->drupalGet($url); + $this->drupalGet($this->getAbsoluteUrl($url)); $this->assertResponse(200, 'Image was generated at the URL.'); $this->assertTrue(file_exists($generated_uri), 'Generated file does exist after we accessed it.'); $image_file = $image_factory->get($generated_uri); @@ -215,7 +215,7 @@ function testImageDimensions() { $style->save(); $this->assertEqual($this->getImageTag($variables), '<img src="' . $url . '" width="41" height="41" alt="" class="image-style-test" />'); $this->assertFalse(file_exists($generated_uri), 'Generated file does not exist.'); - $this->drupalGet($url); + $this->drupalGet($this->getAbsoluteUrl($url)); $this->assertResponse(200, 'Image was generated at the URL.'); $this->assertTrue(file_exists($generated_uri), 'Generated file does exist after we accessed it.'); $image_file = $image_factory->get($generated_uri); @@ -254,10 +254,10 @@ function testImageDimensions() { ]; // PNG original image. Should be resized to 100x100. $generated_uri = 'public://styles/test_uri/public/'. \Drupal::service('file_system')->basename($original_uri); - $url = $style->buildUrl($original_uri); + $url = file_url_transform_relative($style->buildUrl($original_uri)); $this->assertEqual($this->getImageTag($variables), '<img src="' . $url . '" width="100" height="100" alt="" class="image-style-test-uri" />'); $this->assertFalse(file_exists($generated_uri), 'Generated file does not exist.'); - $this->drupalGet($url); + $this->drupalGet($this->getAbsoluteUrl($url)); $this->assertResponse(200, 'Image was generated at the URL.'); $this->assertTrue(file_exists($generated_uri), 'Generated file does exist after we accessed it.'); $image_file = $image_factory->get($generated_uri); @@ -267,11 +267,11 @@ function testImageDimensions() { $file = $files[1]; $original_uri = file_unmanaged_copy($file->uri, 'public://', FILE_EXISTS_RENAME); $generated_uri = 'public://styles/test_uri/public/'. \Drupal::service('file_system')->basename($original_uri); - $url = $style->buildUrl($original_uri); + $url = file_url_transform_relative($style->buildUrl($original_uri)); $variables['#uri'] = $original_uri; $this->assertEqual($this->getImageTag($variables), '<img src="' . $url . '" width="50" height="50" alt="" class="image-style-test-uri" />'); $this->assertFalse(file_exists($generated_uri), 'Generated file does not exist.'); - $this->drupalGet($url); + $this->drupalGet($this->getAbsoluteUrl($url)); $this->assertResponse(200, 'Image was generated at the URL.'); $this->assertTrue(file_exists($generated_uri), 'Generated file does exist after we accessed it.'); $image_file = $image_factory->get($generated_uri); diff --git a/core/modules/image/src/Tests/ImageFieldDisplayTest.php b/core/modules/image/src/Tests/ImageFieldDisplayTest.php index 53ba09f076e59a9c1488586381bacfa47a6cc9f3..2ae0fa45fd185d27642002d68d376cd4da1450fe 100644 --- a/core/modules/image/src/Tests/ImageFieldDisplayTest.php +++ b/core/modules/image/src/Tests/ImageFieldDisplayTest.php @@ -127,6 +127,8 @@ function _testImageFieldFormatters($scheme) { $default_output = '<a href="' . file_create_url($image_uri) . '">' . $renderer->renderRoot($image) . '</a>'; $this->drupalGet('node/' . $nid); $this->assertCacheTag($file->getCacheTags()[0]); + // @todo Remove in https://www.drupal.org/node/2646744. + $this->assertCacheContext('url.site'); $cache_tags_header = $this->drupalGetHeader('X-Drupal-Cache-Tags'); $this->assertTrue(!preg_match('/ image_style\:/', $cache_tags_header), 'No image style cache tag found.'); $this->assertRaw($default_output, 'Image linked to file formatter displaying correctly on full node view.'); @@ -165,7 +167,7 @@ function _testImageFieldFormatters($scheme) { '//a[@href=:path]/img[@src=:url and @alt=:alt and @width=:width and @height=:height]', array( ':path' => $node->url(), - ':url' => file_create_url($image['#uri']), + ':url' => file_url_transform_relative(file_create_url($image['#uri'])), ':width' => $image['#width'], ':height' => $image['#height'], ':alt' => $alt, @@ -256,7 +258,7 @@ function testImageFieldSettings() { $node = $node_storage->load($nid); $file = $node->{$field_name}->entity; - $url = file_create_url(ImageStyle::load('medium')->buildUrl($file->getFileUri())); + $url = file_url_transform_relative(file_create_url(ImageStyle::load('medium')->buildUrl($file->getFileUri()))); $this->assertTrue($this->cssSelect('img[width=40][height=20][class=image-style-medium][src="' . $url . '"]')); // Add alt/title fields to the image and verify that they are displayed. diff --git a/core/modules/image/src/Tests/ImageThemeFunctionTest.php b/core/modules/image/src/Tests/ImageThemeFunctionTest.php index 060a8afb26c298fb8a6eb848bdf4d25da405440b..44f649ef679f732e83ea4bdee90a1a6bd75dba05 100644 --- a/core/modules/image/src/Tests/ImageThemeFunctionTest.php +++ b/core/modules/image/src/Tests/ImageThemeFunctionTest.php @@ -75,7 +75,7 @@ function testImageFormatterTheme() { // Create a style. $style = ImageStyle::create(array('name' => 'test', 'label' => 'Test')); $style->save(); - $url = $style->buildUrl($original_uri); + $url = file_url_transform_relative($style->buildUrl($original_uri)); // Create a test entity with the image field set. $entity = entity_create('entity_test'); @@ -137,7 +137,7 @@ function testImageStyleTheme() { // Create a style. $style = ImageStyle::create(array('name' => 'image_test', 'label' => 'Test')); $style->save(); - $url = $style->buildUrl($original_uri); + $url = file_url_transform_relative($style->buildUrl($original_uri)); // Create the base element that we'll use in the tests below. $base_element = array( diff --git a/core/modules/responsive_image/responsive_image.module b/core/modules/responsive_image/responsive_image.module index d36782d70f76dfc65d95a0059d2e19f3dbc9003b..7df1da81a9f63cd8ff637108fa664cfb607eee8a 100644 --- a/core/modules/responsive_image/responsive_image.module +++ b/core/modules/responsive_image/responsive_image.module @@ -391,7 +391,7 @@ function responsive_image_build_source_attributes(ImageInterface $image, array $ // Use the image width as key so we can sort the array later on. // Images within a srcset should be sorted from small to large, since // the first matching source will be used. - $srcset[intval($dimensions['width'])] = file_create_url(_responsive_image_image_style_url($image_style_name, $image->getSource())) . ' ' . $dimensions['width'] . 'w'; + $srcset[intval($dimensions['width'])] = _responsive_image_image_style_url($image_style_name, $image->getSource()) . ' ' . $dimensions['width'] . 'w'; $sizes = array_merge(explode(',', $image_style_mapping['image_mapping']['sizes']), $sizes); } break; @@ -405,7 +405,7 @@ function responsive_image_build_source_attributes(ImageInterface $image, array $ // be sorted from small to large, since the first matching source will // be used. We multiply it by 100 so multipliers with up to two decimals // can be used. - $srcset[intval(Unicode::substr($multiplier, 0, -1) * 100)] = file_create_url(_responsive_image_image_style_url($image_style_mapping['image_mapping'], $image->getSource())) . ' ' . $multiplier; + $srcset[intval(Unicode::substr($multiplier, 0, -1) * 100)] = _responsive_image_image_style_url($image_style_mapping['image_mapping'], $image->getSource()) . ' ' . $multiplier; break; } } @@ -495,9 +495,9 @@ function _responsive_image_image_style_url($style_name, $path) { } $entity = ImageStyle::load($style_name); if ($entity instanceof Drupal\image\Entity\ImageStyle) { - return $entity->buildUrl($path); + return file_url_transform_relative($entity->buildUrl($path)); } - return file_create_url($path); + return file_url_transform_relative(file_create_url($path)); } /** diff --git a/core/modules/responsive_image/src/Plugin/Field/FieldFormatter/ResponsiveImageFormatter.php b/core/modules/responsive_image/src/Plugin/Field/FieldFormatter/ResponsiveImageFormatter.php index d1290cb2a5fb76db0176b86cef2f587fe91a3161..e66cb82255e1ab2b6c30a5f39ea10a414da790b5 100644 --- a/core/modules/responsive_image/src/Plugin/Field/FieldFormatter/ResponsiveImageFormatter.php +++ b/core/modules/responsive_image/src/Plugin/Field/FieldFormatter/ResponsiveImageFormatter.php @@ -230,7 +230,7 @@ public function viewElements(FieldItemListInterface $items, $langcode) { foreach ($files as $delta => $file) { // Link the <picture> element to the original file. if (isset($link_file)) { - $url = Url::fromUri(file_create_url($file->getFileUri())); + $url = file_url_transform_relative(file_create_url($file->getFileUri())); } // Extract field item attributes for the theme function, and unset them // from the $item so that the field template does not re-render them. diff --git a/core/modules/responsive_image/src/Tests/ResponsiveImageFieldDisplayTest.php b/core/modules/responsive_image/src/Tests/ResponsiveImageFieldDisplayTest.php index f535f3f25cd5573a8a816a0d257ad83e5c34812a..e8e7aaee3ffbf775b8701c0dcba3a49ff92aeaf1 100644 --- a/core/modules/responsive_image/src/Tests/ResponsiveImageFieldDisplayTest.php +++ b/core/modules/responsive_image/src/Tests/ResponsiveImageFieldDisplayTest.php @@ -245,7 +245,7 @@ protected function doTestResponsiveImageFieldFormatters($scheme, $empty_styles = $display->setComponent($field_name, $display_options) ->save(); - $default_output = '<a href="' . file_create_url($image_uri) . '"><picture'; + $default_output = '<a href="' . file_url_transform_relative(file_create_url($image_uri)) . '"><picture'; $this->drupalGet('node/' . $nid); $cache_tags_header = $this->drupalGetHeader('X-Drupal-Cache-Tags'); $this->assertTrue(!preg_match('/ image_style\:/', $cache_tags_header), 'No image style cache tag found.'); @@ -290,10 +290,10 @@ protected function doTestResponsiveImageFieldFormatters($scheme, $empty_styles = $this->assertRaw(''); $thumbnail_style = ImageStyle::load('thumbnail'); // Assert the output of the 'srcset' attribute (small multipliers first). - $this->assertRaw(' 1x, ' . $thumbnail_style->buildUrl($image_uri) . ' 1.5x'); + $this->assertRaw(' 1x, ' . file_url_transform_relative($thumbnail_style->buildUrl($image_uri)) . ' 1.5x'); $this->assertRaw('/styles/medium/'); // Assert the output of the original image. - $this->assertRaw(file_create_url($image_uri) . ' 3x'); + $this->assertRaw(file_url_transform_relative(file_create_url($image_uri)) . ' 3x'); // Assert the output of the breakpoints. $this->assertRaw('media="(min-width: 0px)"'); $this->assertRaw('media="(min-width: 560px)"'); @@ -302,7 +302,7 @@ protected function doTestResponsiveImageFieldFormatters($scheme, $empty_styles = $this->assertPattern('/media="\(min-width: 560px\)".+?sizes="\(min-width: 700px\) 700px, 100vw"/'); // Assert the output of the 'srcset' attribute (small images first). $medium_style = ImageStyle::load('medium'); - $this->assertRaw($medium_style->buildUrl($image_uri) . ' 220w, ' . $large_style->buildUrl($image_uri) . ' 360w'); + $this->assertRaw(file_url_transform_relative($medium_style->buildUrl($image_uri)) . ' 220w, ' . file_url_transform_relative($large_style->buildUrl($image_uri)) . ' 360w'); $this->assertRaw('media="(min-width: 851px)"'); } $this->assertRaw('/styles/large/'); @@ -322,7 +322,7 @@ protected function doTestResponsiveImageFieldFormatters($scheme, $empty_styles = '#alt' => $alt, '#srcset' => array( array( - 'uri' => $large_style->buildUrl($image->getSource()), + 'uri' => file_url_transform_relative($large_style->buildUrl($image->getSource())), ), ), ); @@ -404,7 +404,7 @@ public function testResponsiveImageFieldFormattersEmptyMediaQuery() { $thumbnail_style = ImageStyle::load('thumbnail'); $node = $node_storage->load($nid); $image_uri = File::load($node->{$field_name}->target_id)->getFileUri(); - $this->assertPattern('/srcset="' . preg_quote($thumbnail_style->buildUrl($image_uri), '/') . ' 1x".+?media="\(min-width: 0px\)"/'); + $this->assertPattern('/srcset="' . preg_quote(file_url_transform_relative($thumbnail_style->buildUrl($image_uri)), '/') . ' 1x".+?media="\(min-width: 0px\)"/'); } /** @@ -450,7 +450,7 @@ public function testResponsiveImageFieldFormattersOneSource() { $medium_style = ImageStyle::load('medium'); $node = $node_storage->load($nid); $image_uri = File::load($node->{$field_name}->target_id)->getFileUri(); - $this->assertRaw('<img srcset="' . $medium_style->buildUrl($image_uri) . ' 1x, ' . $large_style->buildUrl($image_uri) . ' 2x"'); + $this->assertRaw('<img srcset="' . file_url_transform_relative($medium_style->buildUrl($image_uri)) . ' 1x, ' . file_url_transform_relative($large_style->buildUrl($image_uri)) . ' 2x"'); } /** @@ -510,7 +510,7 @@ private function assertResponsiveImageFieldFormattersLink($link_type) { switch ($link_type) { case 'file': // Make sure the link to the file is present. - $this->assertPattern('/<a(.*?)href="' . preg_quote(file_create_url($image_uri), '/') . '"(.*?)><picture/'); + $this->assertPattern('/<a(.*?)href="' . preg_quote(file_url_transform_relative(file_create_url($image_uri)), '/') . '"(.*?)><picture/'); break; case 'content': diff --git a/core/modules/simpletest/src/Tests/SimpleTestTest.php b/core/modules/simpletest/src/Tests/SimpleTestTest.php index 41d114ea6d10ef9c9c37d2efdcb5d858f9cd5bb6..812c1b249ac768efe7775ac4e17140037c74b124 100644 --- a/core/modules/simpletest/src/Tests/SimpleTestTest.php +++ b/core/modules/simpletest/src/Tests/SimpleTestTest.php @@ -333,7 +333,7 @@ function getTestResults() { $assertion['file'] = $this->asText($row->td[2]); $assertion['line'] = $this->asText($row->td[3]); $assertion['function'] = $this->asText($row->td[4]); - $ok_url = file_create_url('core/misc/icons/73b355/check.svg'); + $ok_url = file_url_transform_relative(file_create_url('core/misc/icons/73b355/check.svg')); $assertion['status'] = ($row->td[5]->img['src'] == $ok_url) ? 'Pass' : 'Fail'; $results['assertions'][] = $assertion; } diff --git a/core/modules/system/src/Tests/Common/AttachedAssetsTest.php b/core/modules/system/src/Tests/Common/AttachedAssetsTest.php index e3a516e322183326755671d5a903c64dd9c920ec..3ab2d31520cf7d6ad2a9b40bf0c79b936cecc1f7 100644 --- a/core/modules/system/src/Tests/Common/AttachedAssetsTest.php +++ b/core/modules/system/src/Tests/Common/AttachedAssetsTest.php @@ -96,8 +96,8 @@ function testAddFiles() { $rendered_css = $this->renderer->renderPlain($css_render_array); $rendered_js = $this->renderer->renderPlain($js_render_array); $query_string = $this->container->get('state')->get('system.css_js_query_string') ?: '0'; - $this->assertNotIdentical(strpos($rendered_css, '<link rel="stylesheet" href="' . file_create_url('core/modules/system/tests/modules/common_test/bar.css') . '?' . $query_string . '" media="all" />'), FALSE, 'Rendering an external CSS file.'); - $this->assertNotIdentical(strpos($rendered_js, '<script src="' . file_create_url('core/modules/system/tests/modules/common_test/foo.js') . '?' . $query_string . '"></script>'), FALSE, 'Rendering an external JavaScript file.'); + $this->assertNotIdentical(strpos($rendered_css, '<link rel="stylesheet" href="' . file_url_transform_relative(file_create_url('core/modules/system/tests/modules/common_test/bar.css')) . '?' . $query_string . '" media="all" />'), FALSE, 'Rendering an external CSS file.'); + $this->assertNotIdentical(strpos($rendered_js, '<script src="' . file_url_transform_relative(file_create_url('core/modules/system/tests/modules/common_test/foo.js')) . '?' . $query_string . '"></script>'), FALSE, 'Rendering an external JavaScript file.'); } /** @@ -150,7 +150,7 @@ function testAttributes() { $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js); $rendered_js = $this->renderer->renderPlain($js_render_array); $expected_1 = '<script src="http://example.com/deferred-external.js" foo="bar" defer></script>'; - $expected_2 = '<script src="' . file_create_url('core/modules/system/tests/modules/common_test/deferred-internal.js') . '?v=1" defer bar="foo"></script>'; + $expected_2 = '<script src="' . file_url_transform_relative(file_create_url('core/modules/system/tests/modules/common_test/deferred-internal.js')) . '?v=1" defer bar="foo"></script>'; $this->assertNotIdentical(strpos($rendered_js, $expected_1), FALSE, 'Rendered external JavaScript with correct defer and random attributes.'); $this->assertNotIdentical(strpos($rendered_js, $expected_2), FALSE, 'Rendered internal JavaScript with correct defer and random attributes.'); } @@ -166,7 +166,7 @@ function testAggregatedAttributes() { $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js); $rendered_js = $this->renderer->renderPlain($js_render_array); $expected_1 = '<script src="http://example.com/deferred-external.js" foo="bar" defer></script>'; - $expected_2 = '<script src="' . file_create_url('core/modules/system/tests/modules/common_test/deferred-internal.js') . '?v=1" defer bar="foo"></script>'; + $expected_2 = '<script src="' . file_url_transform_relative(file_create_url('core/modules/system/tests/modules/common_test/deferred-internal.js')) . '?v=1" defer bar="foo"></script>'; $this->assertNotIdentical(strpos($rendered_js, $expected_1), FALSE, 'Rendered external JavaScript with correct defer and random attributes.'); $this->assertNotIdentical(strpos($rendered_js, $expected_2), FALSE, 'Rendered internal JavaScript with correct defer and random attributes.'); } @@ -186,7 +186,7 @@ function testAggregation() { $rendered_footer_js = \Drupal::service('asset.js.collection_renderer')->render($footer_js); $this->assertEqual(2, count($rendered_footer_js), 'There are 2 JavaScript assets in the footer.'); $this->assertEqual('drupal-settings-json', $rendered_footer_js[0]['#attributes']['data-drupal-selector'], 'The first of the two JavaScript assets in the footer has drupal settings.'); - $this->assertEqual('http://', substr($rendered_footer_js[1]['#attributes']['src'], 0, 7), 'The second of the two JavaScript assets in the footer has the sole aggregated JavaScript asset.'); + $this->assertEqual(0, strpos($rendered_footer_js[1]['#attributes']['src'], base_path()), 'The second of the two JavaScript assets in the footer has the sole aggregated JavaScript asset.'); } /** @@ -237,9 +237,9 @@ function testHeaderHTML() { $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js); $rendered_js = $this->renderer->renderPlain($js_render_array); $query_string = $this->container->get('state')->get('system.css_js_query_string') ?: '0'; - $this->assertNotIdentical(strpos($rendered_js, '<script src="' . file_create_url('core/modules/system/tests/modules/common_test/header.js') . '?' . $query_string . '"></script>'), FALSE, 'The JS asset in common_test/js-header appears in the header.'); - $this->assertNotIdentical(strpos($rendered_js, '<script src="' . file_create_url('core/misc/drupal.js')), FALSE, 'The JS asset of the direct dependency (core/drupal) of common_test/js-header appears in the header.'); - $this->assertNotIdentical(strpos($rendered_js, '<script src="' . file_create_url('core/assets/vendor/domready/ready.min.js')), FALSE, 'The JS asset of the indirect dependency (core/domready) of common_test/js-header appears in the header.'); + $this->assertNotIdentical(strpos($rendered_js, '<script src="' . file_url_transform_relative(file_create_url('core/modules/system/tests/modules/common_test/header.js')) . '?' . $query_string . '"></script>'), FALSE, 'The JS asset in common_test/js-header appears in the header.'); + $this->assertNotIdentical(strpos($rendered_js, '<script src="' . file_url_transform_relative(file_create_url('core/misc/drupal.js'))), FALSE, 'The JS asset of the direct dependency (core/drupal) of common_test/js-header appears in the header.'); + $this->assertNotIdentical(strpos($rendered_js, '<script src="' . file_url_transform_relative(file_create_url('core/assets/vendor/domready/ready.min.js'))), FALSE, 'The JS asset of the indirect dependency (core/domready) of common_test/js-header appears in the header.'); } /** @@ -267,8 +267,8 @@ function testBrowserConditionalComments() { $js = $this->assetResolver->getJsAssets($assets, FALSE)[1]; $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js); $rendered_js = $this->renderer->renderPlain($js_render_array); - $expected_1 = "<!--[if lte IE 8]>\n" . '<script src="' . file_create_url('core/modules/system/tests/modules/common_test/old-ie.js') . '?' . $default_query_string . '"></script>' . "\n<![endif]-->"; - $expected_2 = "<!--[if !IE]><!-->\n" . '<script src="' . file_create_url('core/modules/system/tests/modules/common_test/no-ie.js') . '?' . $default_query_string . '"></script>' . "\n<!--<![endif]-->"; + $expected_1 = "<!--[if lte IE 8]>\n" . '<script src="' . file_url_transform_relative(file_create_url('core/modules/system/tests/modules/common_test/old-ie.js')) . '?' . $default_query_string . '"></script>' . "\n<![endif]-->"; + $expected_2 = "<!--[if !IE]><!-->\n" . '<script src="' . file_url_transform_relative(file_create_url('core/modules/system/tests/modules/common_test/no-ie.js')) . '?' . $default_query_string . '"></script>' . "\n<!--<![endif]-->"; $this->assertNotIdentical(strpos($rendered_js, $expected_1), FALSE, 'Rendered JavaScript within downlevel-hidden conditional comments.'); $this->assertNotIdentical(strpos($rendered_js, $expected_2), FALSE, 'Rendered JavaScript within downlevel-revealed conditional comments.'); @@ -476,8 +476,8 @@ function testAddJsFileWithQueryString() { $js_render_array = \Drupal::service('asset.js.collection_renderer')->render($js); $rendered_js = $this->renderer->renderPlain($js_render_array); $query_string = $this->container->get('state')->get('system.css_js_query_string') ?: '0'; - $this->assertNotIdentical(strpos($rendered_css, '<link rel="stylesheet" href="' . str_replace('&', '&', file_create_url('core/modules/system/tests/modules/common_test/querystring.css?arg1=value1&arg2=value2')) . '&' . $query_string . '" media="all" />'), FALSE, 'CSS file with query string gets version query string correctly appended..'); - $this->assertNotIdentical(strpos($rendered_js, '<script src="' . str_replace('&', '&', file_create_url('core/modules/system/tests/modules/common_test/querystring.js?arg1=value1&arg2=value2')) . '&' . $query_string . '"></script>'), FALSE, 'JavaScript file with query string gets version query string correctly appended.'); + $this->assertNotIdentical(strpos($rendered_css, '<link rel="stylesheet" href="' . str_replace('&', '&', file_url_transform_relative(file_create_url('core/modules/system/tests/modules/common_test/querystring.css?arg1=value1&arg2=value2'))) . '&' . $query_string . '" media="all" />'), FALSE, 'CSS file with query string gets version query string correctly appended..'); + $this->assertNotIdentical(strpos($rendered_js, '<script src="' . str_replace('&', '&', file_url_transform_relative(file_create_url('core/modules/system/tests/modules/common_test/querystring.js?arg1=value1&arg2=value2'))) . '&' . $query_string . '"></script>'), FALSE, 'JavaScript file with query string gets version query string correctly appended.'); } } diff --git a/core/modules/system/src/Tests/System/ThemeTest.php b/core/modules/system/src/Tests/System/ThemeTest.php index 5a079ec18c5e83ec4e5caa243768f66c4a6ec13c..c2f5a5750685c6c6609e51fe01ac4803bff5a98c 100644 --- a/core/modules/system/src/Tests/System/ThemeTest.php +++ b/core/modules/system/src/Tests/System/ThemeTest.php @@ -65,27 +65,27 @@ function testThemeSettings() { // Raw stream wrapper URI. $file->uri => array( 'form' => file_uri_target($file->uri), - 'src' => file_create_url($file->uri), + 'src' => file_url_transform_relative(file_create_url($file->uri)), ), // Relative path within the public filesystem. file_uri_target($file->uri) => array( 'form' => file_uri_target($file->uri), - 'src' => file_create_url($file->uri), + 'src' => file_url_transform_relative(file_create_url($file->uri)), ), // Relative path to a public file. $file_relative => array( 'form' => $file_relative, - 'src' => file_create_url($file->uri), + 'src' => file_url_transform_relative(file_create_url($file->uri)), ), // Relative path to an arbitrary file. 'core/misc/druplicon.png' => array( 'form' => 'core/misc/druplicon.png', - 'src' => $GLOBALS['base_url'] . '/' . 'core/misc/druplicon.png', + 'src' => base_path() . 'core/misc/druplicon.png', ), // Relative path to a file in a theme. $default_theme_path . '/logo.svg' => array( 'form' => $default_theme_path . '/logo.svg', - 'src' => $GLOBALS['base_url'] . '/' . $default_theme_path . '/logo.svg', + 'src' => base_path() . $default_theme_path . '/logo.svg', ), ); foreach ($supported_paths as $input => $expected) { @@ -186,7 +186,7 @@ function testThemeSettings() { ':rel' => 'home', ) ); - $this->assertEqual($elements[0]['src'], file_create_url($uploaded_filename)); + $this->assertEqual($elements[0]['src'], file_url_transform_relative(file_create_url($uploaded_filename))); $this->container->get('theme_handler')->install(array('bartik')); diff --git a/core/modules/system/src/Tests/Theme/EngineTwigTest.php b/core/modules/system/src/Tests/Theme/EngineTwigTest.php index 283b41d4b502f642590122ee8f8136656c16840b..4e2e563af7bbef7bf0bfae3dd87c25a5f097fd5a 100644 --- a/core/modules/system/src/Tests/Theme/EngineTwigTest.php +++ b/core/modules/system/src/Tests/Theme/EngineTwigTest.php @@ -125,7 +125,7 @@ public function testTwigUrlToString() { */ public function testTwigFileUrls() { $this->drupalGet('/twig-theme-test/file-url'); - $filepath = file_create_url('core/modules/system/tests/modules/twig_theme_test/twig_theme_test.js'); + $filepath = file_url_transform_relative(file_create_url('core/modules/system/tests/modules/twig_theme_test/twig_theme_test.js')); $this->assertRaw('<div>file_url: ' . $filepath . '</div>'); } diff --git a/core/modules/system/src/Tests/Theme/ImageTest.php b/core/modules/system/src/Tests/Theme/ImageTest.php index 87b403c75a005aed542fa1f3f0019008689ff107..d0f93df20d7da633f7eb82b2980cbc5db2577815 100644 --- a/core/modules/system/src/Tests/Theme/ImageTest.php +++ b/core/modules/system/src/Tests/Theme/ImageTest.php @@ -8,6 +8,7 @@ namespace Drupal\system\Tests\Theme; use Drupal\simpletest\KernelTestBase; +use Symfony\Component\HttpFoundation\Request; /** * Tests built-in image theme functions. @@ -32,9 +33,17 @@ class ImageTest extends KernelTestBase { protected function setUp() { parent::setUp(); + + // The code under test uses file_url_transform_relative(), which relies on + // the Request containing the correct hostname. KernelTestBase doesn't set + // it, so push another request onto the stack to ensure it's correct. + $request = Request::create('/', 'GET', [], [], [], $_SERVER); + $this->container = $this->kernel->getContainer(); + $this->container->get('request_stack')->push($request); + $this->testImages = array( - '/core/misc/druplicon.png', - '/core/misc/loading.gif', + 'core/misc/druplicon.png', + 'core/misc/loading.gif', ); } @@ -75,7 +84,7 @@ function testThemeImageWithSrc() { $this->render($image); // Make sure the src attribute has the correct value. - $this->assertRaw(file_create_url($image['#uri']), 'Correct output for an image with the src attribute.'); + $this->assertRaw(file_url_transform_relative(file_create_url($image['#uri'])), 'Correct output for an image with the src attribute.'); } /** @@ -103,7 +112,7 @@ function testThemeImageWithSrcsetMultiplier() { $this->render($image); // Make sure the srcset attribute has the correct value. - $this->assertRaw(file_create_url($this->testImages[0]) . ' 1x, ' . file_create_url($this->testImages[1]) . ' 2x', 'Correct output for image with srcset attribute and multipliers.'); + $this->assertRaw(file_url_transform_relative(file_create_url($this->testImages[0])) . ' 1x, ' . file_url_transform_relative(file_create_url($this->testImages[1])) . ' 2x', 'Correct output for image with srcset attribute and multipliers.'); } /** @@ -135,7 +144,7 @@ function testThemeImageWithSrcsetWidth() { $this->render($image); // Make sure the srcset attribute has the correct value. - $this->assertRaw(file_create_url($this->testImages[0]) . ' ' . $widths[0] . ', ' . file_create_url($this->testImages[1]) . ' ' . $widths[1], 'Correct output for image with srcset attribute and width descriptors.'); + $this->assertRaw(file_url_transform_relative(file_create_url($this->testImages[0])) . ' ' . $widths[0] . ', ' . file_url_transform_relative(file_create_url($this->testImages[1])) . ' ' . $widths[1], 'Correct output for image with srcset attribute and width descriptors.'); } } diff --git a/core/modules/system/src/Tests/Update/UpdatePathTestJavaScriptTest.php b/core/modules/system/src/Tests/Update/UpdatePathTestJavaScriptTest.php index f7ff4c336f9be787e2b53708f30eeb990a9755b6..d949dae7315c892b57855568deb613c0a1ec7338 100644 --- a/core/modules/system/src/Tests/Update/UpdatePathTestJavaScriptTest.php +++ b/core/modules/system/src/Tests/Update/UpdatePathTestJavaScriptTest.php @@ -43,7 +43,9 @@ protected function doSelectionTest() { if (!isset($script['src'])) { continue; } - $src = (string) $script['src']; + // Source is a root-relative URL. Transform it to an absolute URL to allow + // file_get_contents() to access the file. + $src = preg_replace('#^' . $GLOBALS['base_path'] . '(.*)#i', $GLOBALS['base_url'] . '/' . '${1}', (string) $script['src']); $file_content = file_get_contents($src); if (strpos($file_content, 'window.drupalSettings =') !== FALSE) { diff --git a/core/modules/user/src/Tests/UserPictureTest.php b/core/modules/user/src/Tests/UserPictureTest.php index bd28934e64998ab0d39da543f0b65a901e7bd848..9fa28a355b3daffe5efa49854a5702b7fcc9daff 100644 --- a/core/modules/user/src/Tests/UserPictureTest.php +++ b/core/modules/user/src/Tests/UserPictureTest.php @@ -100,7 +100,7 @@ function testPictureOnNodeComment() { $image_style_id = $this->config('core.entity_view_display.user.user.compact')->get('content.user_picture.settings.image_style'); $style = ImageStyle::load($image_style_id); - $image_url = $style->buildUrl($file->getfileUri()); + $image_url = file_url_transform_relative($style->buildUrl($file->getfileUri())); $alt_text = 'Profile picture for user ' . $this->webUser->getUsername(); // Verify that the image is displayed on the node page. diff --git a/core/tests/Drupal/Tests/Core/Asset/CssCollectionRendererUnitTest.php b/core/tests/Drupal/Tests/Core/Asset/CssCollectionRendererUnitTest.php index eab4fe4cd8b92f1bb12017969b0b7823e5e6ec63..48a92ee1e12c3baf9af1dee2d820c0204cd7dbae 100644 --- a/core/tests/Drupal/Tests/Core/Asset/CssCollectionRendererUnitTest.php +++ b/core/tests/Drupal/Tests/Core/Asset/CssCollectionRendererUnitTest.php @@ -9,9 +9,9 @@ namespace { /** - * CssRenderer uses file_create_url(), which *is* available when using the - * Simpletest test runner, but not when using the PHPUnit test runner; hence - * this hack. + * CssCollectionRenderer uses file_create_url() & file_url_transform_relative(), + * which *are* available when using the Simpletest test runner, but not when + * using the PHPUnit test runner; hence this hack. */ if (!function_exists('file_create_url')) { @@ -23,6 +23,17 @@ function file_create_url($uri) { return 'file_create_url:' . $uri; } +} +if (!function_exists('file_url_transform_relative')) { + + /** + * Temporary mock of file_url_transform_relative, until that is moved into + * Component/Utility. + */ + function file_url_transform_relative($uri) { + return 'file_url_transform_relative:' . $uri; + } + } } @@ -153,7 +164,7 @@ function providerTestRender() { 0 => array('group' => 0, 'type' => 'file', 'media' => 'all', 'preprocess' => TRUE, 'data' => 'public://css/file-all', 'browsers' => array()), ), array( - 0 => $create_link_element(file_create_url('public://css/file-all') . '?0', 'all'), + 0 => $create_link_element(file_url_transform_relative(file_create_url('public://css/file-all')) . '?0', 'all'), ), ), // 31 file CSS assets: expect 31 link elements. @@ -192,37 +203,37 @@ function providerTestRender() { 30 => $create_file_css_asset('public://css/31.css'), ), array( - 0 => $create_link_element(file_create_url('public://css/1.css') . '?0'), - 1 => $create_link_element(file_create_url('public://css/2.css') . '?0'), - 2 => $create_link_element(file_create_url('public://css/3.css') . '?0'), - 3 => $create_link_element(file_create_url('public://css/4.css') . '?0'), - 4 => $create_link_element(file_create_url('public://css/5.css') . '?0'), - 5 => $create_link_element(file_create_url('public://css/6.css') . '?0'), - 6 => $create_link_element(file_create_url('public://css/7.css') . '?0'), - 7 => $create_link_element(file_create_url('public://css/8.css') . '?0'), - 8 => $create_link_element(file_create_url('public://css/9.css') . '?0'), - 9 => $create_link_element(file_create_url('public://css/10.css') . '?0'), - 10 => $create_link_element(file_create_url('public://css/11.css') . '?0'), - 11 => $create_link_element(file_create_url('public://css/12.css') . '?0'), - 12 => $create_link_element(file_create_url('public://css/13.css') . '?0'), - 13 => $create_link_element(file_create_url('public://css/14.css') . '?0'), - 14 => $create_link_element(file_create_url('public://css/15.css') . '?0'), - 15 => $create_link_element(file_create_url('public://css/16.css') . '?0'), - 16 => $create_link_element(file_create_url('public://css/17.css') . '?0'), - 17 => $create_link_element(file_create_url('public://css/18.css') . '?0'), - 18 => $create_link_element(file_create_url('public://css/19.css') . '?0'), - 19 => $create_link_element(file_create_url('public://css/20.css') . '?0'), - 20 => $create_link_element(file_create_url('public://css/21.css') . '?0'), - 21 => $create_link_element(file_create_url('public://css/22.css') . '?0'), - 22 => $create_link_element(file_create_url('public://css/23.css') . '?0'), - 23 => $create_link_element(file_create_url('public://css/24.css') . '?0'), - 24 => $create_link_element(file_create_url('public://css/25.css') . '?0'), - 25 => $create_link_element(file_create_url('public://css/26.css') . '?0'), - 26 => $create_link_element(file_create_url('public://css/27.css') . '?0'), - 27 => $create_link_element(file_create_url('public://css/28.css') . '?0'), - 28 => $create_link_element(file_create_url('public://css/29.css') . '?0'), - 29 => $create_link_element(file_create_url('public://css/30.css') . '?0'), - 30 => $create_link_element(file_create_url('public://css/31.css') . '?0'), + 0 => $create_link_element(file_url_transform_relative(file_create_url('public://css/1.css')) . '?0'), + 1 => $create_link_element(file_url_transform_relative(file_create_url('public://css/2.css')) . '?0'), + 2 => $create_link_element(file_url_transform_relative(file_create_url('public://css/3.css')) . '?0'), + 3 => $create_link_element(file_url_transform_relative(file_create_url('public://css/4.css')) . '?0'), + 4 => $create_link_element(file_url_transform_relative(file_create_url('public://css/5.css')) . '?0'), + 5 => $create_link_element(file_url_transform_relative(file_create_url('public://css/6.css')) . '?0'), + 6 => $create_link_element(file_url_transform_relative(file_create_url('public://css/7.css')) . '?0'), + 7 => $create_link_element(file_url_transform_relative(file_create_url('public://css/8.css')) . '?0'), + 8 => $create_link_element(file_url_transform_relative(file_create_url('public://css/9.css')) . '?0'), + 9 => $create_link_element(file_url_transform_relative(file_create_url('public://css/10.css')) . '?0'), + 10 => $create_link_element(file_url_transform_relative(file_create_url('public://css/11.css')) . '?0'), + 11 => $create_link_element(file_url_transform_relative(file_create_url('public://css/12.css')) . '?0'), + 12 => $create_link_element(file_url_transform_relative(file_create_url('public://css/13.css')) . '?0'), + 13 => $create_link_element(file_url_transform_relative(file_create_url('public://css/14.css')) . '?0'), + 14 => $create_link_element(file_url_transform_relative(file_create_url('public://css/15.css')) . '?0'), + 15 => $create_link_element(file_url_transform_relative(file_create_url('public://css/16.css')) . '?0'), + 16 => $create_link_element(file_url_transform_relative(file_create_url('public://css/17.css')) . '?0'), + 17 => $create_link_element(file_url_transform_relative(file_create_url('public://css/18.css')) . '?0'), + 18 => $create_link_element(file_url_transform_relative(file_create_url('public://css/19.css')) . '?0'), + 19 => $create_link_element(file_url_transform_relative(file_create_url('public://css/20.css')) . '?0'), + 20 => $create_link_element(file_url_transform_relative(file_create_url('public://css/21.css')) . '?0'), + 21 => $create_link_element(file_url_transform_relative(file_create_url('public://css/22.css')) . '?0'), + 22 => $create_link_element(file_url_transform_relative(file_create_url('public://css/23.css')) . '?0'), + 23 => $create_link_element(file_url_transform_relative(file_create_url('public://css/24.css')) . '?0'), + 24 => $create_link_element(file_url_transform_relative(file_create_url('public://css/25.css')) . '?0'), + 25 => $create_link_element(file_url_transform_relative(file_create_url('public://css/26.css')) . '?0'), + 26 => $create_link_element(file_url_transform_relative(file_create_url('public://css/27.css')) . '?0'), + 27 => $create_link_element(file_url_transform_relative(file_create_url('public://css/28.css')) . '?0'), + 28 => $create_link_element(file_url_transform_relative(file_create_url('public://css/29.css')) . '?0'), + 29 => $create_link_element(file_url_transform_relative(file_create_url('public://css/30.css')) . '?0'), + 30 => $create_link_element(file_url_transform_relative(file_create_url('public://css/31.css')) . '?0'), ), ), // 32 file CSS assets with the same properties: expect 2 style elements. @@ -263,40 +274,40 @@ function providerTestRender() { ), array( 0 => $create_style_element(' -@import url("' . file_create_url('public://css/1.css') . '?0"); -@import url("' . file_create_url('public://css/2.css') . '?0"); -@import url("' . file_create_url('public://css/3.css') . '?0"); -@import url("' . file_create_url('public://css/4.css') . '?0"); -@import url("' . file_create_url('public://css/5.css') . '?0"); -@import url("' . file_create_url('public://css/6.css') . '?0"); -@import url("' . file_create_url('public://css/7.css') . '?0"); -@import url("' . file_create_url('public://css/8.css') . '?0"); -@import url("' . file_create_url('public://css/9.css') . '?0"); -@import url("' . file_create_url('public://css/10.css') . '?0"); -@import url("' . file_create_url('public://css/11.css') . '?0"); -@import url("' . file_create_url('public://css/12.css') . '?0"); -@import url("' . file_create_url('public://css/13.css') . '?0"); -@import url("' . file_create_url('public://css/14.css') . '?0"); -@import url("' . file_create_url('public://css/15.css') . '?0"); -@import url("' . file_create_url('public://css/16.css') . '?0"); -@import url("' . file_create_url('public://css/17.css') . '?0"); -@import url("' . file_create_url('public://css/18.css') . '?0"); -@import url("' . file_create_url('public://css/19.css') . '?0"); -@import url("' . file_create_url('public://css/20.css') . '?0"); -@import url("' . file_create_url('public://css/21.css') . '?0"); -@import url("' . file_create_url('public://css/22.css') . '?0"); -@import url("' . file_create_url('public://css/23.css') . '?0"); -@import url("' . file_create_url('public://css/24.css') . '?0"); -@import url("' . file_create_url('public://css/25.css') . '?0"); -@import url("' . file_create_url('public://css/26.css') . '?0"); -@import url("' . file_create_url('public://css/27.css') . '?0"); -@import url("' . file_create_url('public://css/28.css') . '?0"); -@import url("' . file_create_url('public://css/29.css') . '?0"); -@import url("' . file_create_url('public://css/30.css') . '?0"); -@import url("' . file_create_url('public://css/31.css') . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/1.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/2.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/3.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/4.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/5.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/6.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/7.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/8.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/9.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/10.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/11.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/12.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/13.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/14.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/15.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/16.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/17.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/18.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/19.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/20.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/21.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/22.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/23.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/24.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/25.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/26.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/27.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/28.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/29.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/30.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/31.css')) . '?0"); ', 'all'), 1 => $create_style_element(' -@import url("' . file_create_url('public://css/32.css') . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/32.css')) . '?0"); ', 'all'), ), ), @@ -340,46 +351,46 @@ function providerTestRender() { ), array( 0 => $create_style_element(' -@import url("' . file_create_url('public://css/1.css') . '?0"); -@import url("' . file_create_url('public://css/2.css') . '?0"); -@import url("' . file_create_url('public://css/3.css') . '?0"); -@import url("' . file_create_url('public://css/4.css') . '?0"); -@import url("' . file_create_url('public://css/5.css') . '?0"); -@import url("' . file_create_url('public://css/6.css') . '?0"); -@import url("' . file_create_url('public://css/7.css') . '?0"); -@import url("' . file_create_url('public://css/8.css') . '?0"); -@import url("' . file_create_url('public://css/9.css') . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/1.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/2.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/3.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/4.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/5.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/6.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/7.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/8.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/9.css')) . '?0"); ', 'all'), 1 => $create_style_element(' -@import url("' . file_create_url('public://css/10.css') . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/10.css')) . '?0"); ', 'screen'), 2 => $create_style_element(' -@import url("' . file_create_url('public://css/11.css') . '?0"); -@import url("' . file_create_url('public://css/12.css') . '?0"); -@import url("' . file_create_url('public://css/13.css') . '?0"); -@import url("' . file_create_url('public://css/14.css') . '?0"); -@import url("' . file_create_url('public://css/15.css') . '?0"); -@import url("' . file_create_url('public://css/16.css') . '?0"); -@import url("' . file_create_url('public://css/17.css') . '?0"); -@import url("' . file_create_url('public://css/18.css') . '?0"); -@import url("' . file_create_url('public://css/19.css') . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/11.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/12.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/13.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/14.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/15.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/16.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/17.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/18.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/19.css')) . '?0"); ', 'all'), 3 => $create_style_element(' -@import url("' . file_create_url('public://css/20.css') . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/20.css')) . '?0"); ', 'print'), 4 => $create_style_element(' -@import url("' . file_create_url('public://css/21.css') . '?0"); -@import url("' . file_create_url('public://css/22.css') . '?0"); -@import url("' . file_create_url('public://css/23.css') . '?0"); -@import url("' . file_create_url('public://css/24.css') . '?0"); -@import url("' . file_create_url('public://css/25.css') . '?0"); -@import url("' . file_create_url('public://css/26.css') . '?0"); -@import url("' . file_create_url('public://css/27.css') . '?0"); -@import url("' . file_create_url('public://css/28.css') . '?0"); -@import url("' . file_create_url('public://css/29.css') . '?0"); -@import url("' . file_create_url('public://css/30.css') . '?0"); -@import url("' . file_create_url('public://css/31.css') . '?0"); -@import url("' . file_create_url('public://css/32.css') . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/21.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/22.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/23.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/24.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/25.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/26.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/27.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/28.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/29.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/30.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/31.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/32.css')) . '?0"); ', 'all'), ), ), @@ -422,40 +433,40 @@ function providerTestRender() { ), array( 0 => $create_style_element(' -@import url("' . file_create_url('public://css/1.css') . '?0"); -@import url("' . file_create_url('public://css/2.css') . '?0"); -@import url("' . file_create_url('public://css/3.css') . '?0"); -@import url("' . file_create_url('public://css/4.css') . '?0"); -@import url("' . file_create_url('public://css/5.css') . '?0"); -@import url("' . file_create_url('public://css/6.css') . '?0"); -@import url("' . file_create_url('public://css/7.css') . '?0"); -@import url("' . file_create_url('public://css/8.css') . '?0"); -@import url("' . file_create_url('public://css/9.css') . '?0"); -@import url("' . file_create_url('public://css/10.css') . '?0"); -@import url("' . file_create_url('public://css/11.css') . '?0"); -@import url("' . file_create_url('public://css/12.css') . '?0"); -@import url("' . file_create_url('public://css/13.css') . '?0"); -@import url("' . file_create_url('public://css/14.css') . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/1.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/2.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/3.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/4.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/5.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/6.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/7.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/8.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/9.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/10.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/11.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/12.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/13.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/14.css')) . '?0"); ', 'all'), - 1 => $create_link_element(file_create_url('public://css/15.css') . '?0'), + 1 => $create_link_element(file_url_transform_relative(file_create_url('public://css/15.css')) . '?0'), 2 => $create_style_element(' -@import url("' . file_create_url('public://css/16.css') . '?0"); -@import url("' . file_create_url('public://css/17.css') . '?0"); -@import url("' . file_create_url('public://css/18.css') . '?0"); -@import url("' . file_create_url('public://css/19.css') . '?0"); -@import url("' . file_create_url('public://css/20.css') . '?0"); -@import url("' . file_create_url('public://css/21.css') . '?0"); -@import url("' . file_create_url('public://css/22.css') . '?0"); -@import url("' . file_create_url('public://css/23.css') . '?0"); -@import url("' . file_create_url('public://css/24.css') . '?0"); -@import url("' . file_create_url('public://css/25.css') . '?0"); -@import url("' . file_create_url('public://css/26.css') . '?0"); -@import url("' . file_create_url('public://css/27.css') . '?0"); -@import url("' . file_create_url('public://css/28.css') . '?0"); -@import url("' . file_create_url('public://css/29.css') . '?0"); -@import url("' . file_create_url('public://css/30.css') . '?0"); -@import url("' . file_create_url('public://css/31.css') . '?0"); -@import url("' . file_create_url('public://css/32.css') . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/16.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/17.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/18.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/19.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/20.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/21.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/22.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/23.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/24.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/25.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/26.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/27.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/28.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/29.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/30.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/31.css')) . '?0"); +@import url("' . file_url_transform_relative(file_create_url('public://css/32.css')) . '?0"); ', 'all'), ), ), diff --git a/core/tests/Drupal/Tests/Core/Asset/CssOptimizerUnitTest.php b/core/tests/Drupal/Tests/Core/Asset/CssOptimizerUnitTest.php index 03bb980f32cc2a96fe97c9614a5132613664f8ce..5922bf1f67583bbdd07e54797554b5d596a46021 100644 --- a/core/tests/Drupal/Tests/Core/Asset/CssOptimizerUnitTest.php +++ b/core/tests/Drupal/Tests/Core/Asset/CssOptimizerUnitTest.php @@ -9,7 +9,8 @@ namespace { /** - * CssOptimizer uses file_create_url(), which *is* available when using the + * CssOptimizer uses file_create_url(), file_uri_scheme() and + * file_url_transform_relative(), which *are* available when using the * Simpletest test runner, but not when using the PHPUnit test runner; hence * this hack. */ @@ -31,6 +32,17 @@ function file_uri_scheme($uri) { return FALSE; } +} +if (!function_exists('file_url_transform_relative')) { + + /** + * Temporary mock of file_url_transform_relative, until that is moved into + * Component/Utility. + */ + function file_url_transform_relative($uri) { + return 'file_url_transform_relative:' . $uri; + } + } } @@ -48,6 +60,11 @@ function file_uri_scheme($uri) { */ class CssOptimizerUnitTest extends UnitTestCase { + /** + * {@inheritdoc} + */ + protected $backupGlobals = FALSE; + /** * A CSS asset optimizer. * @@ -65,7 +82,8 @@ protected function setUp() { * Provides data for the CSS asset optimizing test. */ function providerTestOptimize() { - $path = dirname(__FILE__) . '/css_test_files/'; + $path = 'core/tests/Drupal/Tests/Core/Asset/css_test_files/'; + $absolute_path = dirname(__FILE__) . '/css_test_files/'; return array( // File. Tests: // - Stripped comments and white-space. @@ -82,7 +100,7 @@ function providerTestOptimize() { 'browsers' => array('IE' => TRUE, '!IE' => TRUE), 'basename' => 'css_input_without_import.css', ), - file_get_contents($path . 'css_input_without_import.css.optimized.css'), + file_get_contents($absolute_path . 'css_input_without_import.css.optimized.css'), ), // File. Tests: // - Proper URLs in imported files. (https://www.drupal.org/node/265719) @@ -102,7 +120,7 @@ function providerTestOptimize() { 'browsers' => array('IE' => TRUE, '!IE' => TRUE), 'basename' => 'css_input_with_import.css', ), - str_replace('url(images/icon.png)', 'url(' . file_create_url($path . 'images/icon.png') . ')', file_get_contents($path . 'css_input_with_import.css.optimized.css')), + str_replace('url(images/icon.png)', 'url(' . file_url_transform_relative(file_create_url($path . 'images/icon.png')) . ')', file_get_contents($absolute_path . 'css_input_with_import.css.optimized.css')), ), // File. Tests: // - Retain comment hacks. @@ -117,7 +135,7 @@ function providerTestOptimize() { 'browsers' => array('IE' => TRUE, '!IE' => TRUE), 'basename' => 'comment_hacks.css', ), - file_get_contents($path . 'comment_hacks.css.optimized.css'), + file_get_contents($absolute_path . 'comment_hacks.css.optimized.css'), ), // File in subfolder. Tests: // - CSS import path is properly interpreted. @@ -134,7 +152,7 @@ function providerTestOptimize() { 'browsers' => array('IE' => TRUE, '!IE' => TRUE), 'basename' => 'css_input_with_import.css', ), - str_replace('url(../images/icon.png)', 'url(' . file_create_url($path . 'images/icon.png') . ')', file_get_contents($path . 'css_subfolder/css_input_with_import.css.optimized.css')), + str_replace('url(../images/icon.png)', 'url(' . file_url_transform_relative(file_create_url($path . 'images/icon.png')) . ')', file_get_contents($absolute_path . 'css_subfolder/css_input_with_import.css.optimized.css')), ), // File. Tests: // - Any @charaset declaration at the beginning of a file should be @@ -150,7 +168,7 @@ function providerTestOptimize() { 'browsers' => array('IE' => TRUE, '!IE' => TRUE), 'basename' => 'charset_sameline.css', ), - file_get_contents($path . 'charset.css.optimized.css'), + file_get_contents($absolute_path . 'charset.css.optimized.css'), ), array( array( @@ -163,7 +181,7 @@ function providerTestOptimize() { 'browsers' => array('IE' => TRUE, '!IE' => TRUE), 'basename' => 'charset_newline.css', ), - file_get_contents($path . 'charset.css.optimized.css'), + file_get_contents($absolute_path . 'charset.css.optimized.css'), ), array( array( @@ -226,7 +244,18 @@ function providerTestOptimize() { * @dataProvider providerTestOptimize */ function testOptimize($css_asset, $expected) { + global $base_path; + $original_base_path = $base_path; + $base_path = '/'; + + // \Drupal\Core\Asset\CssOptimizer::loadFile() relies on the current working + // directory being the one that is used when index.php is the entry point. + // Note: PHPUnit automatically restores the original working directory. + chdir(realpath(__DIR__ . '/../../../../../../')); + $this->assertEquals($expected, $this->optimizer->optimize($css_asset), 'Group of file CSS assets optimized correctly.'); + + $base_path = $original_base_path; } /**