@@ -4,10 +4,20 @@ import { dirname, join, relative } from 'path';
4
4
import {
5
5
getComponentsDtsTypesFilePath ,
6
6
isOutputTargetDistCollection ,
7
+ isOutputTargetDistCustomElements ,
7
8
isOutputTargetDistCustomElementsBundle ,
8
9
isOutputTargetDistTypes ,
9
10
} from '../output-targets/output-utils' ;
10
11
12
+ /**
13
+ * Validate the package.json file for a project, checking that various fields
14
+ * are set correctly for the currently-configured output targets.
15
+ *
16
+ * @param config the user-supplied Stencil config
17
+ * @param compilerCtx the compiler context
18
+ * @param buildCtx the build context
19
+ * @returns an empty Promise
20
+ */
11
21
export const validateBuildPackageJson = async ( config : d . Config , compilerCtx : d . CompilerCtx , buildCtx : d . BuildCtx ) => {
12
22
if ( config . watch ) {
13
23
return ;
@@ -16,19 +26,28 @@ export const validateBuildPackageJson = async (config: d.Config, compilerCtx: d.
16
26
return ;
17
27
}
18
28
19
- const outputTargets = config . outputTargets . filter ( isOutputTargetDistCollection ) ;
29
+ const distCollectionOutputTargets = config . outputTargets . filter ( isOutputTargetDistCollection ) ;
20
30
const typesOutputTargets = config . outputTargets . filter ( isOutputTargetDistTypes ) ;
21
31
await Promise . all ( [
22
- ...outputTargets . map ( ( outputsTarget ) => {
23
- return validatePackageJsonOutput ( config , compilerCtx , buildCtx , outputsTarget ) ;
24
- } ) ,
25
- ...typesOutputTargets . map ( ( outputTarget ) => {
26
- return validateTypes ( config , compilerCtx , buildCtx , outputTarget ) ;
27
- } ) ,
32
+ ...distCollectionOutputTargets . map ( ( distCollectionOT ) =>
33
+ validateDistCollectionPkgJson ( config , compilerCtx , buildCtx , distCollectionOT )
34
+ ) ,
35
+ ...typesOutputTargets . map ( ( typesOT ) => validateTypes ( config , compilerCtx , buildCtx , typesOT ) ) ,
36
+ validateModule ( config , compilerCtx , buildCtx ) ,
28
37
] ) ;
29
38
} ;
30
39
31
- const validatePackageJsonOutput = async (
40
+ /**
41
+ * Validate package.json contents for the `DIST_COLLECTION` output target,
42
+ * checking that various fields like `files`, `main`, and so on are set
43
+ * correctly.
44
+ *
45
+ * @param config the stencil config
46
+ * @param compilerCtx the current compiler context
47
+ * @param buildCtx the current build context
48
+ * @param outputTarget a DIST_COLLECTION output target
49
+ */
50
+ const validateDistCollectionPkgJson = async (
32
51
config : d . Config ,
33
52
compilerCtx : d . CompilerCtx ,
34
53
buildCtx : d . BuildCtx ,
@@ -37,12 +56,20 @@ const validatePackageJsonOutput = async (
37
56
await Promise . all ( [
38
57
validatePackageFiles ( config , compilerCtx , buildCtx , outputTarget ) ,
39
58
validateMain ( config , compilerCtx , buildCtx , outputTarget ) ,
40
- validateModule ( config , compilerCtx , buildCtx , outputTarget ) ,
41
59
validateCollection ( config , compilerCtx , buildCtx , outputTarget ) ,
42
60
validateBrowser ( config , compilerCtx , buildCtx ) ,
43
61
] ) ;
44
62
} ;
45
63
64
+ /**
65
+ * Validate that the `files` field in `package.json` contains directories and
66
+ * files that are necessary for the `DIST_COLLECTION` output target.
67
+ *
68
+ * @param config the stencil config
69
+ * @param compilerCtx the current compiler context
70
+ * @param buildCtx the current build context
71
+ * @param outputTarget a DIST_COLLECTION output target
72
+ */
46
73
export const validatePackageFiles = async (
47
74
config : d . Config ,
48
75
compilerCtx : d . CompilerCtx ,
@@ -81,6 +108,15 @@ export const validatePackageFiles = async (
81
108
}
82
109
} ;
83
110
111
+ /**
112
+ * Check that the `main` field is set correctly in `package.json` for the
113
+ * `DIST_COLLECTION` output target.
114
+ *
115
+ * @param config the stencil config
116
+ * @param compilerCtx the current compiler context
117
+ * @param buildCtx the current build context
118
+ * @param outputTarget a DIST_COLLECTION output target
119
+ */
84
120
export const validateMain = (
85
121
config : d . Config ,
86
122
compilerCtx : d . CompilerCtx ,
@@ -99,35 +135,88 @@ export const validateMain = (
99
135
}
100
136
} ;
101
137
102
- export const validateModule = (
103
- config : d . Config ,
104
- compilerCtx : d . CompilerCtx ,
105
- buildCtx : d . BuildCtx ,
106
- outputTarget : d . OutputTargetDistCollection
107
- ) => {
108
- const customElementsOutput = config . outputTargets . find ( isOutputTargetDistCustomElementsBundle ) ;
138
+ /**
139
+ * Validate the package.json 'module' field, taking into account output targets
140
+ * and other configuration details. This will look for a value for the `module`
141
+ * field. If not present it will set a relevant warning message with an
142
+ * output-target specific recommended value. If it is present and is not equal
143
+ * to that recommended value it will set a different warning message.
144
+ *
145
+ * @param config the current user-supplied configuration
146
+ * @param compilerCtx the compiler context
147
+ * @param buildCtx the build context
148
+ * @returns an empty Promise
149
+ */
150
+ export const validateModule = async ( config : d . Config , compilerCtx : d . CompilerCtx , buildCtx : d . BuildCtx ) => {
109
151
const currentModule = buildCtx . packageJson . module ;
110
- const distAbs = join ( outputTarget . dir , 'index.js' ) ;
111
- const distRel = relative ( config . rootDir , distAbs ) ;
112
152
113
- let recommendedRelPath = distRel ;
114
- if ( customElementsOutput ) {
115
- const customElementsAbs = join ( customElementsOutput . dir , 'index.js' ) ;
116
- recommendedRelPath = relative ( config . rootDir , customElementsAbs ) ;
117
- }
153
+ const recommendedRelPath = recommendedModulePath ( config ) ;
118
154
119
155
if ( ! isString ( currentModule ) ) {
120
- const msg = `package.json "module" property is required when generating a distribution. It's recommended to set the "module" property to: ${ recommendedRelPath } ` ;
156
+ let msg = 'package.json "module" property is required when generating a distribution.' ;
157
+
158
+ if ( recommendedRelPath !== null ) {
159
+ msg += ` It's recommended to set the "module" property to: ${ normalizePath ( recommendedRelPath ) } ` ;
160
+ }
121
161
packageJsonWarn ( config , compilerCtx , buildCtx , msg , `"module"` ) ;
122
- } else if (
123
- normalizePath ( currentModule ) !== normalizePath ( recommendedRelPath ) &&
124
- normalizePath ( currentModule ) !== normalizePath ( distRel )
125
- ) {
126
- const msg = `package.json "module" property is set to "${ currentModule } ". It's recommended to set the "module" property to: ${ recommendedRelPath } ` ;
162
+ return ;
163
+ }
164
+
165
+ if ( recommendedRelPath !== null && normalizePath ( recommendedRelPath ) !== normalizePath ( currentModule ) ) {
166
+ const msg = `package.json "module" property is set to "${ currentModule } ". It's recommended to set the "module" property to: ${ normalizePath (
167
+ recommendedRelPath
168
+ ) } `;
127
169
packageJsonWarn ( config , compilerCtx , buildCtx , msg , `"module"` ) ;
128
170
}
129
171
} ;
130
172
173
+ /**
174
+ * Get the recommended `"module"` path for `package.json` given the output
175
+ * targets that a user has set on their config.
176
+ *
177
+ * @param config the user-supplied Stencil configuration
178
+ * @returns a recommended module path or a null value to indicate no default
179
+ * value is supplied
180
+ */
181
+ function recommendedModulePath ( config : d . Config ) : string | null {
182
+ const customElementsBundleOT = config . outputTargets . find ( isOutputTargetDistCustomElementsBundle ) ;
183
+ const customElementsOT = config . outputTargets . find ( isOutputTargetDistCustomElements ) ;
184
+ const distCollectionOT = config . outputTargets . find ( isOutputTargetDistCollection ) ;
185
+
186
+ // If we're using `dist-custom-elements` then the preferred "module" field
187
+ // value is `$OUTPUT_DIR/components/index.js`
188
+ //
189
+ // Additionally, the `DIST_CUSTOM_ELEMENTS` output target should override
190
+ // `DIST_CUSTOM_ELEMENTS_BUNDLE` and `DIST_COLLECTION` output targets if
191
+ // they're also set, so we return first with this one.
192
+ if ( customElementsOT ) {
193
+ const componentsIndexAbs = join ( customElementsOT . dir , 'components' , 'index.js' ) ;
194
+ return relative ( config . rootDir , componentsIndexAbs ) ;
195
+ }
196
+
197
+ if ( customElementsBundleOT ) {
198
+ const customElementsAbs = join ( customElementsBundleOT . dir , 'index.js' ) ;
199
+ return relative ( config . rootDir , customElementsAbs ) ;
200
+ }
201
+
202
+ if ( distCollectionOT ) {
203
+ return relative ( config . rootDir , join ( distCollectionOT . dir , 'index.js' ) ) ;
204
+ }
205
+
206
+ // if no output target for which we define a recommended output target is set
207
+ // we return `null`
208
+ return null ;
209
+ }
210
+
211
+ /**
212
+ * Check that the `types` field is set correctly in `package.json` for the
213
+ * `DIST_COLLECTION` output target.
214
+ *
215
+ * @param config the stencil config
216
+ * @param compilerCtx the current compiler context
217
+ * @param buildCtx the current build context
218
+ * @param outputTarget a DIST_COLLECTION output target
219
+ */
131
220
export const validateTypes = async (
132
221
config : d . Config ,
133
222
compilerCtx : d . CompilerCtx ,
@@ -156,6 +245,15 @@ export const validateTypes = async (
156
245
}
157
246
} ;
158
247
248
+ /**
249
+ * Check that the `collection` field is set correctly in `package.json` for the
250
+ * `DIST_COLLECTION` output target.
251
+ *
252
+ * @param config the stencil config
253
+ * @param compilerCtx the current compiler context
254
+ * @param buildCtx the current build context
255
+ * @param outputTarget a DIST_COLLECTION output target
256
+ */
159
257
export const validateCollection = (
160
258
config : d . Config ,
161
259
compilerCtx : d . CompilerCtx ,
@@ -171,33 +269,65 @@ export const validateCollection = (
171
269
}
172
270
} ;
173
271
272
+ /**
273
+ * Check that the `browser` field is set correctly in `package.json` for the
274
+ * `DIST_COLLECTION` output target.
275
+ *
276
+ * @param config the stencil config
277
+ * @param compilerCtx the current compiler context
278
+ * @param buildCtx the current build context
279
+ */
174
280
export const validateBrowser = ( config : d . Config , compilerCtx : d . CompilerCtx , buildCtx : d . BuildCtx ) => {
175
281
if ( isString ( buildCtx . packageJson . browser ) ) {
176
282
const msg = `package.json "browser" property is set to "${ buildCtx . packageJson . browser } ". However, for maximum compatibility with all bundlers it's recommended to not set the "browser" property and instead ensure both "module" and "main" properties are set.` ;
177
283
packageJsonWarn ( config , compilerCtx , buildCtx , msg , `"browser"` ) ;
178
284
}
179
285
} ;
180
286
287
+ /**
288
+ * Build a diagnostic for an error resulting from a particular field in a
289
+ * package.json file
290
+ *
291
+ * @param config the stencil config
292
+ * @param compilerCtx the current compiler context
293
+ * @param buildCtx the current build context
294
+ * @param msg an error string
295
+ * @param jsonField the key for the field which caused the error, used for
296
+ * finding the error line in the original JSON file
297
+ * @returns a diagnostic object
298
+ */
181
299
const packageJsonError = (
182
300
config : d . Config ,
183
301
compilerCtx : d . CompilerCtx ,
184
302
buildCtx : d . BuildCtx ,
185
303
msg : string ,
186
- warnKey : string
187
- ) => {
188
- const err = buildJsonFileError ( compilerCtx , buildCtx . diagnostics , config . packageJsonFilePath , msg , warnKey ) ;
304
+ jsonField : string
305
+ ) : d . Diagnostic => {
306
+ const err = buildJsonFileError ( compilerCtx , buildCtx . diagnostics , config . packageJsonFilePath , msg , jsonField ) ;
189
307
err . header = `Package Json` ;
190
308
return err ;
191
309
} ;
192
310
311
+ /**
312
+ * Build a diagnostic for a warning resulting from a particular field in a
313
+ * package.json file
314
+ *
315
+ * @param config the stencil config
316
+ * @param compilerCtx the current compiler context
317
+ * @param buildCtx the current build context
318
+ * @param msg an error string
319
+ * @param jsonField the key for the field which caused the error, used for
320
+ * finding the error line in the original JSON file
321
+ * @returns a diagnostic object
322
+ */
193
323
const packageJsonWarn = (
194
324
config : d . Config ,
195
325
compilerCtx : d . CompilerCtx ,
196
326
buildCtx : d . BuildCtx ,
197
327
msg : string ,
198
- warnKey : string
199
- ) => {
200
- const warn = buildJsonFileError ( compilerCtx , buildCtx . diagnostics , config . packageJsonFilePath , msg , warnKey ) ;
328
+ jsonField : string
329
+ ) : d . Diagnostic => {
330
+ const warn = buildJsonFileError ( compilerCtx , buildCtx . diagnostics , config . packageJsonFilePath , msg , jsonField ) ;
201
331
warn . header = `Package Json` ;
202
332
warn . level = 'warn' ;
203
333
return warn ;
0 commit comments