webpack/test/Errors.test.js

397 lines
10 KiB
JavaScript
Raw Normal View History

2017-01-18 18:29:38 +08:00
"use strict";
/*globals describe it */
2017-01-18 18:29:38 +08:00
const path = require("path");
2017-01-18 18:29:38 +08:00
const webpack = require("../lib/webpack");
2017-01-18 18:29:38 +08:00
const base = path.join(__dirname, "fixtures", "errors");
2017-01-18 18:29:38 +08:00
describe("Errors", () => {
jest.setTimeout(20000);
function customOutputFilesystem(c) {
2017-01-18 18:29:38 +08:00
const files = {};
c.outputFileSystem = {
join: path.join.bind(path),
2018-01-24 23:00:43 +08:00
mkdirp(path, callback) {
callback();
},
2018-01-24 23:00:43 +08:00
writeFile(name, content, callback) {
files[name] = content.toString("utf-8");
callback();
}
};
return files;
}
2015-08-09 18:42:43 +08:00
function getErrors(options, callback) {
options.context = base;
2017-01-18 18:29:38 +08:00
const c = webpack(options);
customOutputFilesystem(c);
2017-01-18 18:29:38 +08:00
c.run((err, stats) => {
2018-02-25 09:00:20 +08:00
if (err) throw err;
2018-01-24 23:00:43 +08:00
expect(typeof stats).toBe("object");
2015-08-09 18:42:43 +08:00
stats = stats.toJson({
errorDetails: false
});
2018-01-24 23:00:43 +08:00
expect(typeof stats).toBe("object");
expect(stats).toHaveProperty("errors");
expect(stats).toHaveProperty("warnings");
expect(Array.isArray(stats.errors)).toBe(true);
expect(Array.isArray(stats.warnings)).toBe(true);
callback(stats.errors, stats.warnings);
});
}
2018-02-22 01:12:32 +08:00
2018-02-22 01:01:53 +08:00
function getErrorsPromise(options, callback) {
return new Promise((resolve, reject) => {
getErrors(options, (errors, warnings) => {
callback(errors, warnings);
resolve();
});
});
2018-02-22 01:06:02 +08:00
}
2018-02-25 09:00:20 +08:00
it("should throw an error if file doesn't exist", done => {
getErrors(
{
mode: "development",
entry: "./missingFile"
},
(errors, warnings) => {
2018-02-25 18:46:17 +08:00
expect(errors).toHaveLength(2);
expect(warnings).toHaveLength(0);
2018-02-25 09:00:20 +08:00
errors.sort();
let lines = errors[0].split("\n");
2018-02-25 18:46:17 +08:00
expect(lines[0]).toMatch(/missingFile.js/);
expect(lines[1]).toMatch(/^Module not found/);
expect(lines[1]).toMatch(/\.\/dir\/missing2/);
expect(lines[2]).toMatch(/missingFile.js 12:9/);
2018-02-25 09:00:20 +08:00
lines = errors[1].split("\n");
2018-02-25 18:46:17 +08:00
expect(lines[0]).toMatch(/missingFile.js/);
expect(lines[1]).toMatch(/^Module not found/);
expect(lines[1]).toMatch(/\.\/missing/);
expect(lines[2]).toMatch(/missingFile.js 4:0/);
2018-02-25 09:00:20 +08:00
done();
}
);
});
2018-02-25 09:00:20 +08:00
it("should report require.extensions as unsupported", done => {
getErrors(
{
mode: "development",
entry: "./require.extensions"
},
(errors, warnings) => {
2018-02-25 18:46:17 +08:00
expect(errors).toHaveLength(0);
2018-01-24 23:00:43 +08:00
expect(warnings).toHaveLength(1);
2017-01-18 18:29:38 +08:00
const lines = warnings[0].split("\n");
2018-02-25 18:46:17 +08:00
expect(lines[0]).toMatch(/require.extensions\.js/);
expect(lines[1]).toMatch(
/require.extensions is not supported by webpack/
);
2018-02-25 09:00:20 +08:00
done();
}
2018-02-25 09:00:20 +08:00
);
});
it("should report require.main.require as unsupported", done => {
getErrors(
{
mode: "development",
entry: "./require.main.require"
},
(errors, warnings) => {
expect(errors).toHaveLength(0);
expect(warnings).toHaveLength(1);
const lines = warnings[0].split("\n");
expect(lines[0]).toMatch(/require.main.require\.js/);
expect(lines[1]).toMatch(
/require.main.require is not supported by webpack/
);
done();
}
);
});
it("should report module.parent.require as unsupported", done => {
getErrors(
{
mode: "development",
entry: "./module.parent.require"
},
(errors, warnings) => {
expect(errors).toHaveLength(0);
expect(warnings).toHaveLength(1);
const lines = warnings[0].split("\n");
expect(lines[0]).toMatch(/module.parent.require\.js/);
expect(lines[1]).toMatch(
/module.parent.require is not supported by webpack/
);
done();
}
);
});
2018-02-25 09:00:20 +08:00
it("should warn about case-sensitive module names", done => {
getErrors(
{
mode: "development",
entry: "./case-sensitive"
},
(errors, warnings) => {
if (errors.length === 0) {
2018-02-25 18:46:17 +08:00
expect(warnings).toHaveLength(1);
2018-02-25 09:00:20 +08:00
const lines = warnings[0].split("\n");
2018-02-25 18:46:17 +08:00
expect(lines[4]).toMatch(/FILE\.js/);
expect(lines[5]).toMatch(/Used by/);
expect(lines[6]).toMatch(/case-sensitive/);
expect(lines[7]).toMatch(/file\.js/);
expect(lines[8]).toMatch(/Used by/);
expect(lines[9]).toMatch(/case-sensitive/);
2018-02-25 09:00:20 +08:00
} else {
2018-02-25 18:46:17 +08:00
expect(errors).toHaveLength(1);
expect(warnings).toHaveLength(0);
2018-02-25 09:00:20 +08:00
}
done();
}
);
});
2018-02-25 09:00:20 +08:00
it("should warn when not using mode", done => {
getErrors(
{
entry: "./entry-point"
},
(errors, warnings) => {
2018-02-25 18:46:17 +08:00
expect(errors).toHaveLength(0);
expect(warnings).toHaveLength(1);
2018-02-25 09:00:20 +08:00
let lines = warnings[0].split("\n");
2018-02-25 18:46:17 +08:00
expect(lines[0]).toMatch(/configuration/);
expect(lines[1]).toMatch(/mode/);
expect(lines[1]).toMatch(/development/);
expect(lines[1]).toMatch(/production/);
2018-02-25 09:00:20 +08:00
done();
}
);
});
2018-02-25 09:00:20 +08:00
it("should not warn if the NoEmitOnErrorsPlugin is used over the NoErrorsPlugin", done => {
getErrors(
{
mode: "production",
entry: "./no-errors-deprecate"
},
(errors, warnings) => {
2018-02-25 18:46:17 +08:00
expect(errors).toHaveLength(0);
expect(warnings).toHaveLength(0);
2018-02-25 09:00:20 +08:00
done();
}
);
});
2018-02-25 09:00:20 +08:00
it("should not not emit if NoEmitOnErrorsPlugin is used and there is an error", done => {
getErrors(
{
mode: "production",
entry: "./missingFile"
},
2018-02-25 09:00:20 +08:00
(errors, warnings) => {
2018-02-25 18:46:17 +08:00
expect(errors).toHaveLength(2);
expect(warnings).toHaveLength(0);
2018-02-25 09:00:20 +08:00
errors.sort();
let lines = errors[0].split("\n");
2018-02-25 18:46:17 +08:00
expect(lines[0]).toMatch(/missingFile.js/);
expect(lines[1]).toMatch(/^Module not found/);
expect(lines[1]).toMatch(/\.\/dir\/missing2/);
expect(lines[2]).toMatch(/missingFile.js 12:9/);
2018-02-25 09:00:20 +08:00
lines = errors[1].split("\n");
2018-02-25 18:46:17 +08:00
expect(lines[0]).toMatch(/missingFile.js/);
expect(lines[1]).toMatch(/^Module not found/);
expect(lines[1]).toMatch(/\.\/missing/);
expect(lines[2]).toMatch(/missingFile.js 4:0/);
2018-02-25 09:00:20 +08:00
done();
}
);
});
it("should throw an error when trying to use [chunkhash] when it's invalid", done => {
getErrors(
{
mode: "development",
entry: {
a: "./entry-point",
b: "./entry-point",
c: "./entry-point"
},
output: {
filename: "[chunkhash].js"
},
plugins: [new webpack.HotModuleReplacementPlugin()]
},
2018-02-25 09:00:20 +08:00
(errors, warnings) => {
2018-02-25 18:46:17 +08:00
expect(errors).toHaveLength(3);
expect(warnings).toHaveLength(0);
2018-02-25 09:00:20 +08:00
errors.forEach(error => {
const lines = error.split("\n");
2018-02-25 18:46:17 +08:00
expect(lines[0]).toMatch(/chunk (a|b|c)/);
expect(lines[2]).toMatch(/\[chunkhash\].js/);
expect(lines[2]).toMatch(/use \[hash\] instead/);
2018-02-25 09:00:20 +08:00
});
done();
}
);
});
2018-03-02 17:56:34 +08:00
it("should show loader name when emit/throw errors or warnings from loaders", () => {
2018-02-22 01:01:53 +08:00
return Promise.all([
2018-03-02 17:56:34 +08:00
getErrorsPromise(
{
mode: "development",
entry: "./entry-point-error-loader-required.js"
},
(errors, warnings) => {
2018-05-01 22:39:05 +08:00
expect(warnings).toHaveLength(1);
expect(warnings[0].split("\n")[1]).toMatch(
/^Module Warning \(from .\/emit-error-loader.js\):$/
);
expect(errors).toHaveLength(1);
expect(errors[0].split("\n")[1]).toMatch(
/^Module Error \(from .\/emit-error-loader.js\):$/
);
2018-03-02 17:56:34 +08:00
}
),
getErrorsPromise(
{
mode: "development",
entry: path.resolve(base, "./emit-error-loader") + "!./entry-point.js"
},
(errors, warnings) => {
2018-05-01 22:39:05 +08:00
expect(warnings).toHaveLength(1);
expect(warnings[0].split("\n")[1]).toMatch(
/^Module Warning \(from .\/emit-error-loader.js\):$/
);
expect(errors).toHaveLength(1);
expect(errors[0].split("\n")[1]).toMatch(
/^Module Error \(from .\/emit-error-loader.js\):$/
);
2018-03-02 17:56:34 +08:00
}
),
getErrorsPromise(
{
mode: "development",
entry: "./not-a-json.js",
module: {
rules: [
2018-02-22 01:01:53 +08:00
{
2018-03-02 17:56:34 +08:00
test: /not-a-json\.js$/,
use: [
"json-loader",
{
loader: path.resolve(base, "./emit-error-loader")
}
]
2018-02-22 01:01:53 +08:00
}
]
2018-03-02 17:56:34 +08:00
}
},
(errors, warnings) => {
2018-05-01 22:39:05 +08:00
expect(warnings).toHaveLength(1);
expect(warnings[0].split("\n")[1]).toMatch(
/^Module Warning \(from .\/emit-error-loader.js\):$/
);
expect(errors).toHaveLength(2);
expect(errors[0].split("\n")[1]).toMatch(
/^Module Error \(from .\/emit-error-loader.js\):$/
);
expect(errors[1].split("\n")[1]).toMatch(
/^Module build failed \(from \(webpack\)\/node_modules\/json-loader\/index.js\):$/
);
2018-02-22 01:01:53 +08:00
}
2018-03-02 17:56:34 +08:00
),
getErrorsPromise(
{
mode: "development",
entry: "./entry-point.js",
module: {
rules: [
{
test: /entry-point\.js$/,
use: path.resolve(base, "./async-error-loader")
}
]
}
},
(errors, warnings) => {
2018-05-01 22:39:05 +08:00
expect(errors).toHaveLength(1);
expect(errors[0].split("\n")[1]).toMatch(
/^Module build failed \(from .\/async-error-loader.js\):$/
);
2018-02-22 01:01:53 +08:00
}
2018-03-02 17:56:34 +08:00
),
getErrorsPromise(
{
mode: "development",
entry: "./entry-point.js",
module: {
rules: [
{
test: /entry-point\.js$/,
use: path.resolve(base, "./throw-error-loader")
}
]
}
},
(errors, warnings) => {
2018-05-01 22:39:05 +08:00
expect(errors).toHaveLength(1);
expect(errors[0].split("\n")[1]).toMatch(
/^Module build failed \(from .\/throw-error-loader.js\):$/
);
2018-02-22 01:01:53 +08:00
}
2018-03-02 17:56:34 +08:00
),
getErrorsPromise(
{
mode: "development",
entry: "./entry-point.js",
module: {
rules: [
{
test: /entry-point\.js$/,
use: path.resolve(base, "./irregular-error-loader")
}
]
}
},
(errors, warnings) => {
2018-05-01 22:39:05 +08:00
expect(warnings).toHaveLength(2);
expect(warnings[0].split("\n")[1]).toMatch(
/^Module Warning \(from .\/irregular-error-loader.js\):$/
);
expect(warnings[1].split("\n")[1]).toMatch(
/^Module Warning \(from .\/irregular-error-loader.js\):$/
);
2018-02-22 01:01:53 +08:00
2018-05-01 22:39:05 +08:00
expect(errors).toHaveLength(3);
expect(errors[0].split("\n")[1]).toMatch(
/^Module Error \(from .\/irregular-error-loader.js\):$/
);
expect(errors[1].split("\n")[1]).toMatch(
/^Module Error \(from .\/irregular-error-loader.js\):$/
);
expect(errors[2].split("\n")[1]).toMatch(
/^Module build failed \(from .\/irregular-error-loader.js\):$/
);
}
2018-03-02 17:56:34 +08:00
)
2018-02-22 01:01:53 +08:00
]);
2018-02-21 17:43:02 +08:00
});
2018-04-29 11:59:50 +08:00
it("should throw a build error if no source be returned after run loaders", done => {
getErrors(
{
mode: "development",
entry: path.resolve(base, "./no-return-loader") + "!./entry-point.js"
},
(errors, warnings) => {
2018-05-01 22:39:05 +08:00
expect(errors).toHaveLength(1);
const messages = errors[0].split("\n");
2018-06-05 22:50:37 +08:00
expect(messages[1]).toMatch(
/^Module build failed: Error: Final loader \(.+\) didn't return a Buffer or String/
);
2018-04-29 11:59:50 +08:00
done();
}
);
});
});