FIX Do not insert requirements more than once in includeInHTML

This change consolidates the string replacements used to insert requirements into the page content to help ensure
that they are not compounding and overwriting eachother.

The added test case includes where a user may have a Javascript snippet that contains a closing head tag, and the
test ensures that it does not get injected with requirements as well as the actual head tag in the DOM.
This commit is contained in:
Robbie Averill 2017-04-07 12:17:38 +12:00 committed by Sam Minnée
parent 2fd3b84b46
commit 55eb7ebdcc
2 changed files with 57 additions and 13 deletions

View File

@ -452,6 +452,36 @@ class RequirementsTest extends SapphireTest {
$this->assertNotRegexp('/RequirementsTest_b\.css\?m=[\d]*&foo=bar&bla=blubb"/', $html); $this->assertNotRegexp('/RequirementsTest_b\.css\?m=[\d]*&foo=bar&bla=blubb"/', $html);
} }
/**
* Ensure that if a JS snippet somewhere in the page (via requirements) contains </head> that it doesn't
* get injected with requirements
*/
public function testHeadTagIsNotInjectedTwice() {
$template = '<html><head></head><body><header>My header</header><p>Body</p></body></html>';
$basePath = $this->getCurrentRelativePath();
$backend = new Requirements_Backend;
// The offending snippet that contains a closing head tag
$backend->customScript('var myvar="<head></head>";');
// Insert a variety of random requirements
$backend->css($basePath .'/RequirementsTest_a.css');
$backend->javascript($basePath .'/RequirementsTest_a.js');
$backend->insertHeadTags('<link rel="alternate" type="application/atom+xml" title="test" href="/" />');
// Case A: JS forced to bottom
$backend->set_force_js_to_bottom(true);
$this->assertContains('var myvar="<head></head>";', $backend->includeInHTML(false, $template));
// Case B: JS written to body
$backend->set_force_js_to_bottom(false);
$backend->set_write_js_to_body(true);
$this->assertContains('var myvar="<head></head>";', $backend->includeInHTML(false, $template));
// Case C: Neither of the above
$backend->set_write_js_to_body(false);
$this->assertContains('var myvar="<head></head>";', $backend->includeInHTML(false, $template));
}
public function assertFileIncluded($backend, $type, $files) { public function assertFileIncluded($backend, $type, $files) {
$type = strtolower($type); $type = strtolower($type);
switch (strtolower($type)) { switch (strtolower($type)) {

View File

@ -863,19 +863,18 @@ class Requirements_Backend {
$requirements .= "$customHeadTag\n"; $requirements .= "$customHeadTag\n";
} }
$replacements = array();
if ($this->force_js_to_bottom) { if ($this->force_js_to_bottom) {
// Remove all newlines from code to preserve layout $jsRequirements = $this->removeNewlinesFromCode($jsRequirements);
$jsRequirements = preg_replace('/>\n*/', '>', $jsRequirements);
// Forcefully put the scripts at the bottom of the body instead of before the first // Forcefully put the scripts at the bottom of the body instead of before the first
// script tag. // script tag.
$content = preg_replace("/(<\/body[^>]*>)/i", $jsRequirements . "\\1", $content); $replacements["/(<\/body[^>]*>)/i"] = $jsRequirements . "\\1";
// Put CSS at the bottom of the head // Put CSS at the bottom of the head
$content = preg_replace("/(<\/head>)/i", $requirements . "\\1", $content); $replacements["/(<\/head>)/i"] = $requirements . "\\1";
} elseif ($this->write_js_to_body) { } elseif ($this->write_js_to_body) {
// Remove all newlines from code to preserve layout $jsRequirements = $this->removeNewlinesFromCode($jsRequirements);
$jsRequirements = preg_replace('/>\n*/', '>', $jsRequirements);
// If your template already has script tags in the body, then we try to put our script // If your template already has script tags in the body, then we try to put our script
// tags just before those. Otherwise, we put it at the bottom. // tags just before those. Otherwise, we put it at the bottom.
@ -895,20 +894,35 @@ class Requirements_Backend {
if ($canWriteToBody) { if ($canWriteToBody) {
$content = substr($content, 0, $p1) . $jsRequirements . substr($content, $p1); $content = substr($content, 0, $p1) . $jsRequirements . substr($content, $p1);
} else { } else {
$content = preg_replace("/(<\/body[^>]*>)/i", $jsRequirements . "\\1", $content); $replacements["/(<\/body[^>]*>)/i"] = $jsRequirements . "\\1";
} }
// Put CSS at the bottom of the head // Put CSS at the bottom of the head
$content = preg_replace("/(<\/head>)/i", $requirements . "\\1", $content); $replacements["/(<\/head>)/i"] = $requirements . "\\1";
} else { } else {
$content = preg_replace("/(<\/head>)/i", $requirements . "\\1", $content); // Put CSS and Javascript together before the closing head tag
$content = preg_replace("/(<\/head>)/i", $jsRequirements . "\\1", $content); $replacements["/(<\/head>)/i"] = $requirements . $jsRequirements. "\\1";
}
if (!empty($replacements)) {
// Replace everything at once (only once)
$content = preg_replace(array_keys($replacements), array_values($replacements), $content, 1);
} }
} }
return $content; return $content;
} }
/**
* Remove all newlines from code to preserve layout
*
* @param string $code
* @return string
*/
protected function removeNewlinesFromCode($code) {
return preg_replace('/>\n*/', '>', $code);
}
/** /**
* Attach requirements inclusion to X-Include-JS and X-Include-CSS headers on the given * Attach requirements inclusion to X-Include-JS and X-Include-CSS headers on the given
* HTTP Response * HTTP Response