mirror of https://github.com/webpack/webpack.git
fix: no extra runtime for external asset modules in CSS
This commit is contained in:
commit
abe931a19b
|
@ -55,8 +55,9 @@ const { register } = require("./util/serialization");
|
||||||
|
|
||||||
/** @typedef {{ attributes?: ImportAttributes, externalType: "import" | "module" | undefined }} ImportDependencyMeta */
|
/** @typedef {{ attributes?: ImportAttributes, externalType: "import" | "module" | undefined }} ImportDependencyMeta */
|
||||||
/** @typedef {{ layer?: string, supports?: string, media?: string }} CssImportDependencyMeta */
|
/** @typedef {{ layer?: string, supports?: string, media?: string }} CssImportDependencyMeta */
|
||||||
|
/** @typedef {{ sourceType: "css-url" }} AssetDependencyMeta */
|
||||||
|
|
||||||
/** @typedef {ImportDependencyMeta | CssImportDependencyMeta} DependencyMeta */
|
/** @typedef {ImportDependencyMeta | CssImportDependencyMeta | AssetDependencyMeta} DependencyMeta */
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @typedef {object} SourceData
|
* @typedef {object} SourceData
|
||||||
|
@ -69,6 +70,7 @@ const { register } = require("./util/serialization");
|
||||||
|
|
||||||
const TYPES = new Set(["javascript"]);
|
const TYPES = new Set(["javascript"]);
|
||||||
const CSS_TYPES = new Set(["css-import"]);
|
const CSS_TYPES = new Set(["css-import"]);
|
||||||
|
const CSS_URL_TYPES = new Set(["css-url"]);
|
||||||
const RUNTIME_REQUIREMENTS = new Set([RuntimeGlobals.module]);
|
const RUNTIME_REQUIREMENTS = new Set([RuntimeGlobals.module]);
|
||||||
const RUNTIME_REQUIREMENTS_FOR_SCRIPT = new Set([RuntimeGlobals.loadScript]);
|
const RUNTIME_REQUIREMENTS_FOR_SCRIPT = new Set([RuntimeGlobals.loadScript]);
|
||||||
const RUNTIME_REQUIREMENTS_FOR_MODULE = new Set([
|
const RUNTIME_REQUIREMENTS_FOR_MODULE = new Set([
|
||||||
|
@ -500,7 +502,18 @@ class ExternalModule extends Module {
|
||||||
* @returns {SourceTypes} types available (do not mutate)
|
* @returns {SourceTypes} types available (do not mutate)
|
||||||
*/
|
*/
|
||||||
getSourceTypes() {
|
getSourceTypes() {
|
||||||
return this.externalType === "css-import" ? CSS_TYPES : TYPES;
|
if (
|
||||||
|
this.externalType === "asset" &&
|
||||||
|
this.dependencyMeta &&
|
||||||
|
/** @type {AssetDependencyMeta} */
|
||||||
|
(this.dependencyMeta).sourceType === "css-url"
|
||||||
|
) {
|
||||||
|
return CSS_URL_TYPES;
|
||||||
|
} else if (this.externalType === "css-import") {
|
||||||
|
return CSS_TYPES;
|
||||||
|
}
|
||||||
|
|
||||||
|
return TYPES;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -674,12 +687,24 @@ class ExternalModule extends Module {
|
||||||
if (externalType === "module-import") {
|
if (externalType === "module-import") {
|
||||||
if (
|
if (
|
||||||
this.dependencyMeta &&
|
this.dependencyMeta &&
|
||||||
/** @type {ImportDependencyMeta} */ (this.dependencyMeta).externalType
|
/** @type {ImportDependencyMeta} */
|
||||||
|
(this.dependencyMeta).externalType
|
||||||
) {
|
) {
|
||||||
return /** @type {ImportDependencyMeta} */ (this.dependencyMeta)
|
return /** @type {ImportDependencyMeta} */ (this.dependencyMeta)
|
||||||
.externalType;
|
.externalType;
|
||||||
}
|
}
|
||||||
return "module";
|
return "module";
|
||||||
|
} else if (externalType === "asset") {
|
||||||
|
if (
|
||||||
|
this.dependencyMeta &&
|
||||||
|
/** @type {AssetDependencyMeta} */
|
||||||
|
(this.dependencyMeta).sourceType
|
||||||
|
) {
|
||||||
|
return /** @type {AssetDependencyMeta} */ (this.dependencyMeta)
|
||||||
|
.sourceType;
|
||||||
|
}
|
||||||
|
|
||||||
|
return "asset";
|
||||||
}
|
}
|
||||||
|
|
||||||
return externalType;
|
return externalType;
|
||||||
|
@ -816,10 +841,13 @@ class ExternalModule extends Module {
|
||||||
new RawSource(`module.exports = ${JSON.stringify(request)};`)
|
new RawSource(`module.exports = ${JSON.stringify(request)};`)
|
||||||
);
|
);
|
||||||
const data = new Map();
|
const data = new Map();
|
||||||
data.set("url", {
|
data.set("url", { javascript: request });
|
||||||
javascript: request,
|
return { sources, runtimeRequirements: RUNTIME_REQUIREMENTS, data };
|
||||||
"css-url": request
|
}
|
||||||
});
|
case "css-url": {
|
||||||
|
const sources = new Map();
|
||||||
|
const data = new Map();
|
||||||
|
data.set("url", { "css-url": request });
|
||||||
return { sources, runtimeRequirements: RUNTIME_REQUIREMENTS, data };
|
return { sources, runtimeRequirements: RUNTIME_REQUIREMENTS, data };
|
||||||
}
|
}
|
||||||
case "css-import": {
|
case "css-import": {
|
||||||
|
|
|
@ -9,6 +9,7 @@ const util = require("util");
|
||||||
const ExternalModule = require("./ExternalModule");
|
const ExternalModule = require("./ExternalModule");
|
||||||
const ContextElementDependency = require("./dependencies/ContextElementDependency");
|
const ContextElementDependency = require("./dependencies/ContextElementDependency");
|
||||||
const CssImportDependency = require("./dependencies/CssImportDependency");
|
const CssImportDependency = require("./dependencies/CssImportDependency");
|
||||||
|
const CssUrlDependency = require("./dependencies/CssUrlDependency");
|
||||||
const HarmonyImportDependency = require("./dependencies/HarmonyImportDependency");
|
const HarmonyImportDependency = require("./dependencies/HarmonyImportDependency");
|
||||||
const ImportDependency = require("./dependencies/ImportDependency");
|
const ImportDependency = require("./dependencies/ImportDependency");
|
||||||
const { resolveByProperty, cachedSetProperty } = require("./util/cleverMerge");
|
const { resolveByProperty, cachedSetProperty } = require("./util/cleverMerge");
|
||||||
|
@ -117,6 +118,8 @@ class ExternalModuleFactoryPlugin {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const resolvedType = /** @type {string} */ (type || globalType);
|
||||||
|
|
||||||
// TODO make it pluggable/add hooks to `ExternalModule` to allow output modules own externals?
|
// TODO make it pluggable/add hooks to `ExternalModule` to allow output modules own externals?
|
||||||
/** @type {DependencyMeta | undefined} */
|
/** @type {DependencyMeta | undefined} */
|
||||||
let dependencyMeta;
|
let dependencyMeta;
|
||||||
|
@ -145,12 +148,18 @@ class ExternalModuleFactoryPlugin {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
resolvedType === "asset" &&
|
||||||
|
dependency instanceof CssUrlDependency
|
||||||
|
) {
|
||||||
|
dependencyMeta = { sourceType: "css-url" };
|
||||||
|
}
|
||||||
|
|
||||||
callback(
|
callback(
|
||||||
null,
|
null,
|
||||||
new ExternalModule(
|
new ExternalModule(
|
||||||
externalConfig,
|
externalConfig,
|
||||||
/** @type {string} */
|
resolvedType,
|
||||||
(type || globalType),
|
|
||||||
dependency.request,
|
dependency.request,
|
||||||
dependencyMeta
|
dependencyMeta
|
||||||
)
|
)
|
||||||
|
|
|
@ -4921,7 +4921,7 @@ Array [
|
||||||
background:
|
background:
|
||||||
url(img.png),
|
url(img.png),
|
||||||
url(img.png),
|
url(img.png),
|
||||||
url();
|
url(d4da020aedcd249a7a41.png);
|
||||||
url(),
|
url(),
|
||||||
url(resource.png),
|
url(resource.png),
|
||||||
url(),
|
url(),
|
||||||
|
@ -4929,6 +4929,22 @@ Array [
|
||||||
url(https://example.com/img.png);
|
url(https://example.com/img.png);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.class-2 {
|
||||||
|
background: url(shared.png);
|
||||||
|
}
|
||||||
|
|
||||||
|
.class-3 {
|
||||||
|
background: url(shared-external.png);
|
||||||
|
}
|
||||||
|
|
||||||
|
.class-4 {
|
||||||
|
background: url(cde81354a9a8ce8d5f51.gif);
|
||||||
|
}
|
||||||
|
|
||||||
|
.class-5 {
|
||||||
|
background: url(5649e83cc54c4b57bc28.png);
|
||||||
|
}
|
||||||
|
|
||||||
head{--webpack-main:&\\\\.\\\\/style\\\\.css;}",
|
head{--webpack-main:&\\\\.\\\\/style\\\\.css;}",
|
||||||
]
|
]
|
||||||
`;
|
`;
|
||||||
|
|
|
@ -4921,7 +4921,7 @@ Array [
|
||||||
background:
|
background:
|
||||||
url(img.png),
|
url(img.png),
|
||||||
url(img.png),
|
url(img.png),
|
||||||
url();
|
url(d4da020aedcd249a7a41.png);
|
||||||
url(),
|
url(),
|
||||||
url(resource.png),
|
url(resource.png),
|
||||||
url(),
|
url(),
|
||||||
|
@ -4929,6 +4929,22 @@ Array [
|
||||||
url(https://example.com/img.png);
|
url(https://example.com/img.png);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.class-2 {
|
||||||
|
background: url(shared.png);
|
||||||
|
}
|
||||||
|
|
||||||
|
.class-3 {
|
||||||
|
background: url(shared-external.png);
|
||||||
|
}
|
||||||
|
|
||||||
|
.class-4 {
|
||||||
|
background: url(cde81354a9a8ce8d5f51.gif);
|
||||||
|
}
|
||||||
|
|
||||||
|
.class-5 {
|
||||||
|
background: url(5649e83cc54c4b57bc28.png);
|
||||||
|
}
|
||||||
|
|
||||||
head{--webpack-main:&\\\\.\\\\/style\\\\.css;}",
|
head{--webpack-main:&\\\\.\\\\/style\\\\.css;}",
|
||||||
]
|
]
|
||||||
`;
|
`;
|
||||||
|
|
|
@ -10,5 +10,38 @@ it("should compile", () => {
|
||||||
}
|
}
|
||||||
|
|
||||||
expect(css).toMatchSnapshot();
|
expect(css).toMatchSnapshot();
|
||||||
expect(Object.keys(__webpack_modules__).length).toBe(3)
|
expect(Object.keys(__webpack_modules__).length).toBe(7);
|
||||||
|
expect(__webpack_modules__['./index.js']).toBeDefined();
|
||||||
|
expect(__webpack_modules__['./shared-external.png']).toBeDefined();
|
||||||
|
expect(__webpack_modules__['./shared.png']).toBeDefined();
|
||||||
|
expect(__webpack_modules__['%3D']).toBeDefined();
|
||||||
|
expect(__webpack_modules__['']).toBeDefined();
|
||||||
|
expect(__webpack_modules__['https://example.com/only-external.png']).toBeDefined();
|
||||||
|
expect(__webpack_modules__['./style.css']).toBeDefined();
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should work with shared asset module", () => {
|
||||||
|
const img = new URL("./shared.png", import.meta.url);
|
||||||
|
expect(img.href.endsWith("shared.png")).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should work with shared external asset module", () => {
|
||||||
|
const img = new URL("./shared-external.png", import.meta.url);
|
||||||
|
expect(img.href.endsWith("shared-external.png")).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should work with external asset module", () => {
|
||||||
|
const img = new URL("https://example.com/only-external.png", import.meta.url);
|
||||||
|
expect(img.href.endsWith("only-external.png")).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should work and extract DataURI", () => {
|
||||||
|
const img = new URL("", import.meta.url);
|
||||||
|
expect(img.href.endsWith(".svg")).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should work and extract shared DataURI", () => {
|
||||||
|
const img = new URL("%3D", import.meta.url);
|
||||||
|
expect(img.href.endsWith(".png")).toBe(true);
|
||||||
});
|
});
|
||||||
|
|
Binary file not shown.
After Width: | Height: | Size: 76 KiB |
Binary file not shown.
After Width: | Height: | Size: 76 KiB |
|
@ -10,3 +10,19 @@
|
||||||
url('data:text/html,%3Ch1%3EHello%2C%20World!%3C%2Fh1%3E'),
|
url('data:text/html,%3Ch1%3EHello%2C%20World!%3C%2Fh1%3E'),
|
||||||
url('https://example.com/img.png');
|
url('https://example.com/img.png');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.class-2 {
|
||||||
|
background: url("./shared.png");
|
||||||
|
}
|
||||||
|
|
||||||
|
.class-3 {
|
||||||
|
background: url("./shared-external.png");
|
||||||
|
}
|
||||||
|
|
||||||
|
.class-4 {
|
||||||
|
background: url("");
|
||||||
|
}
|
||||||
|
|
||||||
|
.class-5 {
|
||||||
|
background: url("%3D");
|
||||||
|
}
|
||||||
|
|
|
@ -23,10 +23,25 @@ module.exports = {
|
||||||
{
|
{
|
||||||
mimetype: "text/html",
|
mimetype: "text/html",
|
||||||
type: "asset/resource"
|
type: "asset/resource"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
mimetype: "image/svg",
|
||||||
|
type: "asset/resource"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
mimetype: "image/gif",
|
||||||
|
type: "asset/resource"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
mimetype: "image/png",
|
||||||
|
type: "asset/resource"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
output: {
|
output: {
|
||||||
assetModuleFilename: "[name][ext]"
|
assetModuleFilename: "[name][ext]"
|
||||||
|
},
|
||||||
|
externals: {
|
||||||
|
"shared-external.png": "asset shared-external.png"
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -275,6 +275,9 @@ declare interface Asset {
|
||||||
*/
|
*/
|
||||||
info: AssetInfo;
|
info: AssetInfo;
|
||||||
}
|
}
|
||||||
|
declare interface AssetDependencyMeta {
|
||||||
|
sourceType: "css-url";
|
||||||
|
}
|
||||||
declare interface AssetEmittedInfo {
|
declare interface AssetEmittedInfo {
|
||||||
content: Buffer;
|
content: Buffer;
|
||||||
source: Source;
|
source: Source;
|
||||||
|
@ -4588,12 +4591,18 @@ declare class ExternalModule extends Module {
|
||||||
request: string | string[] | RequestRecord,
|
request: string | string[] | RequestRecord,
|
||||||
type: string,
|
type: string,
|
||||||
userRequest: string,
|
userRequest: string,
|
||||||
dependencyMeta?: ImportDependencyMeta | CssImportDependencyMeta
|
dependencyMeta?:
|
||||||
|
| ImportDependencyMeta
|
||||||
|
| CssImportDependencyMeta
|
||||||
|
| AssetDependencyMeta
|
||||||
);
|
);
|
||||||
request: string | string[] | Record<string, string | string[]>;
|
request: string | string[] | Record<string, string | string[]>;
|
||||||
externalType: string;
|
externalType: string;
|
||||||
userRequest: string;
|
userRequest: string;
|
||||||
dependencyMeta?: ImportDependencyMeta | CssImportDependencyMeta;
|
dependencyMeta?:
|
||||||
|
| ImportDependencyMeta
|
||||||
|
| CssImportDependencyMeta
|
||||||
|
| AssetDependencyMeta;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* restore unsafe cache data
|
* restore unsafe cache data
|
||||||
|
|
Loading…
Reference in New Issue