chromium/third_party/google-closure-library/closure/goog/dom/safe_test.js

/**
 * @license
 * Copyright The Closure Library Authors.
 * SPDX-License-Identifier: Apache-2.0
 */

/** @fileoverview Unit tests for safe. */

goog.module('goog.dom.safeTest');
goog.setTestOnly();

const Const = goog.require('goog.string.Const');
const InsertAdjacentHtmlPosition = goog.require('goog.dom.safe.InsertAdjacentHtmlPosition');
const SafeHtml = goog.require('goog.html.SafeHtml');
const SafeScript = goog.require('goog.html.SafeScript');
const SafeStyle = goog.require('goog.html.SafeStyle');
const SafeUrl = goog.require('goog.html.SafeUrl');
const TagName = goog.require('goog.dom.TagName');
const TrustedResourceUrl = goog.require('goog.html.TrustedResourceUrl');
const asserts = goog.require('goog.asserts');
const dom = goog.require('goog.dom');
const googString = goog.require('goog.string');
const googTesting = goog.require('goog.testing');
const safe = goog.require('goog.dom.safe');
const testSuite = goog.require('goog.testing.testSuite');
const testing = goog.require('goog.html.testing');
const userAgent = goog.require('goog.userAgent');

let mockWindowOpen;

/**
 * Returns a link element, incorrectly typed as a Location.
 * @return {!Location}
 * @suppress {checkTypes}
 */
function makeLinkElementTypedAsLocation() {
  return document.createElement('LINK');
}

/**
 * Tests that f raises an AssertionError and runs f while disabling assertion
 * errors.
 * @param {function():*} f function with a failing assertion.
 * @return {*} the return value of f.
 */
function withAssertionFailure(f) {
  assertThrows(f);
  asserts.setErrorHandler((error) => {});
  try {
    return f();
  } finally {
    asserts.setErrorHandler(asserts.DEFAULT_ERROR_HANDLER);
  }
}
testSuite({
  tearDown() {
    if (mockWindowOpen) {
      mockWindowOpen.$tearDown();
    }
  },

  testInsertAdjacentHtml() {
    let writtenHtml;
    let writtenPosition;
    const mockNode = /** @type {!Node} */ ({
      'insertAdjacentHTML': function(position, html) {
        writtenPosition = position;
        writtenHtml = html.toString();
      },
    });

    safe.insertAdjacentHtml(
        mockNode, InsertAdjacentHtmlPosition.BEFOREBEGIN,
        SafeHtml.create('div', {}, 'foobar'));
    assertEquals('<div>foobar</div>', writtenHtml);
    assertEquals('beforebegin', writtenPosition);
  },

  testSetInnerHtml() {
    const mockElement =
        /** @type {!Element} */ ({'tagName': 'DIV', 'innerHTML': 'blarg'});
    const html = '<script>somethingTrusted();<' +
        '/script>';
    const safeHtml = testing.newSafeHtmlForTest(html);
    safe.setInnerHtml(mockElement, safeHtml);
    assertEquals(html, mockElement.innerHTML.toString());
  },

  testSetInnerHtml_doesntAllowScript() {
    const script =
        /** @type {!Element} */ ({'tagName': 'SCRIPT', 'innerHTML': 'blarg'});
    const safeHtml = SafeHtml.htmlEscape('alert(1);');
    assertThrows(() => {
      safe.setInnerHtml(script, safeHtml);
    });
  },

  testSetInnerHtml_doesntAllowStyle() {
    const style =
        /** @type {!Element} */ ({'tagName': 'STYLE', 'innerHTML': 'blarg'});
    const safeHtml = SafeHtml.htmlEscape('A { color: red; }');
    assertThrows(() => {
      safe.setInnerHtml(style, safeHtml);
    });
  },

  /**
   * When innerHTML is assigned on an element in IE, IE recursively severs all
   * parent-children links in the removed content. This test ensures that that
   * doesn't happen when re-rendering an element with soy.
   */
  testSetInnerHtml_leavesChildrenInIE() {
    // Given a div with existing content.
    const grandchildDiv = dom.createElement(TagName.DIV);
    const childDiv = dom.createDom(TagName.DIV, null, [grandchildDiv]);
    const testDiv = dom.createDom(TagName.DIV, null, [childDiv]);
    // Expect parent/children links.
    assertArrayEquals(
        'Expect testDiv to contain childDiv.', [childDiv],
        Array.from(testDiv.children));
    assertEquals(
        'Expect childDiv to be contained in testDiv.', testDiv,
        childDiv.parentElement);
    assertArrayEquals(
        'Expect childDiv to contain grandchildDiv.', [grandchildDiv],
        Array.from(childDiv.children));
    assertEquals(
        'Expect grandchildDiv to be contained in childDiv.', childDiv,
        grandchildDiv.parentElement);

    // When the div's content is re-rendered.
    const safeHtml = testing.newSafeHtmlForTest('<a></a>');
    safe.setInnerHtml(testDiv, safeHtml);
    assertEquals(
        `Expect testDiv's contents to complete change`, '<a></a>',
        testDiv.innerHTML.toLowerCase());
    // Expect the previous childDiv tree to retain its parent-child connections.
    assertArrayEquals(
        'Expect childDiv to still contain grandchildDiv.', [grandchildDiv],
        Array.from(childDiv.children));
    assertEquals(
        'Expect grandchildDiv to still be contained in childDiv.', childDiv,
        grandchildDiv.parentElement);
  },

  testSetInnerHtmlFromConstant() {
    const element = document.createElement('div');
    const html = '<b>c</b>';
    safe.setInnerHtmlFromConstant(element, Const.from(html));
    assertEquals(html, element.innerHTML);
  },

  testSetStyle() {
    const style = SafeStyle.fromConstant(Const.from('color: red;'));
    const elem = document.createElement('div');
    assertEquals('', elem.style.color);  // sanity check

    safe.setStyle(elem, style);
    assertEquals('red', elem.style.color);
  },

  testDocumentWrite() {
    const mockDoc = /** @type {!Document} */ ({
      'html': null,
      /** @suppress {globalThis} */
      'write': function(html) {
        this['html'] = html.toString();
      },
    });
    const html = '<script>somethingTrusted();<' +
        '/script>';
    const safeHtml = testing.newSafeHtmlForTest(html);
    safe.documentWrite(mockDoc, safeHtml);
    assertEquals(html, mockDoc.html);
  },

  testsetLinkHrefAndRel_trustedResourceUrl() {
    const mockLink =
        /** @type {!HTMLLinkElement} */ (
            {'href': null, 'rel': null, setAttribute: () => {}});

    const url =
        TrustedResourceUrl.fromConstant(Const.from('javascript:trusted();'));
    // Test case-insensitive too.
    safe.setLinkHrefAndRel(mockLink, url, 'foo, Stylesheet, bar');
    assertEquals('javascript:trusted();', mockLink.href);

    safe.setLinkHrefAndRel(mockLink, url, 'foo, bar');
    assertEquals('javascript:trusted();', mockLink.href);
  },

  testsetLinkHrefAndRel_safeUrl() {
    const mockLink =
        /** @type {!HTMLLinkElement} */ (
            {'href': null, 'rel': null, setAttribute: () => {}});

    const url = SafeUrl.fromConstant(Const.from('javascript:trusted();'));
    assertThrows(() => {
      safe.setLinkHrefAndRel(mockLink, url, 'foo, stylesheet, bar');
    });

    safe.setLinkHrefAndRel(mockLink, url, 'foo, bar');
    assertEquals('javascript:trusted();', mockLink.href);
  },

  testsetLinkHrefAndRel_string() {
    const mockLink =
        /** @type {!HTMLLinkElement} */ (
            {'href': null, 'rel': null, setAttribute: () => {}});

    assertThrows(() => {
      safe.setLinkHrefAndRel(
          mockLink, 'javascript:evil();', 'foo, stylesheet, bar');
    });
    withAssertionFailure(() => {
      safe.setLinkHrefAndRel(mockLink, 'javascript:evil();', 'foo, bar');
    });
    assertEquals('about:invalid#zClosurez', mockLink.href);
  },

  testsetLinkHrefAndRel_assertsType() {
    if (!userAgent.IE || userAgent.isVersionOrHigher(10)) {
      const otherElement = document.createElement('A');
      const ex = assertThrows(() => {
        safe.setLinkHrefAndRel(
            /** @type {!HTMLLinkElement} */ (otherElement),
            'http://example.com/', 'author');
      });
      assert(
          googString.contains(ex.message, 'Argument is not a HTMLLinkElement'));
    }
  },

  testSetLocationHref() {
    let mockLoc = /** @type {!Location} */ ({'href': 'blarg'});
    withAssertionFailure(() => {
      safe.setLocationHref(mockLoc, 'javascript:evil();');
    });
    assertEquals('about:invalid#zClosurez', mockLoc.href);

    mockLoc = /** @type {!Location} */ ({'href': 'blarg'});
    const safeUrl = SafeUrl.fromConstant(Const.from('javascript:trusted();'));
    safe.setLocationHref(mockLoc, safeUrl);
    assertEquals('javascript:trusted();', mockLoc.href);

    // Asserts correct runtime type.
    if (!userAgent.IE || userAgent.isVersionOrHigher(10)) {
      const ex = assertThrows(() => {
        safe.setLocationHref(makeLinkElementTypedAsLocation(), safeUrl);
      });
      assert(googString.contains(ex.message, 'Argument is not a Location'));
    }
  },

  testReplaceLocationSafeString() {
    // TODO(bangert): the mocks don't work on IE 8
    if (!userAgent.IE || userAgent.isVersionOrHigher(10)) {
      /** @type {?} */
      const mockLoc = new googTesting.StrictMock(window.location);
      mockLoc.replace('http://example.com/');
      mockLoc.$replay();
      safe.replaceLocation(mockLoc, 'http://example.com/');
      mockLoc.$verify();
      mockLoc.$reset();
    }
  },

  testReplaceLocationEvilString() {
    // TODO(bangert): the mocks don't work on IE 8
    if (!userAgent.IE || userAgent.isVersionOrHigher(10)) {
      /** @type {?} */
      const mockLoc = new googTesting.StrictMock(window.location);
      mockLoc.replace('about:invalid#zClosurez');
      mockLoc.$replay();
      withAssertionFailure(() => {
        safe.replaceLocation(mockLoc, 'javascript:evil();');
      });
      mockLoc.$verify();
      mockLoc.$reset();
    }
  },

  testReplaceLocationSafeUrl() {
    // TODO(bangert): the mocks don't work on IE 8
    if (!userAgent.IE || userAgent.isVersionOrHigher(10)) {
      const safeUrl = SafeUrl.fromConstant(Const.from('javascript:trusted();'));
      /** @type {?} */
      const mockLoc = new googTesting.StrictMock(window.location);
      mockLoc.replace('javascript:trusted();');
      mockLoc.$replay();
      safe.replaceLocation(mockLoc, safeUrl);
      mockLoc.$verify();
      mockLoc.$reset();
    }
  },

  testAssignLocationSafeString() {
    let location;
    const fakeLoc = /** @type {!Location} */ ({
      assign: function(value) {
        location = value;
      },
    });
    safe.assignLocation(fakeLoc, 'http://example.com/');
    assertEquals(location, 'http://example.com/');
  },

  testAssignLocationEvilString() {
    let location;
    const fakeLoc = /** @type {!Location} */ ({
      assign: function(value) {
        location = value;
      },
    });
    withAssertionFailure(() => {
      safe.assignLocation(fakeLoc, 'javascript:evil();');
    });
    assertEquals(location, 'about:invalid#zClosurez');
  },

  testAssignLocationSafeUrl() {
    let location;
    const fakeLoc = /** @type {!Location} */ ({
      assign: function(value) {
        location = value;
      },
    });
    const safeUrl = SafeUrl.fromConstant(Const.from('javascript:trusted();'));
    safe.assignLocation(fakeLoc, safeUrl);
    assertEquals(location, 'javascript:trusted();');
  },

  testSetAnchorHref() {
    let anchor =
        /** @type {!HTMLAnchorElement} */ (document.createElement('A'));
    withAssertionFailure(() => {
      safe.setAnchorHref(anchor, 'javascript:evil();');
    });
    assertEquals('about:invalid#zClosurez', anchor.href);

    anchor = /** @type {!HTMLAnchorElement} */ (document.createElement('A'));
    let safeUrl = SafeUrl.fromConstant(Const.from('javascript:trusted();'));
    safe.setAnchorHref(anchor, safeUrl);
    assertEquals('javascript:trusted();', anchor.href);

    // Works with mocks too.
    let mockAnchor = /** @type {!HTMLAnchorElement} */ ({'href': 'blarg'});
    withAssertionFailure(() => {
      safe.setAnchorHref(mockAnchor, 'javascript:evil();');
    });
    assertEquals('about:invalid#zClosurez', mockAnchor.href);

    mockAnchor = /** @type {!HTMLAnchorElement} */ ({'href': 'blarg'});
    safeUrl = SafeUrl.fromConstant(Const.from('javascript:trusted();'));
    safe.setAnchorHref(mockAnchor, safeUrl);
    assertEquals('javascript:trusted();', mockAnchor.href);

    // Asserts correct runtime type.
    if (!userAgent.IE || userAgent.isVersionOrHigher(10)) {
      const otherElement = document.createElement('LINK');
      const ex = assertThrows(() => {
        safe.setAnchorHref(
            /** @type {!HTMLAnchorElement} */ (otherElement), safeUrl);
      });
      assert(googString.contains(
          ex.message, 'Argument is not a HTMLAnchorElement'));
    }
  },

  testSetInputFormActionHarmlessString() {
    const element = dom.createElement(TagName.INPUT);
    safe.setInputFormAction(element, 'http://foo.com/');
    assertEquals('http://foo.com/', element.formAction);
  },

  testSetInputFormActionEvilString() {
    const element = dom.createElement(TagName.INPUT);
    withAssertionFailure(() => {
      safe.setInputFormAction(element, 'javascript:evil();');
    });
    assertEquals('about:invalid#zClosurez', element.formAction);
  },

  testSetInputFormActionSafeUrl() {
    const element = dom.createElement(TagName.INPUT);
    safe.setInputFormAction(
        element, SafeUrl.fromConstant(Const.from('javascript:trusted();')));
    assertEquals('javascript:trusted();', element.formAction);
  },

  testSetInputFormActionAssertsType() {
    /** @type {?} */
    const element = dom.createElement(TagName.FORM);
    withAssertionFailure(() => {
      safe.setInputFormAction(element, 'foo');
    });
    assertEquals('foo', element.formAction);
  },

  testSetButtonFormActionHarmlessString() {
    const element = dom.createElement(TagName.BUTTON);
    safe.setButtonFormAction(element, 'http://foo.com/');
    assertEquals('http://foo.com/', element.formAction);
  },

  testSetButtonFormActionEvilString() {
    const element = dom.createElement(TagName.BUTTON);
    withAssertionFailure(() => {
      safe.setButtonFormAction(element, 'javascript:evil();');
    });
    assertEquals('about:invalid#zClosurez', element.formAction);
  },

  testSetButtonFormActionSafeUrl() {
    const element = dom.createElement(TagName.BUTTON);
    safe.setButtonFormAction(
        element, SafeUrl.fromConstant(Const.from('javascript:trusted();')));
    assertEquals('javascript:trusted();', element.formAction);
  },

  testSetFormElementActionAssertsType() {
    /** @type {?} */
    const element = dom.createElement(TagName.INPUT);
    withAssertionFailure(() => {
      safe.setFormElementAction(element, 'javascript:evil();');
    });
    assertEquals('about:invalid#zClosurez', element.action);
  },

  testSetFormElementActionHarmlessString() {
    const element = dom.createElement(TagName.FORM);
    safe.setFormElementAction(element, 'http://foo.com');
    assertEquals('http://foo.com/', element.action);  // url is normalized
  },

  testSetFormElementActionEvilString() {
    const element = dom.createElement(TagName.FORM);
    withAssertionFailure(() => {
      safe.setFormElementAction(element, 'javascript:evil();');
    });
    assertEquals('about:invalid#zClosurez', element.action);
  },

  testSetFormElementActionSafeUrl() {
    const element = dom.createElement(TagName.FORM);
    safe.setFormElementAction(
        element, SafeUrl.fromConstant(Const.from('javascript:trusted();')));
    assertEquals('javascript:trusted();', element.action);
  },

  testSetImageSrc_withSafeUrlObject() {
    let mockImageElement = /** @type {!HTMLImageElement} */ ({'src': 'blarg'});
    withAssertionFailure(() => {
      safe.setImageSrc(mockImageElement, 'javascript:evil();');
    });
    assertEquals('about:invalid#zClosurez', mockImageElement.src);

    mockImageElement = /** @type {!HTMLImageElement} */ ({'src': 'blarg'});
    const safeUrl = SafeUrl.fromConstant(Const.from('javascript:trusted();'));
    safe.setImageSrc(mockImageElement, safeUrl);
    assertEquals('javascript:trusted();', mockImageElement.src);

    // Asserts correct runtime type.
    if (!userAgent.IE || userAgent.isVersionOrHigher(10)) {
      const otherElement = document.createElement('SCRIPT');
      const ex = assertThrows(() => {
        safe.setImageSrc(
            /** @type {!HTMLImageElement} */ (otherElement), safeUrl);
      });
      assert(googString.contains(
          ex.message, 'Argument is not a HTMLImageElement'));
    }
  },

  testSetImageSrc_withHttpsUrl() {
    const mockImageElement =
        /** @type {!HTMLImageElement} */ ({'src': 'blarg'});

    const safeUrl = 'https://trusted_url';
    safe.setImageSrc(mockImageElement, safeUrl);
    assertEquals(safeUrl, mockImageElement.src);
  },

  testSetImageSrc_withDataUrl() {
    const mockImageElement =
        /** @type {!HTMLImageElement} */ ({'src': 'blarg'});
    const safeUrl = '';
    safe.setImageSrc(mockImageElement, safeUrl);
    assertEquals(safeUrl, mockImageElement.src);
    assertThrows(() => {
      safe.setImageSrc(mockImageElement, 'data:text/plain;base64,a');
    });
    assertThrows(() => {
      safe.setImageSrc(mockImageElement, 'data:image/gif;bad');
    });
  },

  testSetAudioSrc() {
    let mockAudioElement = /** @type {!HTMLAudioElement} */ ({'src': 'blarg'});
    let safeUrl = 'https://trusted_url';
    safe.setAudioSrc(mockAudioElement, safeUrl);
    assertEquals(safeUrl, mockAudioElement.src);

    mockAudioElement = /** @type {!HTMLAudioElement} */ ({'src': 'blarg'});
    withAssertionFailure(() => {
      safe.setAudioSrc(mockAudioElement, 'javascript:evil();');
    });
    assertEquals('about:invalid#zClosurez', mockAudioElement.src);

    mockAudioElement = /** @type {!HTMLAudioElement} */ ({'src': 'blarg'});
    safeUrl = SafeUrl.fromConstant(Const.from('javascript:trusted();'));
    safe.setAudioSrc(mockAudioElement, safeUrl);
    assertEquals('javascript:trusted();', mockAudioElement.src);

    // Asserts correct runtime type.
    if (!userAgent.IE || userAgent.isVersionOrHigher(10)) {
      const otherElement = document.createElement('SCRIPT');
      const ex = assertThrows(() => {
        safe.setAudioSrc(
            /** @type {!HTMLAudioElement} */ (otherElement), safeUrl);
      });
      assert(googString.contains(
          ex.message, 'Argument is not a HTMLAudioElement'));
    }
  },

  testSetAudioSrc_withDataUrl() {
    const mockAudioElement =
        /** @type {!HTMLAudioElement} */ ({'src': 'blarg'});
    const safeUrl = 'data:audio/mp3;base64,a';
    safe.setAudioSrc(mockAudioElement, safeUrl);
    assertEquals(safeUrl, mockAudioElement.src);
    assertThrows(() => {
      safe.setAudioSrc(mockAudioElement, '');
    });
  },

  testSetVideoSrc() {
    let mockVideoElement = /** @type {!HTMLVideoElement} */ ({'src': 'blarg'});
    let safeUrl = 'https://trusted_url';
    safe.setVideoSrc(mockVideoElement, safeUrl);
    assertEquals(safeUrl, mockVideoElement.src);

    mockVideoElement = /** @type {!HTMLVideoElement} */ ({'src': 'blarg'});
    withAssertionFailure(() => {
      safe.setVideoSrc(mockVideoElement, 'javascript:evil();');
    });
    assertEquals('about:invalid#zClosurez', mockVideoElement.src);

    mockVideoElement = /** @type {!HTMLVideoElement} */ ({'src': 'blarg'});
    safeUrl = SafeUrl.fromConstant(Const.from('javascript:trusted();'));
    safe.setVideoSrc(mockVideoElement, safeUrl);
    assertEquals('javascript:trusted();', mockVideoElement.src);

    // Asserts correct runtime type.
    if (!userAgent.IE || userAgent.isVersionOrHigher(10)) {
      const otherElement = document.createElement('SCRIPT');
      const ex = assertThrows(() => {
        safe.setVideoSrc(
            /** @type {!HTMLVideoElement} */ (otherElement), safeUrl);
      });
      assert(googString.contains(
          ex.message, 'Argument is not a HTMLVideoElement'));
    }
  },

  testSetVideoSrc_withDataUrl() {
    const mockVideoElement =
        /** @type {!HTMLVideoElement} */ ({'src': 'blarg'});
    const safeUrl = 'data:video/mp4;base64,a';
    safe.setVideoSrc(mockVideoElement, safeUrl);
    assertEquals(safeUrl, mockVideoElement.src);
    assertThrows(() => {
      safe.setVideoSrc(mockVideoElement, '');
    });
  },

  testSetEmbedSrc() {
    const url =
        TrustedResourceUrl.fromConstant(Const.from('javascript:trusted();'));
    const mockElement = /** @type {!HTMLEmbedElement} */ ({'src': 'blarg'});
    safe.setEmbedSrc(mockElement, url);
    assertEquals('javascript:trusted();', mockElement.src.toString());

    // Asserts correct runtime type.
    if (!userAgent.IE || userAgent.isVersionOrHigher(10)) {
      const otherElement = document.createElement('IMAGE');
      const ex = assertThrows(() => {
        safe.setEmbedSrc(
            /** @type {!HTMLEmbedElement} */ (otherElement), url);
      });
      assert(googString.contains(
          ex.message, 'Argument is not a HTMLEmbedElement'));
    }
  },

  testSetFrameSrc() {
    const url =
        TrustedResourceUrl.fromConstant(Const.from('javascript:trusted();'));
    const mockElement = /** @type {!HTMLFrameElement} */ ({'src': 'blarg'});
    safe.setFrameSrc(mockElement, url);
    assertEquals('javascript:trusted();', mockElement.src);

    // Asserts correct runtime type.
    if (!userAgent.IE || userAgent.isVersionOrHigher(10)) {
      const otherElement = document.createElement('IMAGE');
      const ex = assertThrows(() => {
        safe.setFrameSrc(
            /** @type {!HTMLFrameElement} */ (otherElement), url);
      });
      assert(googString.contains(
          ex.message, 'Argument is not a HTMLFrameElement'));
    }
  },

  testSetIframeSrc() {
    const url =
        TrustedResourceUrl.fromConstant(Const.from('javascript:trusted();'));
    const mockElement = /** @type {!HTMLIFrameElement} */ ({'src': 'blarg'});
    safe.setIframeSrc(mockElement, url);
    assertEquals('javascript:trusted();', mockElement.src);

    // Asserts correct runtime type.
    if (!userAgent.IE || userAgent.isVersionOrHigher(10)) {
      const otherElement = document.createElement('IMAGE');
      const ex = assertThrows(() => {
        safe.setIframeSrc(
            /** @type {!HTMLIFrameElement} */ (otherElement), url);
      });
      assert(googString.contains(
          ex.message, 'Argument is not a HTMLIFrameElement'));
    }
  },

  testSetIframeSrcdoc() {
    const html = SafeHtml.create('div', {}, 'foobar');
    const mockIframe = /** @type {!HTMLIFrameElement} */ ({'srcdoc': ''});
    safe.setIframeSrcdoc(mockIframe, html);
    assertEquals('<div>foobar</div>', mockIframe.srcdoc.toString());

    // Asserts correct runtime type.
    if (!userAgent.IE || userAgent.isVersionOrHigher(10)) {
      const otherElement = document.createElement('IMAGE');
      const ex = assertThrows(() => {
        safe.setIframeSrcdoc(
            /** @type {!HTMLIFrameElement} */ (otherElement), html);
      });
      assert(googString.contains(
          ex.message, 'Argument is not a HTMLIFrameElement'));
    }
  },

  testSetObjectData() {
    const url =
        TrustedResourceUrl.fromConstant(Const.from('javascript:trusted();'));
    const mockElement = /** @type {!HTMLObjectElement} */ ({'data': 'blarg'});
    safe.setObjectData(mockElement, url);
    assertEquals('javascript:trusted();', mockElement.data.toString());

    // Asserts correct runtime type.
    if (!userAgent.IE || userAgent.isVersionOrHigher(10)) {
      const otherElement = document.createElement('IMAGE');
      const ex = assertThrows(() => {
        safe.setObjectData(
            /** @type {!HTMLObjectElement} */ (otherElement), url);
      });
      assert(googString.contains(
          ex.message, 'Argument is not a HTMLObjectElement'));
    }
  },

  testSetScriptSrc() {
    const url =
        TrustedResourceUrl.fromConstant(Const.from('javascript:trusted();'));
    const mockElement = /** @type {!HTMLScriptElement} */ ({
      'src': 'blarg',
      /** @suppress {globalThis} */
      'setAttribute': function(attr, value) {
        this[attr] = value;
      },
    });
    let nonce = safe.getScriptNonce();
    if (!nonce) {
      // Place a nonced script in the page.
      nonce = 'ThisIsANonceThisIsANonceThisIsANonce';
    }

    const noncedScript = dom.createElement(TagName.SCRIPT);
    noncedScript.setAttribute('nonce', nonce);
    document.body.appendChild(noncedScript);
    safe.setScriptSrc(mockElement, url);

    try {
      assertEquals('javascript:trusted();', mockElement.src.toString());
      assertEquals(nonce, mockElement.nonce);
    } finally {
      dom.removeNode(noncedScript);
    }
    // Asserts correct runtime type.
    if (!userAgent.IE || userAgent.isVersionOrHigher(10)) {
      const otherElement = document.createElement('IMAGE');
      const ex = assertThrows(() => {
        safe.setScriptSrc(
            /** @type {!HTMLScriptElement} */ (otherElement), url);
      });
      assert(googString.contains(
          ex.message, 'Argument is not a HTMLScriptElement'));
    }
  },

  testSetScriptSrc_withIframe() {
    const url =
        TrustedResourceUrl.fromConstant(Const.from('javascript:trusted();'));
    // create the iframe and set up a script inside the iframe.
    let nonce = safe.getScriptNonce();
    if (!nonce) {
      nonce = 'ThisIsANonceThisIsANonceThisIsANonce';
    }

    const iframe = dom.createElement(TagName.IFRAME);
    document.body.appendChild(iframe);
    const iframeWindow = iframe.contentWindow;
    const iframeDocument = iframeWindow.document;
    iframeDocument.write('<HTML><BODY></BODY></HTML>');
    iframeDocument.close();
    const iframeScript = iframeDocument.createElement('SCRIPT');
    iframeScript.setAttribute('nonce', nonce);
    iframeDocument.body.appendChild(iframeScript);
    const mockElement = /** @type {!HTMLScriptElement} */ ({
      'src': 'blarg',
      /** @suppress {globalThis} */
      'setAttribute': function(attr, value) {
        this[attr] = value;
      },
      ownerDocument: {defaultView: iframeWindow}
    });
    safe.setScriptSrc(mockElement, url);
    try {
      assertEquals('javascript:trusted();', mockElement.src.toString());
      assertEquals(nonce, mockElement.nonce);
    } finally {
      dom.removeNode(iframe);
    }
  },

  testSetScriptContent() {
    const mockScriptElement = /** @type {!HTMLScriptElement} */ ({
      /** @suppress {globalThis} */
      'setAttribute': function(attr, value) {
        this[attr] = value;
      },
    });
    // Place a nonced script in the page.
    let nonce = safe.getScriptNonce();
    if (!nonce) {
      nonce = 'ThisIsANonceThisIsANonceThisIsANonce';
    }

    const noncedScript = dom.createElement(TagName.SCRIPT);
    noncedScript.setAttribute('nonce', nonce);
    document.body.appendChild(noncedScript);
    const content = SafeScript.fromConstant(Const.from('alert(1);'));
    safe.setScriptContent(mockScriptElement, content);

    try {
      assertEquals(
          SafeScript.unwrap(content), mockScriptElement.textContent.toString());
      assertEquals(nonce, mockScriptElement.nonce);
    } finally {
      dom.removeNode(noncedScript);
    }
  },

  testSetScriptContentWithSpecialCharacters() {
    const scriptElement = dom.createElement(TagName.SCRIPT);
    document.body.appendChild(scriptElement);
    const TEST_PROPERTY = 'scriptContentTestProperty';
    const content = SafeScript.fromConstant(Const.from(`
      // Comment to ensure newlines are preserved.
      window.${TEST_PROPERTY} = 'tricky<{}>value';
    `));
    safe.setScriptContent(scriptElement, content);

    try {
      assertEquals(SafeScript.unwrap(content), scriptElement.text);
      assertEquals('tricky<{}>value', window[TEST_PROPERTY]);

      // Ensure no <br> tags were inserted into the script tag.
      assertEquals(1, scriptElement.childNodes.length);
    } finally {
      dom.removeNode(scriptElement);
      delete window[TEST_PROPERTY];
    }
  },

  testOpenInWindow() {
    mockWindowOpen =
        /** @type {?} */ (googTesting.createMethodMock(window, 'open'));
    const fakeWindow = {};

    mockWindowOpen('about:invalid#zClosurez', 'name', 'specs')
        .$returns(fakeWindow);
    mockWindowOpen.$replay();
    let retVal = withAssertionFailure(
        () => safe.openInWindow(
            'javascript:evil();', window, Const.from('name'), 'specs'));
    mockWindowOpen.$verify();
    assertEquals(
        'openInWindow should return the created window', fakeWindow, retVal);

    mockWindowOpen.$reset();
    retVal = null;

    const safeUrl = SafeUrl.fromConstant(Const.from('javascript:trusted();'));
    mockWindowOpen('javascript:trusted();', 'name', 'specs')
        .$returns(fakeWindow);
    mockWindowOpen.$replay();
    retVal = safe.openInWindow(safeUrl, window, Const.from('name'), 'specs');
    mockWindowOpen.$verify();
    assertEquals(
        'openInWindow should return the created window', fakeWindow, retVal);
  },

  /**
     @suppress {strictMissingProperties} suppression added to enable type
     checking
   */
  testParseFromStringHtml() {
    if (userAgent.IE && !userAgent.isVersionOrHigher('10')) {
      return;
    }
    const html = SafeHtml.create('A', {'class': 'b'}, 'c');
    const node =
        safe.parseFromStringHtml(new DOMParser(), html).body.firstChild;
    assertEquals('A', node.tagName);
    assertEquals('b', node.className);
    assertEquals('c', node.textContent);
  },

  testParseFromString() {
    if (userAgent.IE && !userAgent.isVersionOrHigher('10')) {
      return;
    }
    const html = SafeHtml.create('a', {'class': 'b'}, 'c');
    const node = safe.parseFromString(new DOMParser(), html, 'application/xml')
                     .firstChild;
    assertEquals('a', node.tagName);
    assertEquals('b', node.getAttribute('class'));
    assertEquals('c', node.textContent);
  },

  testCreateImageFromBlob() {
    const blob = new Blob(['data'], {type: 'image/svg+xml'});
    const fakeObjectUrl = 'blob:http://fakeurl.com';
    const mockCreateObject = /** @type {?} */ (
        googTesting.createMethodMock(window.URL, 'createObjectURL'));
    const mockRevokeObject = /** @type {?} */ (
        googTesting.createMethodMock(window.URL, 'revokeObjectURL'));
    mockCreateObject(blob).$returns(fakeObjectUrl);
    mockRevokeObject(fakeObjectUrl);
    mockCreateObject.$replay();
    mockRevokeObject.$replay();

    const image = safe.createImageFromBlob(blob);

    mockCreateObject.$verify();
    assertEquals('function', typeof image.onload);
    image.onload(null);  // Trigger image load.
    mockRevokeObject.$verify();
    assertEquals(fakeObjectUrl, image.src);
    assertTrue(image instanceof HTMLImageElement);
  },

  testCreateImageFromBlobBadMimeType() {
    // Skip unsupported test if IE9 or lower.
    if (userAgent.IE && !userAgent.isVersionOrHigher('10')) {
      return;
    }
    const blob = new Blob(['data'], {type: 'badmimetype'});
    assertThrows(() => {
      safe.createImageFromBlob(blob);
    });
  },

  /**
     @suppress {strictMissingProperties} suppression added to enable type
     checking
   */
  testCreateContextualFragment() {
    if (userAgent.IE && !userAgent.isVersionOrHigher('11')) {
      return;
    }
    const html = SafeHtml.create('A', {'class': 'b'}, 'c');
    const node = safe.createContextualFragment(
                         /** @type {!Range} */ (document.createRange()), html)
                     .childNodes[0];
    assertEquals('A', node.tagName);
    assertEquals('b', node.className);
    assertEquals('c', node.textContent);
  },

  testGetScriptNonce() {
    assertEquals('CSP+Nonce+For+Tests+Only', safe.getScriptNonce());
  },

  testGetStyleNonce() {
    assertEquals('NONCE', safe.getStyleNonce());
    const style = document.querySelector('style[nonce]');
    style.parentNode.removeChild(style);
    assertEquals('NONCE2', safe.getStyleNonce(window));
  }
});