|
@@ -102,13 +102,11 @@ const STAGES = [{
|
|
|
{ action: "insertShadowRootContents" },
|
|
{ action: "insertShadowRootContents" },
|
|
|
{ action: "setInputValues" },
|
|
{ action: "setInputValues" },
|
|
|
{ option: "moveStylesInHead", action: "moveStylesInHead" },
|
|
{ option: "moveStylesInHead", action: "moveStylesInHead" },
|
|
|
- { option: "removeScripts", action: "removeScripts" },
|
|
|
|
|
|
|
+ { option: "blockScripts", action: "removeEmbedScripts" },
|
|
|
{ option: "selected", action: "removeUnselectedElements" },
|
|
{ option: "selected", action: "removeUnselectedElements" },
|
|
|
- { option: "removeVideoSrc", action: "insertVideoPosters" },
|
|
|
|
|
- { option: "removeVideoSrc", action: "insertVideoLinks" },
|
|
|
|
|
|
|
+ { option: "blockVideos", action: "insertVideoPosters" },
|
|
|
|
|
+ { option: "blockVideos", action: "insertVideoLinks" },
|
|
|
{ option: "removeFrames", action: "removeFrames" },
|
|
{ option: "removeFrames", action: "removeFrames" },
|
|
|
- { option: "removeVideoSrc", action: "removeVideoSources" },
|
|
|
|
|
- { option: "removeAudioSrc", action: "removeAudioSources" },
|
|
|
|
|
{ action: "removeDiscardedResources" },
|
|
{ action: "removeDiscardedResources" },
|
|
|
{ option: "removeHiddenElements", action: "removeHiddenElements" },
|
|
{ option: "removeHiddenElements", action: "removeHiddenElements" },
|
|
|
{ action: "resolveHrefs" },
|
|
{ action: "resolveHrefs" },
|
|
@@ -129,7 +127,7 @@ const STAGES = [{
|
|
|
{ action: "processStylesheets" },
|
|
{ action: "processStylesheets" },
|
|
|
{ action: "processStyleAttributes" },
|
|
{ action: "processStyleAttributes" },
|
|
|
{ action: "processPageResources" },
|
|
{ action: "processPageResources" },
|
|
|
- { option: "!removeScripts", action: "processScripts" }
|
|
|
|
|
|
|
+ { action: "processScripts" }
|
|
|
]
|
|
]
|
|
|
}, {
|
|
}, {
|
|
|
sequential: [
|
|
sequential: [
|
|
@@ -399,7 +397,6 @@ class BatchRequest {
|
|
|
// ---------
|
|
// ---------
|
|
|
const PREFIXES_FORBIDDEN_DATA_URI = ["data:text/"];
|
|
const PREFIXES_FORBIDDEN_DATA_URI = ["data:text/"];
|
|
|
const PREFIX_DATA_URI_IMAGE_SVG = "data:image/svg+xml";
|
|
const PREFIX_DATA_URI_IMAGE_SVG = "data:image/svg+xml";
|
|
|
-const EMPTY_IMAGE = "";
|
|
|
|
|
const SCRIPT_TAG_FOUND = /<script/gi;
|
|
const SCRIPT_TAG_FOUND = /<script/gi;
|
|
|
const NOSCRIPT_TAG_FOUND = /<noscript/gi;
|
|
const NOSCRIPT_TAG_FOUND = /<noscript/gi;
|
|
|
const CANVAS_TAG_FOUND = /<canvas/gi;
|
|
const CANVAS_TAG_FOUND = /<canvas/gi;
|
|
@@ -525,6 +522,9 @@ class Processor {
|
|
|
this.doc.head.appendChild(metaElement);
|
|
this.doc.head.appendChild(metaElement);
|
|
|
}
|
|
}
|
|
|
}
|
|
}
|
|
|
|
|
+ const styleElement = this.doc.createElement("style");
|
|
|
|
|
+ styleElement.textContent = "img[src=\"data:,\"],source[src=\"data:,\"]{display:none!important}";
|
|
|
|
|
+ this.doc.head.appendChild(styleElement);
|
|
|
let size;
|
|
let size;
|
|
|
if (this.options.displayStats) {
|
|
if (this.options.displayStats) {
|
|
|
size = util.getContentSize(this.doc.documentElement.outerHTML);
|
|
size = util.getContentSize(this.doc.documentElement.outerHTML);
|
|
@@ -583,14 +583,14 @@ class Processor {
|
|
|
(imageData.size && !imageData.size.pxWidth && !imageData.size.pxHeight) ||
|
|
(imageData.size && !imageData.size.pxWidth && !imageData.size.pxHeight) ||
|
|
|
(imgElement.getAttribute(util.HIDDEN_CONTENT_ATTRIBUTE_NAME) == "")
|
|
(imgElement.getAttribute(util.HIDDEN_CONTENT_ATTRIBUTE_NAME) == "")
|
|
|
)) {
|
|
)) {
|
|
|
- imgElement.setAttribute("src", EMPTY_IMAGE);
|
|
|
|
|
|
|
+ imgElement.setAttribute("src", util.EMPTY_RESOURCE);
|
|
|
} else {
|
|
} else {
|
|
|
if (imageData.currentSrc) {
|
|
if (imageData.currentSrc) {
|
|
|
imgElement.dataset.singleFileOriginURL = imgElement.getAttribute("src");
|
|
imgElement.dataset.singleFileOriginURL = imgElement.getAttribute("src");
|
|
|
imgElement.setAttribute("src", imageData.currentSrc);
|
|
imgElement.setAttribute("src", imageData.currentSrc);
|
|
|
}
|
|
}
|
|
|
if (this.options.loadDeferredImages) {
|
|
if (this.options.loadDeferredImages) {
|
|
|
- if ((!imgElement.getAttribute("src") || imgElement.getAttribute("src") == EMPTY_IMAGE) && imgElement.getAttribute("data-src")) {
|
|
|
|
|
|
|
+ if ((!imgElement.getAttribute("src") || imgElement.getAttribute("src") == util.EMPTY_RESOURCE) && imgElement.getAttribute("data-src")) {
|
|
|
imageData.src = imgElement.dataset.src;
|
|
imageData.src = imgElement.dataset.src;
|
|
|
imgElement.setAttribute("src", imgElement.dataset.src);
|
|
imgElement.setAttribute("src", imgElement.dataset.src);
|
|
|
imgElement.removeAttribute("data-src");
|
|
imgElement.removeAttribute("data-src");
|
|
@@ -738,15 +738,16 @@ class Processor {
|
|
|
this.doc.querySelectorAll("iframe, frame, object[type=\"text/html\"][data]").forEach(element => element.remove());
|
|
this.doc.querySelectorAll("iframe, frame, object[type=\"text/html\"][data]").forEach(element => element.remove());
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
- removeScripts() {
|
|
|
|
|
|
|
+ removeEmbedScripts() {
|
|
|
|
|
+ const JAVASCRIPT_URI_PREFIX = "javascript:";
|
|
|
this.onEventAttributeNames.forEach(attributeName => this.doc.querySelectorAll("[" + attributeName + "]").forEach(element => element.removeAttribute(attributeName)));
|
|
this.onEventAttributeNames.forEach(attributeName => this.doc.querySelectorAll("[" + attributeName + "]").forEach(element => element.removeAttribute(attributeName)));
|
|
|
this.doc.querySelectorAll("[href]").forEach(element => {
|
|
this.doc.querySelectorAll("[href]").forEach(element => {
|
|
|
- if (element.href && element.href.match && element.href.match(/^\s*javascript:/)) {
|
|
|
|
|
- element.setAttribute("href", "");
|
|
|
|
|
|
|
+ if (element.href && element.href.match && element.href.trim().startsWith(JAVASCRIPT_URI_PREFIX)) {
|
|
|
|
|
+ element.removeAttribute("href");
|
|
|
}
|
|
}
|
|
|
});
|
|
});
|
|
|
this.doc.querySelectorAll("[src]").forEach(element => {
|
|
this.doc.querySelectorAll("[src]").forEach(element => {
|
|
|
- if (element.src && element.src.match(/^\s*javascript:/)) {
|
|
|
|
|
|
|
+ if (element.src && element.src.trim().startsWith(JAVASCRIPT_URI_PREFIX)) {
|
|
|
element.removeAttribute("src");
|
|
element.removeAttribute("src");
|
|
|
}
|
|
}
|
|
|
});
|
|
});
|
|
@@ -756,32 +757,6 @@ class Processor {
|
|
|
scriptElements.forEach(element => element.remove());
|
|
scriptElements.forEach(element => element.remove());
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
- removeVideoSources() {
|
|
|
|
|
- const videoSourceElements = this.doc.querySelectorAll("video[src], video > source");
|
|
|
|
|
- this.stats.set("discarded", "video sources", videoSourceElements.length);
|
|
|
|
|
- this.stats.set("processed", "video sources", videoSourceElements.length);
|
|
|
|
|
- videoSourceElements.forEach(element => {
|
|
|
|
|
- if (element.tagName == "SOURCE") {
|
|
|
|
|
- element.remove();
|
|
|
|
|
- } else {
|
|
|
|
|
- videoSourceElements.forEach(element => element.removeAttribute("src"));
|
|
|
|
|
- }
|
|
|
|
|
- });
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- removeAudioSources() {
|
|
|
|
|
- const audioSourceElements = this.doc.querySelectorAll("audio[src], audio > source[src]");
|
|
|
|
|
- this.stats.set("discarded", "audio sources", audioSourceElements.length);
|
|
|
|
|
- this.stats.set("processed", "audio sources", audioSourceElements.length);
|
|
|
|
|
- audioSourceElements.forEach(element => {
|
|
|
|
|
- if (element.tagName == "SOURCE") {
|
|
|
|
|
- element.remove();
|
|
|
|
|
- } else {
|
|
|
|
|
- audioSourceElements.forEach(element => element.removeAttribute("src"));
|
|
|
|
|
- }
|
|
|
|
|
- });
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
removeDiscardedResources() {
|
|
removeDiscardedResources() {
|
|
|
this.doc.querySelectorAll("." + util.SINGLE_FILE_UI_ELEMENT_CLASS).forEach(element => element.remove());
|
|
this.doc.querySelectorAll("." + util.SINGLE_FILE_UI_ELEMENT_CLASS).forEach(element => element.remove());
|
|
|
const noscriptPlaceholders = new Map();
|
|
const noscriptPlaceholders = new Map();
|
|
@@ -803,13 +778,7 @@ class Processor {
|
|
|
objectElements.forEach(element => element.remove());
|
|
objectElements.forEach(element => element.remove());
|
|
|
const replacedAttributeValue = this.doc.querySelectorAll("link[rel~=preconnect], link[rel~=prerender], link[rel~=dns-prefetch], link[rel~=preload], link[rel~=manifest], link[rel~=prefetch]");
|
|
const replacedAttributeValue = this.doc.querySelectorAll("link[rel~=preconnect], link[rel~=prerender], link[rel~=dns-prefetch], link[rel~=preload], link[rel~=manifest], link[rel~=prefetch]");
|
|
|
replacedAttributeValue.forEach(element => {
|
|
replacedAttributeValue.forEach(element => {
|
|
|
- let regExp;
|
|
|
|
|
- if (this.options.removeScripts) {
|
|
|
|
|
- regExp = /(preconnect|prerender|dns-prefetch|preload|prefetch|manifest)/g;
|
|
|
|
|
- } else {
|
|
|
|
|
- regExp = /(preconnect|prerender|dns-prefetch|prefetch|manifest)/g;
|
|
|
|
|
- }
|
|
|
|
|
- const relValue = element.getAttribute("rel").replace(regExp, "").trim();
|
|
|
|
|
|
|
+ const relValue = element.getAttribute("rel").replace(/(preconnect|prerender|dns-prefetch|preload|prefetch|manifest)/g, "").trim();
|
|
|
if (relValue.length) {
|
|
if (relValue.length) {
|
|
|
element.setAttribute("rel", relValue);
|
|
element.setAttribute("rel", relValue);
|
|
|
} else {
|
|
} else {
|
|
@@ -1004,21 +973,7 @@ class Processor {
|
|
|
|
|
|
|
|
async resolveStylesheetURLs() {
|
|
async resolveStylesheetURLs() {
|
|
|
await Promise.all(Array.from(this.doc.querySelectorAll("style, link[rel*=stylesheet]")).map(async element => {
|
|
await Promise.all(Array.from(this.doc.querySelectorAll("style, link[rel*=stylesheet]")).map(async element => {
|
|
|
- const options = {
|
|
|
|
|
- maxResourceSize: this.options.maxResourceSize,
|
|
|
|
|
- maxResourceSizeEnabled: this.options.maxResourceSizeEnabled,
|
|
|
|
|
- url: this.options.url,
|
|
|
|
|
- charset: this.charset,
|
|
|
|
|
- compressCSS: this.options.compressCSS,
|
|
|
|
|
- updatedResources: this.options.updatedResources,
|
|
|
|
|
- rootDocument: this.options.rootDocument,
|
|
|
|
|
- frameId: this.options.windowId,
|
|
|
|
|
- resourceReferrer: this.options.resourceReferrer,
|
|
|
|
|
- blockMixedContent: this.options.blockMixedContent,
|
|
|
|
|
- saveOriginalURLs: this.options.saveOriginalURLs,
|
|
|
|
|
- acceptHeaders: this.options.acceptHeaders,
|
|
|
|
|
- networkTimeout: this.options.networkTimeout
|
|
|
|
|
- };
|
|
|
|
|
|
|
+ const options = Object.assign({}, this.options, { charset: this.charset });
|
|
|
let mediaText;
|
|
let mediaText;
|
|
|
if (element.media) {
|
|
if (element.media) {
|
|
|
mediaText = element.media.toLowerCase();
|
|
mediaText = element.media.toLowerCase();
|
|
@@ -1045,17 +1000,19 @@ class Processor {
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
async function processElement(element, stylesheetInfo, stylesheets, baseURI, options, workStyleElement) {
|
|
async function processElement(element, stylesheetInfo, stylesheets, baseURI, options, workStyleElement) {
|
|
|
- stylesheets.set(element, stylesheetInfo);
|
|
|
|
|
- let stylesheetContent = await getStylesheetContent(element, baseURI, options, workStyleElement);
|
|
|
|
|
- if (!matchCharsetEquals(stylesheetContent, options.charset)) {
|
|
|
|
|
- options = Object.assign({}, options, { charset: getCharset(stylesheetContent) });
|
|
|
|
|
- stylesheetContent = await getStylesheetContent(element, baseURI, options, workStyleElement);
|
|
|
|
|
- }
|
|
|
|
|
let stylesheet;
|
|
let stylesheet;
|
|
|
- try {
|
|
|
|
|
- stylesheet = cssTree.parse(removeCssComments(stylesheetContent));
|
|
|
|
|
- } catch (error) {
|
|
|
|
|
- // ignored
|
|
|
|
|
|
|
+ stylesheets.set(element, stylesheetInfo);
|
|
|
|
|
+ if (!options.blockStylesheets) {
|
|
|
|
|
+ let stylesheetContent = await getStylesheetContent(element, baseURI, options, workStyleElement);
|
|
|
|
|
+ if (!matchCharsetEquals(stylesheetContent, options.charset)) {
|
|
|
|
|
+ options = Object.assign({}, options, { charset: getCharset(stylesheetContent) });
|
|
|
|
|
+ stylesheetContent = await getStylesheetContent(element, baseURI, options, workStyleElement);
|
|
|
|
|
+ }
|
|
|
|
|
+ try {
|
|
|
|
|
+ stylesheet = cssTree.parse(removeCssComments(stylesheetContent));
|
|
|
|
|
+ } catch (error) {
|
|
|
|
|
+ // ignored
|
|
|
|
|
+ }
|
|
|
}
|
|
}
|
|
|
if (stylesheet && stylesheet.children) {
|
|
if (stylesheet && stylesheet.children) {
|
|
|
if (options.compressCSS) {
|
|
if (options.compressCSS) {
|
|
@@ -1069,10 +1026,12 @@ class Processor {
|
|
|
|
|
|
|
|
async function getStylesheetContent(element, baseURI, options, workStyleElement) {
|
|
async function getStylesheetContent(element, baseURI, options, workStyleElement) {
|
|
|
let content;
|
|
let content;
|
|
|
- if (element.tagName == "LINK") {
|
|
|
|
|
- content = await ProcessorHelper.resolveLinkStylesheetURLs(element.href, baseURI, options, workStyleElement);
|
|
|
|
|
- } else {
|
|
|
|
|
- content = await ProcessorHelper.resolveImportURLs(element.textContent, baseURI, options, workStyleElement);
|
|
|
|
|
|
|
+ if (!options.blockStylesheets) {
|
|
|
|
|
+ if (element.tagName == "LINK") {
|
|
|
|
|
+ content = await ProcessorHelper.resolveLinkStylesheetURLs(element.href, baseURI, options, workStyleElement);
|
|
|
|
|
+ } else {
|
|
|
|
|
+ content = await ProcessorHelper.resolveImportURLs(element.textContent, baseURI, options, workStyleElement);
|
|
|
|
|
+ }
|
|
|
}
|
|
}
|
|
|
return content || "";
|
|
return content || "";
|
|
|
}
|
|
}
|
|
@@ -1133,9 +1092,9 @@ class Processor {
|
|
|
insertShadowRootContents() {
|
|
insertShadowRootContents() {
|
|
|
const doc = this.doc;
|
|
const doc = this.doc;
|
|
|
const options = this.options;
|
|
const options = this.options;
|
|
|
- if (this.options.shadowRoots && this.options.shadowRoots.length) {
|
|
|
|
|
|
|
+ if (options.shadowRoots && options.shadowRoots.length) {
|
|
|
processElement(this.doc);
|
|
processElement(this.doc);
|
|
|
- if (this.options.removeScripts) {
|
|
|
|
|
|
|
+ if (options.blockScripts) {
|
|
|
this.doc.querySelectorAll("script[" + SCRIPT_TEMPLATE_SHADOW_ROOT + "]").forEach(element => element.remove());
|
|
this.doc.querySelectorAll("script[" + SCRIPT_TEMPLATE_SHADOW_ROOT + "]").forEach(element => element.remove());
|
|
|
}
|
|
}
|
|
|
const scriptElement = doc.createElement("script");
|
|
const scriptElement = doc.createElement("script");
|
|
@@ -1272,19 +1231,18 @@ class Processor {
|
|
|
["image", "xlink:href"],
|
|
["image", "xlink:href"],
|
|
|
["image", "href"]
|
|
["image", "href"]
|
|
|
];
|
|
];
|
|
|
|
|
+ if (this.options.blockImages) {
|
|
|
|
|
+ this.doc.querySelectorAll("svg").forEach(element => element.remove());
|
|
|
|
|
+ }
|
|
|
let resourcePromises = processAttributeArgs.map(([selector, attributeName, processDuplicates, removeElementIfMissing]) =>
|
|
let resourcePromises = processAttributeArgs.map(([selector, attributeName, processDuplicates, removeElementIfMissing]) =>
|
|
|
ProcessorHelper.processAttribute(this.doc.querySelectorAll(selector), attributeName, this.baseURI, this.options, "image", this.cssVariables, this.styles, this.batchRequest, processDuplicates, removeElementIfMissing)
|
|
ProcessorHelper.processAttribute(this.doc.querySelectorAll(selector), attributeName, this.baseURI, this.options, "image", this.cssVariables, this.styles, this.batchRequest, processDuplicates, removeElementIfMissing)
|
|
|
);
|
|
);
|
|
|
resourcePromises = resourcePromises.concat([
|
|
resourcePromises = resourcePromises.concat([
|
|
|
ProcessorHelper.processXLinks(this.doc.querySelectorAll("use"), this.doc, this.baseURI, this.options, this.batchRequest),
|
|
ProcessorHelper.processXLinks(this.doc.querySelectorAll("use"), this.doc, this.baseURI, this.options, this.batchRequest),
|
|
|
- ProcessorHelper.processSrcset(this.doc.querySelectorAll("img[srcset], source[srcset]"), "srcset", this.baseURI, this.batchRequest)
|
|
|
|
|
|
|
+ ProcessorHelper.processSrcset(this.doc.querySelectorAll("img[srcset], source[srcset]"), this.baseURI, this.options, this.batchRequest)
|
|
|
]);
|
|
]);
|
|
|
- if (!this.options.removeAudioSrc) {
|
|
|
|
|
- resourcePromises.push(ProcessorHelper.processAttribute(this.doc.querySelectorAll("audio[src], audio > source[src]"), "src", this.baseURI, this.options, "audio", this.cssVariables, this.styles, this.batchRequest));
|
|
|
|
|
- }
|
|
|
|
|
- if (!this.options.removeVideoSrc) {
|
|
|
|
|
- resourcePromises.push(ProcessorHelper.processAttribute(this.doc.querySelectorAll("video[src], video > source[src]"), "src", this.baseURI, this.options, "video", this.cssVariables, this.styles, this.batchRequest));
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ resourcePromises.push(ProcessorHelper.processAttribute(this.doc.querySelectorAll("audio[src], audio > source[src]"), "src", this.baseURI, this.options, "audio", this.cssVariables, this.styles, this.batchRequest));
|
|
|
|
|
+ resourcePromises.push(ProcessorHelper.processAttribute(this.doc.querySelectorAll("video[src], video > source[src]"), "src", this.baseURI, this.options, "video", this.cssVariables, this.styles, this.batchRequest));
|
|
|
await Promise.all(resourcePromises);
|
|
await Promise.all(resourcePromises);
|
|
|
if (this.options.saveFavicon) {
|
|
if (this.options.saveFavicon) {
|
|
|
ProcessorHelper.processShortcutIcons(this.doc);
|
|
ProcessorHelper.processShortcutIcons(this.doc);
|
|
@@ -1292,60 +1250,46 @@ class Processor {
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
async processScripts() {
|
|
async processScripts() {
|
|
|
- await Promise.all(Array.from(this.doc.querySelectorAll("script[src], link[rel=preload][as=script][href]")).map(async element => {
|
|
|
|
|
|
|
+ await Promise.all(Array.from(this.doc.querySelectorAll("script[src]")).map(async element => {
|
|
|
let resourceURL;
|
|
let resourceURL;
|
|
|
let scriptSrc;
|
|
let scriptSrc;
|
|
|
- if (element.tagName == "SCRIPT") {
|
|
|
|
|
- scriptSrc = element.getAttribute("src");
|
|
|
|
|
- if (this.options.saveOriginalURLs && !isDataURL(scriptSrc)) {
|
|
|
|
|
- element.setAttribute("data-sf-original-src", scriptSrc);
|
|
|
|
|
- }
|
|
|
|
|
- } else {
|
|
|
|
|
- scriptSrc = element.getAttribute("href");
|
|
|
|
|
- if (this.options.saveOriginalURLs && !isDataURL(scriptSrc)) {
|
|
|
|
|
- element.setAttribute("data-sf-original-href", scriptSrc);
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ scriptSrc = element.getAttribute("src");
|
|
|
|
|
+ if (this.options.saveOriginalURLs && !isDataURL(scriptSrc)) {
|
|
|
|
|
+ element.setAttribute("data-sf-original-src", scriptSrc);
|
|
|
}
|
|
}
|
|
|
element.removeAttribute("integrity");
|
|
element.removeAttribute("integrity");
|
|
|
- element.textContent = "";
|
|
|
|
|
- try {
|
|
|
|
|
- resourceURL = util.resolveURL(scriptSrc, this.baseURI);
|
|
|
|
|
- } catch (error) {
|
|
|
|
|
- // ignored
|
|
|
|
|
- }
|
|
|
|
|
- if (testValidURL(resourceURL)) {
|
|
|
|
|
- if (element.tagName == "SCRIPT") {
|
|
|
|
|
- element.removeAttribute("src");
|
|
|
|
|
- } else {
|
|
|
|
|
- element.removeAttribute("href");
|
|
|
|
|
|
|
+ if (!this.options.blockScripts) {
|
|
|
|
|
+ element.textContent = "";
|
|
|
|
|
+ try {
|
|
|
|
|
+ resourceURL = util.resolveURL(scriptSrc, this.baseURI);
|
|
|
|
|
+ } catch (error) {
|
|
|
|
|
+ // ignored
|
|
|
}
|
|
}
|
|
|
- this.stats.add("processed", "scripts", 1);
|
|
|
|
|
- const content = await util.getContent(resourceURL, {
|
|
|
|
|
- asBinary: true,
|
|
|
|
|
- charset: this.charset != UTF8_CHARSET && this.charset,
|
|
|
|
|
- maxResourceSize: this.options.maxResourceSize,
|
|
|
|
|
- maxResourceSizeEnabled: this.options.maxResourceSizeEnabled,
|
|
|
|
|
- frameId: this.options.windowId,
|
|
|
|
|
- resourceReferrer: this.options.resourceReferrer,
|
|
|
|
|
- baseURI: this.options.baseURI,
|
|
|
|
|
- blockMixedContent: this.options.blockMixedContent,
|
|
|
|
|
- expectedType: "script",
|
|
|
|
|
- acceptHeaders: this.options.acceptHeaders,
|
|
|
|
|
- networkTimeout: this.options.networkTimeout
|
|
|
|
|
- });
|
|
|
|
|
- content.data = getUpdatedResourceContent(resourceURL, content, this.options);
|
|
|
|
|
- if (element.tagName == "SCRIPT") {
|
|
|
|
|
|
|
+ if (testValidURL(resourceURL)) {
|
|
|
|
|
+ element.removeAttribute("src");
|
|
|
|
|
+ const content = await util.getContent(resourceURL, {
|
|
|
|
|
+ asBinary: true,
|
|
|
|
|
+ charset: this.charset != UTF8_CHARSET && this.charset,
|
|
|
|
|
+ maxResourceSize: this.options.maxResourceSize,
|
|
|
|
|
+ maxResourceSizeEnabled: this.options.maxResourceSizeEnabled,
|
|
|
|
|
+ frameId: this.options.windowId,
|
|
|
|
|
+ resourceReferrer: this.options.resourceReferrer,
|
|
|
|
|
+ baseURI: this.options.baseURI,
|
|
|
|
|
+ blockMixedContent: this.options.blockMixedContent,
|
|
|
|
|
+ expectedType: "script",
|
|
|
|
|
+ acceptHeaders: this.options.acceptHeaders,
|
|
|
|
|
+ networkTimeout: this.options.networkTimeout
|
|
|
|
|
+ });
|
|
|
|
|
+ content.data = getUpdatedResourceContent(resourceURL, content, this.options);
|
|
|
element.setAttribute("src", content.data);
|
|
element.setAttribute("src", content.data);
|
|
|
- if (element.getAttribute("async") == "" || element.getAttribute("async") == "async" || element.getAttribute(util.ASYNC_SCRIPT_ATTRIBUTE_NAME) == "") {
|
|
|
|
|
|
|
+ if (element.getAttribute("async") == "async" || element.getAttribute(util.ASYNC_SCRIPT_ATTRIBUTE_NAME) == "") {
|
|
|
element.setAttribute("async", "");
|
|
element.setAttribute("async", "");
|
|
|
}
|
|
}
|
|
|
- } else {
|
|
|
|
|
- const scriptElement = this.doc.createElement("script");
|
|
|
|
|
- scriptElement.setAttribute("src", content.data);
|
|
|
|
|
- scriptElement.setAttribute("async", "");
|
|
|
|
|
- element.parentElement.replaceChild(scriptElement, element);
|
|
|
|
|
}
|
|
}
|
|
|
|
|
+ } else {
|
|
|
|
|
+ element.removeAttribute("src");
|
|
|
}
|
|
}
|
|
|
|
|
+ this.stats.add("processed", "scripts", 1);
|
|
|
}));
|
|
}));
|
|
|
}
|
|
}
|
|
|
|
|
|
|
@@ -1580,7 +1524,6 @@ class Processor {
|
|
|
// ---------------
|
|
// ---------------
|
|
|
const DATA_URI_PREFIX = "data:";
|
|
const DATA_URI_PREFIX = "data:";
|
|
|
const ABOUT_BLANK_URI = "about:blank";
|
|
const ABOUT_BLANK_URI = "about:blank";
|
|
|
-const EMPTY_DATA_URI = "data:null;base64,";
|
|
|
|
|
const REGEXP_URL_HASH = /(#.+?)$/;
|
|
const REGEXP_URL_HASH = /(#.+?)$/;
|
|
|
const SINGLE_FILE_VARIABLE_NAME_PREFIX = "--sf-img-";
|
|
const SINGLE_FILE_VARIABLE_NAME_PREFIX = "--sf-img-";
|
|
|
const SINGLE_FILE_VARIABLE_MAX_SIZE = 512 * 1024;
|
|
const SINGLE_FILE_VARIABLE_MAX_SIZE = 512 * 1024;
|
|
@@ -1819,9 +1762,9 @@ class ProcessorHelper {
|
|
|
} else {
|
|
} else {
|
|
|
let newUrlFunction;
|
|
let newUrlFunction;
|
|
|
if (originalResourceURL) {
|
|
if (originalResourceURL) {
|
|
|
- newUrlFunction = urlFunction.replace(originalResourceURL, EMPTY_DATA_URI);
|
|
|
|
|
|
|
+ newUrlFunction = urlFunction.replace(originalResourceURL, util.EMPTY_RESOURCE);
|
|
|
} else {
|
|
} else {
|
|
|
- newUrlFunction = "url(" + EMPTY_DATA_URI + ")";
|
|
|
|
|
|
|
+ newUrlFunction = "url(" + util.EMPTY_RESOURCE + ")";
|
|
|
}
|
|
}
|
|
|
stylesheetContent = stylesheetContent.replace(getRegExp(urlFunction), newUrlFunction);
|
|
stylesheetContent = stylesheetContent.replace(getRegExp(urlFunction), newUrlFunction);
|
|
|
}
|
|
}
|
|
@@ -1890,9 +1833,9 @@ class ProcessorHelper {
|
|
|
const urlFunctions = getUrlFunctions(getCSSValue(declaration.value), true);
|
|
const urlFunctions = getUrlFunctions(getCSSValue(declaration.value), true);
|
|
|
await Promise.all(urlFunctions.map(async urlFunction => {
|
|
await Promise.all(urlFunctions.map(async urlFunction => {
|
|
|
const originalResourceURL = matchURL(urlFunction);
|
|
const originalResourceURL = matchURL(urlFunction);
|
|
|
- const resourceURL = normalizeURL(originalResourceURL);
|
|
|
|
|
- if (!testIgnoredPath(resourceURL)) {
|
|
|
|
|
- if (testValidURL(resourceURL)) {
|
|
|
|
|
|
|
+ if (!options.blockFonts) {
|
|
|
|
|
+ const resourceURL = normalizeURL(originalResourceURL);
|
|
|
|
|
+ if (!testIgnoredPath(resourceURL) && testValidURL(resourceURL)) {
|
|
|
let { content } = await batchRequest.addURL(resourceURL,
|
|
let { content } = await batchRequest.addURL(resourceURL,
|
|
|
{ asBinary: true, expectedType: "font", baseURI, blockMixedContent: options.blockMixedContent });
|
|
{ asBinary: true, expectedType: "font", baseURI, blockMixedContent: options.blockMixedContent });
|
|
|
let resourceURLs = options.fontDeclarations.get(declaration);
|
|
let resourceURLs = options.fontDeclarations.get(declaration);
|
|
@@ -1903,6 +1846,8 @@ class ProcessorHelper {
|
|
|
resourceURLs.push(resourceURL);
|
|
resourceURLs.push(resourceURL);
|
|
|
replaceURLs(declaration, originalResourceURL, content);
|
|
replaceURLs(declaration, originalResourceURL, content);
|
|
|
}
|
|
}
|
|
|
|
|
+ } else {
|
|
|
|
|
+ replaceURLs(declaration, originalResourceURL, util.EMPTY_RESOURCE);
|
|
|
}
|
|
}
|
|
|
}));
|
|
}));
|
|
|
}
|
|
}
|
|
@@ -1931,9 +1876,9 @@ class ProcessorHelper {
|
|
|
const urlFunctions = getUrlFunctions(getCSSValue(declaration.value));
|
|
const urlFunctions = getUrlFunctions(getCSSValue(declaration.value));
|
|
|
await Promise.all(urlFunctions.map(async urlFunction => {
|
|
await Promise.all(urlFunctions.map(async urlFunction => {
|
|
|
const originalResourceURL = matchURL(urlFunction);
|
|
const originalResourceURL = matchURL(urlFunction);
|
|
|
- const resourceURL = normalizeURL(originalResourceURL);
|
|
|
|
|
- if (!testIgnoredPath(resourceURL)) {
|
|
|
|
|
- if (testValidURL(resourceURL)) {
|
|
|
|
|
|
|
+ if (!options.blockImages) {
|
|
|
|
|
+ const resourceURL = normalizeURL(originalResourceURL);
|
|
|
|
|
+ if (!testIgnoredPath(resourceURL) && testValidURL(resourceURL)) {
|
|
|
let { content, indexResource, duplicate } = await batchRequest.addURL(resourceURL,
|
|
let { content, indexResource, duplicate } = await batchRequest.addURL(resourceURL,
|
|
|
{ asBinary: true, expectedType: "image", groupDuplicates: options.groupDuplicateImages });
|
|
{ asBinary: true, expectedType: "image", groupDuplicates: options.groupDuplicateImages });
|
|
|
let variableDefined;
|
|
let variableDefined;
|
|
@@ -1954,6 +1899,8 @@ class ProcessorHelper {
|
|
|
tokens.forEach(({ parent, token, value }) => parent.replace(token, value));
|
|
tokens.forEach(({ parent, token, value }) => parent.replace(token, value));
|
|
|
}
|
|
}
|
|
|
}
|
|
}
|
|
|
|
|
+ } else {
|
|
|
|
|
+ findURLToken(originalResourceURL, declaration.value.children, token => token.data.value.value = util.EMPTY_RESOURCE);
|
|
|
}
|
|
}
|
|
|
}));
|
|
}));
|
|
|
}
|
|
}
|
|
@@ -1981,62 +1928,66 @@ class ProcessorHelper {
|
|
|
resourceElement.setAttribute("data-sf-original-" + attributeName, resourceURL);
|
|
resourceElement.setAttribute("data-sf-original-" + attributeName, resourceURL);
|
|
|
}
|
|
}
|
|
|
delete resourceElement.dataset.singleFileOriginURL;
|
|
delete resourceElement.dataset.singleFileOriginURL;
|
|
|
- if (!testIgnoredPath(resourceURL)) {
|
|
|
|
|
- resourceElement.setAttribute(attributeName, EMPTY_IMAGE);
|
|
|
|
|
- if (testValidPath(resourceURL)) {
|
|
|
|
|
- try {
|
|
|
|
|
- resourceURL = util.resolveURL(resourceURL, baseURI);
|
|
|
|
|
- } catch (error) {
|
|
|
|
|
- // ignored
|
|
|
|
|
- }
|
|
|
|
|
- if (testValidURL(resourceURL)) {
|
|
|
|
|
- let { content, indexResource, duplicate } = await batchRequest.addURL(resourceURL,
|
|
|
|
|
- { asBinary: true, expectedType, groupDuplicates: options.groupDuplicateImages && resourceElement.tagName == "IMG" && attributeName == "src" });
|
|
|
|
|
- if (originURL) {
|
|
|
|
|
- if (content == EMPTY_DATA_URI) {
|
|
|
|
|
- try {
|
|
|
|
|
- originURL = util.resolveURL(originURL, baseURI);
|
|
|
|
|
- } catch (error) {
|
|
|
|
|
- // ignored
|
|
|
|
|
- }
|
|
|
|
|
- try {
|
|
|
|
|
- resourceURL = originURL;
|
|
|
|
|
- content = (await util.getContent(resourceURL, {
|
|
|
|
|
- asBinary: true,
|
|
|
|
|
- expectedType: "image",
|
|
|
|
|
- maxResourceSize: options.maxResourceSize,
|
|
|
|
|
- maxResourceSizeEnabled: options.maxResourceSizeEnabled,
|
|
|
|
|
- frameId: options.windowId,
|
|
|
|
|
- resourceReferrer: options.resourceReferrer,
|
|
|
|
|
- acceptHeaders: options.acceptHeaders,
|
|
|
|
|
- networkTimeout: options.networkTimeout
|
|
|
|
|
- })).data;
|
|
|
|
|
- } catch (error) {
|
|
|
|
|
- // ignored
|
|
|
|
|
|
|
+ if (!options["block" + expectedType.charAt(0).toUpperCase() + expectedType.substring(1) + "s"]) {
|
|
|
|
|
+ if (!testIgnoredPath(resourceURL)) {
|
|
|
|
|
+ resourceElement.setAttribute(attributeName, util.EMPTY_RESOURCE);
|
|
|
|
|
+ if (testValidPath(resourceURL)) {
|
|
|
|
|
+ try {
|
|
|
|
|
+ resourceURL = util.resolveURL(resourceURL, baseURI);
|
|
|
|
|
+ } catch (error) {
|
|
|
|
|
+ // ignored
|
|
|
|
|
+ }
|
|
|
|
|
+ if (testValidURL(resourceURL)) {
|
|
|
|
|
+ let { content, indexResource, duplicate } = await batchRequest.addURL(resourceURL,
|
|
|
|
|
+ { asBinary: true, expectedType, groupDuplicates: options.groupDuplicateImages && resourceElement.tagName == "IMG" && attributeName == "src" });
|
|
|
|
|
+ if (originURL) {
|
|
|
|
|
+ if (content == util.EMPTY_RESOURCE) {
|
|
|
|
|
+ try {
|
|
|
|
|
+ originURL = util.resolveURL(originURL, baseURI);
|
|
|
|
|
+ } catch (error) {
|
|
|
|
|
+ // ignored
|
|
|
|
|
+ }
|
|
|
|
|
+ try {
|
|
|
|
|
+ resourceURL = originURL;
|
|
|
|
|
+ content = (await util.getContent(resourceURL, {
|
|
|
|
|
+ asBinary: true,
|
|
|
|
|
+ expectedType,
|
|
|
|
|
+ maxResourceSize: options.maxResourceSize,
|
|
|
|
|
+ maxResourceSizeEnabled: options.maxResourceSizeEnabled,
|
|
|
|
|
+ frameId: options.windowId,
|
|
|
|
|
+ resourceReferrer: options.resourceReferrer,
|
|
|
|
|
+ acceptHeaders: options.acceptHeaders,
|
|
|
|
|
+ networkTimeout: options.networkTimeout
|
|
|
|
|
+ })).data;
|
|
|
|
|
+ } catch (error) {
|
|
|
|
|
+ // ignored
|
|
|
|
|
+ }
|
|
|
}
|
|
}
|
|
|
}
|
|
}
|
|
|
- }
|
|
|
|
|
- if (removeElementIfMissing && content == EMPTY_DATA_URI) {
|
|
|
|
|
- resourceElement.remove();
|
|
|
|
|
- } else if (content !== EMPTY_DATA_URI) {
|
|
|
|
|
- const forbiddenPrefixFound = PREFIXES_FORBIDDEN_DATA_URI.filter(prefixDataURI => content.startsWith(prefixDataURI)).length;
|
|
|
|
|
- if (!forbiddenPrefixFound) {
|
|
|
|
|
- const isSVG = content.startsWith(PREFIX_DATA_URI_IMAGE_SVG);
|
|
|
|
|
- if (processDuplicates && duplicate && options.groupDuplicateImages && !isSVG && util.getContentSize(content) < SINGLE_FILE_VARIABLE_MAX_SIZE) {
|
|
|
|
|
- if (ProcessorHelper.replaceImageSource(resourceElement, SINGLE_FILE_VARIABLE_NAME_PREFIX + indexResource, options)) {
|
|
|
|
|
- cssVariables.set(indexResource, { content, url: originURL });
|
|
|
|
|
- const declarationList = cssTree.parse(resourceElement.getAttribute("style"), { context: "declarationList" });
|
|
|
|
|
- styles.set(resourceElement, declarationList);
|
|
|
|
|
|
|
+ if (removeElementIfMissing && content == util.EMPTY_RESOURCE) {
|
|
|
|
|
+ resourceElement.remove();
|
|
|
|
|
+ } else if (content !== util.EMPTY_RESOURCE) {
|
|
|
|
|
+ const forbiddenPrefixFound = PREFIXES_FORBIDDEN_DATA_URI.filter(prefixDataURI => content.startsWith(prefixDataURI)).length;
|
|
|
|
|
+ if (!forbiddenPrefixFound) {
|
|
|
|
|
+ const isSVG = content.startsWith(PREFIX_DATA_URI_IMAGE_SVG);
|
|
|
|
|
+ if (expectedType == "image" && processDuplicates && duplicate && !isSVG && util.getContentSize(content) < SINGLE_FILE_VARIABLE_MAX_SIZE) {
|
|
|
|
|
+ if (ProcessorHelper.replaceImageSource(resourceElement, SINGLE_FILE_VARIABLE_NAME_PREFIX + indexResource, options)) {
|
|
|
|
|
+ cssVariables.set(indexResource, { content, url: originURL });
|
|
|
|
|
+ const declarationList = cssTree.parse(resourceElement.getAttribute("style"), { context: "declarationList" });
|
|
|
|
|
+ styles.set(resourceElement, declarationList);
|
|
|
|
|
+ } else {
|
|
|
|
|
+ resourceElement.setAttribute(attributeName, content);
|
|
|
|
|
+ }
|
|
|
} else {
|
|
} else {
|
|
|
resourceElement.setAttribute(attributeName, content);
|
|
resourceElement.setAttribute(attributeName, content);
|
|
|
}
|
|
}
|
|
|
- } else {
|
|
|
|
|
- resourceElement.setAttribute(attributeName, content);
|
|
|
|
|
}
|
|
}
|
|
|
}
|
|
}
|
|
|
}
|
|
}
|
|
|
}
|
|
}
|
|
|
}
|
|
}
|
|
|
|
|
+ } else {
|
|
|
|
|
+ resourceElement.setAttribute(attributeName, util.EMPTY_RESOURCE);
|
|
|
}
|
|
}
|
|
|
}
|
|
}
|
|
|
}));
|
|
}));
|
|
@@ -2054,75 +2005,83 @@ class ProcessorHelper {
|
|
|
resourceElement.setAttribute("data-sf-original-href", originalResourceURL);
|
|
resourceElement.setAttribute("data-sf-original-href", originalResourceURL);
|
|
|
}
|
|
}
|
|
|
let resourceURL = normalizeURL(originalResourceURL);
|
|
let resourceURL = normalizeURL(originalResourceURL);
|
|
|
- if (testValidPath(resourceURL) && !testIgnoredPath(resourceURL)) {
|
|
|
|
|
- resourceElement.setAttribute(attributeName, EMPTY_IMAGE);
|
|
|
|
|
- try {
|
|
|
|
|
- resourceURL = util.resolveURL(resourceURL, baseURI);
|
|
|
|
|
- } catch (error) {
|
|
|
|
|
- // ignored
|
|
|
|
|
- }
|
|
|
|
|
- if (testValidURL(resourceURL)) {
|
|
|
|
|
- const hashMatch = originalResourceURL.match(REGEXP_URL_HASH);
|
|
|
|
|
- if (originalResourceURL.startsWith(baseURI + "#")) {
|
|
|
|
|
- resourceElement.setAttribute(attributeName, hashMatch[0]);
|
|
|
|
|
- } else {
|
|
|
|
|
- const response = await batchRequest.addURL(resourceURL, { expectedType: "image" });
|
|
|
|
|
- const svgDoc = util.parseSVGContent(response.content);
|
|
|
|
|
- if (hashMatch && hashMatch[0]) {
|
|
|
|
|
- let symbolElement;
|
|
|
|
|
- try {
|
|
|
|
|
- symbolElement = svgDoc.querySelector(hashMatch[0]);
|
|
|
|
|
- } catch (error) {
|
|
|
|
|
- // ignored
|
|
|
|
|
- }
|
|
|
|
|
- if (symbolElement) {
|
|
|
|
|
- resourceElement.setAttribute(attributeName, hashMatch[0]);
|
|
|
|
|
- resourceElement.parentElement.insertBefore(symbolElement, resourceElement.parentElement.firstChild);
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ if (!options.blockImages) {
|
|
|
|
|
+ if (testValidPath(resourceURL) && !testIgnoredPath(resourceURL)) {
|
|
|
|
|
+ resourceElement.setAttribute(attributeName, util.EMPTY_RESOURCE);
|
|
|
|
|
+ try {
|
|
|
|
|
+ resourceURL = util.resolveURL(resourceURL, baseURI);
|
|
|
|
|
+ } catch (error) {
|
|
|
|
|
+ // ignored
|
|
|
|
|
+ }
|
|
|
|
|
+ if (testValidURL(resourceURL)) {
|
|
|
|
|
+ const hashMatch = originalResourceURL.match(REGEXP_URL_HASH);
|
|
|
|
|
+ if (originalResourceURL.startsWith(baseURI + "#")) {
|
|
|
|
|
+ resourceElement.setAttribute(attributeName, hashMatch[0]);
|
|
|
} else {
|
|
} else {
|
|
|
- const content = await batchRequest.addURL(resourceURL, { expectedType: "image" });
|
|
|
|
|
- resourceElement.setAttribute(attributeName, PREFIX_DATA_URI_IMAGE_SVG + "," + content);
|
|
|
|
|
|
|
+ const response = await batchRequest.addURL(resourceURL, { expectedType: "image" });
|
|
|
|
|
+ const svgDoc = util.parseSVGContent(response.content);
|
|
|
|
|
+ if (hashMatch && hashMatch[0]) {
|
|
|
|
|
+ let symbolElement;
|
|
|
|
|
+ try {
|
|
|
|
|
+ symbolElement = svgDoc.querySelector(hashMatch[0]);
|
|
|
|
|
+ } catch (error) {
|
|
|
|
|
+ // ignored
|
|
|
|
|
+ }
|
|
|
|
|
+ if (symbolElement) {
|
|
|
|
|
+ resourceElement.setAttribute(attributeName, hashMatch[0]);
|
|
|
|
|
+ resourceElement.parentElement.insertBefore(symbolElement, resourceElement.parentElement.firstChild);
|
|
|
|
|
+ }
|
|
|
|
|
+ } else {
|
|
|
|
|
+ const content = await batchRequest.addURL(resourceURL, { expectedType: "image" });
|
|
|
|
|
+ resourceElement.setAttribute(attributeName, PREFIX_DATA_URI_IMAGE_SVG + "," + content);
|
|
|
|
|
+ }
|
|
|
}
|
|
}
|
|
|
}
|
|
}
|
|
|
|
|
+ } else if (resourceURL == options.url) {
|
|
|
|
|
+ resourceElement.setAttribute(attributeName, originalResourceURL.substring(resourceURL.length));
|
|
|
}
|
|
}
|
|
|
- } else if (resourceURL == options.url) {
|
|
|
|
|
- resourceElement.setAttribute(attributeName, originalResourceURL.substring(resourceURL.length));
|
|
|
|
|
|
|
+ } else {
|
|
|
|
|
+ resourceElement.setAttribute(attributeName, util.EMPTY_RESOURCE);
|
|
|
}
|
|
}
|
|
|
}));
|
|
}));
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
- static async processSrcset(resourceElements, attributeName, baseURI, batchRequest) {
|
|
|
|
|
|
|
+ static async processSrcset(resourceElements, baseURI, options, batchRequest) {
|
|
|
await Promise.all(Array.from(resourceElements).map(async resourceElement => {
|
|
await Promise.all(Array.from(resourceElements).map(async resourceElement => {
|
|
|
- const originSrcset = resourceElement.getAttribute(attributeName);
|
|
|
|
|
|
|
+ const originSrcset = resourceElement.getAttribute("srcset");
|
|
|
const srcset = util.parseSrcset(originSrcset);
|
|
const srcset = util.parseSrcset(originSrcset);
|
|
|
resourceElement.setAttribute("data-sf-original-srcset", originSrcset);
|
|
resourceElement.setAttribute("data-sf-original-srcset", originSrcset);
|
|
|
- const srcsetValues = await Promise.all(srcset.map(async srcsetValue => {
|
|
|
|
|
- let resourceURL = normalizeURL(srcsetValue.url);
|
|
|
|
|
- if (!testIgnoredPath(resourceURL)) {
|
|
|
|
|
- if (testValidPath(resourceURL)) {
|
|
|
|
|
- try {
|
|
|
|
|
- resourceURL = util.resolveURL(resourceURL, baseURI);
|
|
|
|
|
- } catch (error) {
|
|
|
|
|
- // ignored
|
|
|
|
|
- }
|
|
|
|
|
- if (testValidURL(resourceURL)) {
|
|
|
|
|
- const { content } = await batchRequest.addURL(resourceURL, { asBinary: true, expectedType: "image" });
|
|
|
|
|
- const forbiddenPrefixFound = PREFIXES_FORBIDDEN_DATA_URI.filter(prefixDataURI => content.startsWith(prefixDataURI)).length;
|
|
|
|
|
- if (forbiddenPrefixFound) {
|
|
|
|
|
|
|
+ if (!options.blockImages) {
|
|
|
|
|
+ const srcsetValues = await Promise.all(srcset.map(async srcsetValue => {
|
|
|
|
|
+ let resourceURL = normalizeURL(srcsetValue.url);
|
|
|
|
|
+ if (!testIgnoredPath(resourceURL)) {
|
|
|
|
|
+ if (testValidPath(resourceURL)) {
|
|
|
|
|
+ try {
|
|
|
|
|
+ resourceURL = util.resolveURL(resourceURL, baseURI);
|
|
|
|
|
+ } catch (error) {
|
|
|
|
|
+ // ignored
|
|
|
|
|
+ }
|
|
|
|
|
+ if (testValidURL(resourceURL)) {
|
|
|
|
|
+ const { content } = await batchRequest.addURL(resourceURL, { asBinary: true, expectedType: "image" });
|
|
|
|
|
+ const forbiddenPrefixFound = PREFIXES_FORBIDDEN_DATA_URI.filter(prefixDataURI => content.startsWith(prefixDataURI)).length;
|
|
|
|
|
+ if (forbiddenPrefixFound) {
|
|
|
|
|
+ return "";
|
|
|
|
|
+ }
|
|
|
|
|
+ return content + (srcsetValue.w ? " " + srcsetValue.w + "w" : srcsetValue.d ? " " + srcsetValue.d + "x" : "");
|
|
|
|
|
+ } else {
|
|
|
return "";
|
|
return "";
|
|
|
}
|
|
}
|
|
|
- return content + (srcsetValue.w ? " " + srcsetValue.w + "w" : srcsetValue.d ? " " + srcsetValue.d + "x" : "");
|
|
|
|
|
} else {
|
|
} else {
|
|
|
return "";
|
|
return "";
|
|
|
}
|
|
}
|
|
|
} else {
|
|
} else {
|
|
|
- return "";
|
|
|
|
|
|
|
+ return resourceURL + (srcsetValue.w ? " " + srcsetValue.w + "w" : srcsetValue.d ? " " + srcsetValue.d + "x" : "");
|
|
|
}
|
|
}
|
|
|
- } else {
|
|
|
|
|
- return resourceURL + (srcsetValue.w ? " " + srcsetValue.w + "w" : srcsetValue.d ? " " + srcsetValue.d + "x" : "");
|
|
|
|
|
- }
|
|
|
|
|
- }));
|
|
|
|
|
- resourceElement.setAttribute(attributeName, srcsetValues.join(", "));
|
|
|
|
|
|
|
+ }));
|
|
|
|
|
+ resourceElement.setAttribute("srcset", srcsetValues.join(", "));
|
|
|
|
|
+ } else {
|
|
|
|
|
+ resourceElement.setAttribute("srcset", "");
|
|
|
|
|
+ }
|
|
|
}));
|
|
}));
|
|
|
}
|
|
}
|
|
|
|
|
|
|
@@ -2303,7 +2262,7 @@ function getImportFunctions(stylesheetContent) {
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
function findShortcutIcon(shortcutIcons) {
|
|
function findShortcutIcon(shortcutIcons) {
|
|
|
- shortcutIcons = shortcutIcons.filter(linkElement => linkElement.href != EMPTY_IMAGE);
|
|
|
|
|
|
|
+ shortcutIcons = shortcutIcons.filter(linkElement => linkElement.href != util.EMPTY_RESOURCE);
|
|
|
shortcutIcons.sort((linkElement1, linkElement2) => (parseInt(linkElement2.sizes, 10) || 16) - (parseInt(linkElement1.sizes, 10) || 16));
|
|
shortcutIcons.sort((linkElement1, linkElement2) => (parseInt(linkElement2.sizes, 10) || 16) - (parseInt(linkElement1.sizes, 10) || 16));
|
|
|
return shortcutIcons[0];
|
|
return shortcutIcons[0];
|
|
|
}
|
|
}
|