2017-02-02 19:56:41 +08:00
/* globals describe, it */
2017-01-07 10:36:33 +08:00
"use strict" ;
2017-02-02 19:56:41 +08:00
require ( "should" ) ;
2017-11-03 06:28:34 +08:00
2017-01-07 10:36:33 +08:00
const webpack = require ( "../lib/webpack" ) ;
2016-09-19 06:54:35 +08:00
2017-01-18 23:29:47 +08:00
describe ( "Validation" , ( ) => {
2017-01-07 10:36:33 +08:00
const testCases = [ {
2016-09-19 06:54:35 +08:00
name : "undefined configuration" ,
config : undefined ,
message : [
" - configuration should be an object."
]
} , {
name : "null configuration" ,
config : null ,
message : [
" - configuration should be an object."
]
} , {
name : "empty configuration" ,
config : { } ,
message : [
" - configuration misses the property 'entry'." ,
2017-01-09 15:31:49 +08:00
" object { <key>: non-empty string | [non-empty string] } | non-empty string | [non-empty string] | function" ,
2017-11-16 15:59:22 +08:00
" -> The entry point(s) of the compilation."
2016-09-19 06:54:35 +08:00
]
} , {
name : "empty entry string" ,
config : {
entry : ""
} ,
message : [
" - configuration.entry should be one of these:" ,
2017-01-09 15:31:49 +08:00
" object { <key>: non-empty string | [non-empty string] } | non-empty string | [non-empty string] | function" ,
2017-11-16 15:59:22 +08:00
" -> The entry point(s) of the compilation." ,
2016-12-14 18:34:31 +08:00
" Details:" ,
" * configuration.entry should be an object." ,
2017-11-16 15:59:22 +08:00
" -> Multiple entry bundles are created. The key is the chunk name. The value can be a string or an array." ,
2016-12-14 18:34:31 +08:00
" * configuration.entry should not be empty." ,
2017-11-16 15:59:22 +08:00
" -> An entry point without name. The string is resolved to a module which is loaded upon startup." ,
2016-12-14 18:34:31 +08:00
" * configuration.entry should be an array:" ,
2017-01-09 15:31:49 +08:00
" [non-empty string]" ,
" * configuration.entry should be an instance of function" ,
2017-11-16 15:59:22 +08:00
" -> A Function returning an entry object, an entry string, an entry array or a promise to these things."
2016-09-19 06:54:35 +08:00
]
2017-01-04 13:23:57 +08:00
} , {
name : "empty entry bundle array" ,
config : {
entry : {
"bundle" : [ ]
}
} ,
message : [
" - configuration.entry should be one of these:" ,
2017-01-09 15:31:49 +08:00
" object { <key>: non-empty string | [non-empty string] } | non-empty string | [non-empty string] | function" ,
2017-11-16 15:59:22 +08:00
" -> The entry point(s) of the compilation." ,
2017-01-04 13:23:57 +08:00
" Details:" ,
" * configuration.entry['bundle'] should be a string." ,
2017-11-16 15:59:22 +08:00
" -> The string is resolved to a module which is loaded upon startup." ,
2017-01-04 13:23:57 +08:00
" * configuration.entry['bundle'] should not be empty." ,
" * configuration.entry should be a string." ,
2017-11-16 15:59:22 +08:00
" -> An entry point without name. The string is resolved to a module which is loaded upon startup." ,
2017-01-04 13:23:57 +08:00
" * configuration.entry should be an array:" ,
2017-01-09 15:31:49 +08:00
" [non-empty string]" ,
" * configuration.entry should be an instance of function" ,
2017-11-16 15:59:22 +08:00
" -> A Function returning an entry object, an entry string, an entry array or a promise to these things."
2017-01-04 13:23:57 +08:00
]
2016-11-04 06:52:59 +08:00
} , {
name : "invalid instanceof" ,
config : {
entry : "a" ,
module : {
2016-11-04 07:31:49 +08:00
wrappedContextRegExp : 1337
2016-11-04 06:52:59 +08:00
}
} ,
message : [
2017-11-16 15:59:22 +08:00
" - configuration.module.wrappedContextRegExp should be an instance of RegExp" ,
" -> Set the inner regular expression for partial dynamic dependencies"
]
} , {
name : "invalid minimum" ,
config : {
entry : "a" ,
parallelism : 0
} ,
message : [
" - configuration.parallelism should be >= 1." ,
" -> The number of parallel processed modules in the compilation."
]
} , {
name : "repeated value" ,
config : {
entry : [ "abc" , "def" , "abc" ]
} ,
message : [
" - configuration.entry should be one of these:" ,
" object { <key>: non-empty string | [non-empty string] } | non-empty string | [non-empty string] | function" ,
" -> The entry point(s) of the compilation." ,
" Details:" ,
" * configuration.entry should be an object." ,
" -> Multiple entry bundles are created. The key is the chunk name. The value can be a string or an array." ,
" * configuration.entry should be a string." ,
" -> An entry point without name. The string is resolved to a module which is loaded upon startup." ,
" * configuration.entry should not contain the item 'abc' twice." ,
" * configuration.entry should be an instance of function" ,
" -> A Function returning an entry object, an entry string, an entry array or a promise to these things." ,
2016-11-04 06:52:59 +08:00
]
2016-09-19 06:54:35 +08:00
} , {
name : "multiple errors" ,
config : {
entry : [ /a/ ] ,
output : {
filename : /a/
}
} ,
message : [
" - configuration.entry should be one of these:" ,
2017-01-09 15:31:49 +08:00
" object { <key>: non-empty string | [non-empty string] } | non-empty string | [non-empty string] | function" ,
2017-11-16 15:59:22 +08:00
" -> The entry point(s) of the compilation." ,
2016-12-14 18:34:31 +08:00
" Details:" ,
" * configuration.entry should be an object." ,
2017-11-16 15:59:22 +08:00
" -> Multiple entry bundles are created. The key is the chunk name. The value can be a string or an array." ,
2016-12-14 18:34:31 +08:00
" * configuration.entry should be a string." ,
2017-11-16 15:59:22 +08:00
" -> An entry point without name. The string is resolved to a module which is loaded upon startup." ,
2016-12-14 18:34:31 +08:00
" * configuration.entry[0] should be a string." ,
2017-11-16 15:59:22 +08:00
" -> A non-empty string" ,
2017-01-09 15:31:49 +08:00
" * configuration.entry should be an instance of function" ,
2017-11-16 15:59:22 +08:00
" -> A Function returning an entry object, an entry string, an entry array or a promise to these things." ,
" - configuration.output.filename should be a string." ,
" -> Specifies the name of each output file on disk. You must **not** specify an absolute path here! The `output.path` option determines the location on disk the files are written to, filename is used solely for naming the individual files."
2016-09-19 06:54:35 +08:00
]
} , {
name : "multiple configurations" ,
config : [ {
entry : [ /a/ ] ,
} , {
entry : "a" ,
output : {
filename : /a/
}
} ] ,
message : [
" - configuration[0].entry should be one of these:" ,
2017-01-09 15:31:49 +08:00
" object { <key>: non-empty string | [non-empty string] } | non-empty string | [non-empty string] | function" ,
2017-11-16 15:59:22 +08:00
" -> The entry point(s) of the compilation." ,
2016-12-14 18:34:31 +08:00
" Details:" ,
" * configuration[0].entry should be an object." ,
2017-11-16 15:59:22 +08:00
" -> Multiple entry bundles are created. The key is the chunk name. The value can be a string or an array." ,
2016-12-14 18:34:31 +08:00
" * configuration[0].entry should be a string." ,
2017-11-16 15:59:22 +08:00
" -> An entry point without name. The string is resolved to a module which is loaded upon startup." ,
2016-12-14 18:34:31 +08:00
" * configuration[0].entry[0] should be a string." ,
2017-11-16 15:59:22 +08:00
" -> A non-empty string" ,
2017-01-09 15:31:49 +08:00
" * configuration[0].entry should be an instance of function" ,
2017-11-16 15:59:22 +08:00
" -> A Function returning an entry object, an entry string, an entry array or a promise to these things." ,
" - configuration[1].output.filename should be a string." ,
" -> Specifies the name of each output file on disk. You must **not** specify an absolute path here! The `output.path` option determines the location on disk the files are written to, filename is used solely for naming the individual files."
2016-09-19 06:54:35 +08:00
]
} , {
name : "deep error" ,
config : {
entry : "a" ,
module : {
rules : [ {
oneOf : [ {
2017-02-02 19:36:15 +08:00
test : "/a" ,
2016-09-19 06:54:35 +08:00
paser : {
amd : false
}
} ]
} ]
}
} ,
message : [
" - configuration.module.rules[0].oneOf[0] has an unknown property 'paser'. These properties are valid:" ,
2017-11-16 15:59:22 +08:00
" object { enforce?, exclude?, include?, issuer?, loader?, loaders?, oneOf?, options?, parser?, query?, resource?, resourceQuery?, compiler?, rules?, test?, use? }" ,
" -> A rule"
2016-09-19 06:54:35 +08:00
]
2016-09-21 02:18:52 +08:00
} , {
name : "additional key on root" ,
config : {
entry : "a" ,
2017-01-18 23:29:47 +08:00
postcss : ( ) => { }
2016-09-21 02:18:52 +08:00
} ,
message : [
" - configuration has an unknown property 'postcss'. These properties are valid:" ,
2016-09-21 02:29:07 +08:00
" object { amd?, bail?, cache?, context?, dependencies?, devServer?, devtool?, entry, externals?, " +
2017-08-11 22:22:46 +08:00
"loader?, module?, name?, node?, output?, parallelism?, performance?, plugins?, profile?, recordsInputPath?, " +
"recordsOutputPath?, recordsPath?, resolve?, resolveLoader?, stats?, target?, watch?, watchOptions? }" ,
2016-09-21 02:18:52 +08:00
" For typos: please correct them." ,
" For loader options: webpack 2 no longer allows custom properties in configuration." ,
" Loaders should be updated to allow passing options via loader options in module.rules." ,
" Until loaders are updated one can use the LoaderOptionsPlugin to pass these options to the loader:" ,
2016-09-28 18:54:46 +08:00
" plugins: [" ,
2016-09-21 02:18:52 +08:00
" new webpack.LoaderOptionsPlugin({" ,
2016-09-21 23:44:57 +08:00
" // test: /\\.xxx$/, // may apply this only for some modules" ,
2016-09-21 02:18:52 +08:00
" options: {" ,
" postcss: ..." ,
" }" ,
" })" ,
2016-09-28 18:54:46 +08:00
" ]"
2016-09-21 02:18:52 +08:00
]
2016-12-14 19:03:24 +08:00
} , {
name : "enum" ,
config : {
entry : "a" ,
devtool : true
} ,
message : [
" - configuration.devtool should be one of these:" ,
" string | false" ,
2017-11-16 15:59:22 +08:00
" -> A developer tool to enhance debugging." ,
2016-12-14 19:03:24 +08:00
" Details:" ,
" * configuration.devtool should be a string." ,
" * configuration.devtool should be false"
]
2017-02-03 21:49:38 +08:00
} , {
2017-02-10 04:47:46 +08:00
name : "relative path" ,
2017-02-03 21:49:38 +08:00
config : {
entry : "foo.js" ,
output : {
filename : "/bar"
}
} ,
message : [
2017-02-10 04:47:46 +08:00
" - configuration.output.filename: A relative path is expected. However the provided value \"/bar\" is an absolute path!" ,
2017-11-16 15:59:22 +08:00
" -> Specifies the name of each output file on disk. You must **not** specify an absolute path here! The `output.path` option determines the location on disk the files are written to, filename is used solely for naming the individual files." ,
2017-03-25 05:21:30 +08:00
" Please use output.path to specify absolute path and output.filename for the file name."
2017-02-03 21:49:38 +08:00
]
} , {
name : "absolute path" ,
config : {
entry : "foo.js" ,
output : {
filename : "bar"
} ,
context : "baz"
} ,
message : [
2017-02-08 19:54:55 +08:00
" - configuration.context: The provided value \"baz\" is not an absolute path!" ,
2017-11-16 15:59:22 +08:00
" -> The base directory (absolute path!) for resolving the `entry` option. If `output.pathinfo` is set, the included pathinfo is shortened to this directory."
2017-02-03 21:49:38 +08:00
]
2017-10-14 06:12:24 +08:00
} , {
name : "missing stats option" ,
config : {
entry : "foo.js" ,
stats : {
foobar : true
}
} ,
2017-11-03 06:28:34 +08:00
test ( err ) {
err . message . should . startWith ( "Invalid configuration object." ) ;
err . message . split ( "\n" ) . slice ( 1 ) [ 0 ] . should . be . eql (
2017-10-14 06:12:24 +08:00
" - configuration.stats should be one of these:"
) ;
}
2016-09-19 06:54:35 +08:00
} ] ;
2017-11-03 06:28:34 +08:00
2017-01-18 23:29:47 +08:00
testCases . forEach ( ( testCase ) => {
it ( "should fail validation for " + testCase . name , ( ) => {
2016-09-19 06:54:35 +08:00
try {
webpack ( testCase . config ) ;
2017-11-03 06:28:34 +08:00
} catch ( err ) {
if ( err . name !== 'WebpackOptionsValidationError' ) throw err ;
2017-10-14 06:12:24 +08:00
if ( testCase . test ) {
2017-11-03 06:28:34 +08:00
testCase . test ( err ) ;
2017-10-14 06:12:24 +08:00
return ;
}
2017-11-03 06:28:34 +08:00
err . message . should . startWith ( "Invalid configuration object." ) ;
err . message . split ( "\n" ) . slice ( 1 ) . should . be . eql ( testCase . message ) ;
2016-09-19 06:54:35 +08:00
return ;
}
2017-11-03 06:28:34 +08:00
2016-09-19 06:54:35 +08:00
throw new Error ( "Validation didn't fail" ) ;
2017-02-02 19:56:41 +08:00
} ) ;
2016-09-19 06:54:35 +08:00
} ) ;
} ) ;