-
Notifications
You must be signed in to change notification settings - Fork 12k
/
Copy pathloader.ts
83 lines (70 loc) · 2.51 KB
/
loader.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
/**
* @license
* Copyright Google LLC All Rights Reserved.
*
* Use of this source code is governed by an MIT-style license that can be
* found in the LICENSE file at https://angular.dev/license
*/
import * as path from 'path';
import type { LoaderContext } from 'webpack';
import { AngularPluginSymbol, FileEmitterCollection } from './symbol';
const JS_FILE_REGEXP = /\.[cm]?js$/;
export function angularWebpackLoader(
this: LoaderContext<unknown>,
content: string,
map: string,
): void {
const callback = this.async();
if (!callback) {
throw new Error('Invalid webpack version');
}
const fileEmitter = (
this as LoaderContext<unknown> & { [AngularPluginSymbol]?: FileEmitterCollection }
)[AngularPluginSymbol];
if (!fileEmitter || typeof fileEmitter !== 'object') {
if (JS_FILE_REGEXP.test(this.resourcePath)) {
// Passthrough for JS files when no plugin is used
this.callback(undefined, content, map);
return;
}
callback(new Error('The Angular Webpack loader requires the AngularWebpackPlugin.'));
return;
}
fileEmitter
.emit(this.resourcePath)
.then((result) => {
if (!result) {
if (JS_FILE_REGEXP.test(this.resourcePath)) {
// Return original content for JS files if not compiled by TypeScript ("allowJs")
this.callback(undefined, content, map);
} else {
// File is not part of the compilation
const message =
`${this.resourcePath} is missing from the TypeScript compilation. ` +
`Please make sure it is in your tsconfig via the 'files' or 'include' property.`;
callback(new Error(message));
}
return;
}
result.dependencies.forEach((dependency) => this.addDependency(dependency));
let resultContent = result.content || '';
let resultMap;
if (result.map) {
resultContent = resultContent.replace(/^\/\/# sourceMappingURL=[^\r\n]*/gm, '');
resultMap = JSON.parse(result.map) as Exclude<
Parameters<typeof callback>[2],
string | undefined
>;
resultMap.sources = resultMap.sources.map((source: string) =>
path.join(path.dirname(this.resourcePath), source),
);
}
callback(undefined, resultContent, resultMap);
})
.catch((err) => {
// The below is needed to hide stacktraces from users.
const message = err instanceof Error ? err.message : err;
callback(new Error(message));
});
}
export { angularWebpackLoader as default };