nx/packages/web/src/utils/normalize.ts

130 lines
3.7 KiB
TypeScript

import { WebBuildBuilderOptions } from '../builders/build/build.impl';
import { normalize } from '@angular-devkit/core';
import { resolve, dirname, relative, basename } from 'path';
import { BuildBuilderOptions } from './types';
import { statSync } from 'fs';
import { ScriptTarget } from 'typescript';
export interface FileReplacement {
replace: string;
with: string;
}
export function normalizeBuildOptions<T extends BuildBuilderOptions>(
options: T,
root: string,
sourceRoot: string
): T {
return {
...options,
main: resolve(root, options.main),
outputPath: resolve(root, options.outputPath),
tsConfig: resolve(root, options.tsConfig),
fileReplacements: normalizeFileReplacements(root, options.fileReplacements),
assets: normalizeAssets(options.assets, root, sourceRoot),
webpackConfig: options.webpackConfig
? resolve(root, options.webpackConfig)
: options.webpackConfig
};
}
function normalizeAssets(
assets: any[],
root: string,
sourceRoot: string
): any[] {
return assets.map(asset => {
if (typeof asset === 'string') {
const assetPath = normalize(asset);
const resolvedAssetPath = resolve(root, assetPath);
const resolvedSourceRoot = resolve(root, sourceRoot);
if (!resolvedAssetPath.startsWith(resolvedSourceRoot)) {
throw new Error(
`The ${resolvedAssetPath} asset path must start with the project source root: ${sourceRoot}`
);
}
const isDirectory = statSync(resolvedAssetPath).isDirectory();
const input = isDirectory
? resolvedAssetPath
: dirname(resolvedAssetPath);
const output = relative(resolvedSourceRoot, resolve(root, input));
const glob = isDirectory ? '**/*' : basename(resolvedAssetPath);
return {
input,
output,
glob
};
} else {
if (asset.output.startsWith('..')) {
throw new Error(
'An asset cannot be written to a location outside of the output path.'
);
}
return {
...asset,
// Now we remove starting slash to make Webpack place it from the output root.
output: asset.output.replace(/^\//, '')
};
}
});
}
function normalizeFileReplacements(
root: string,
fileReplacements: FileReplacement[]
): FileReplacement[] {
return fileReplacements.map(fileReplacement => ({
replace: resolve(root, fileReplacement.replace),
with: resolve(root, fileReplacement.with)
}));
}
export function normalizeWebBuildOptions(
options: WebBuildBuilderOptions,
root: string,
sourceRoot: string
): WebBuildBuilderOptions {
return {
...normalizeBuildOptions(options, root, sourceRoot),
optimization:
typeof options.optimization !== 'object'
? {
scripts: options.optimization,
styles: options.optimization
}
: options.optimization,
sourceMap:
typeof options.sourceMap === 'object'
? options.sourceMap
: {
scripts: options.sourceMap,
styles: options.sourceMap,
hidden: false,
vendors: false
},
polyfills: options.polyfills ? resolve(root, options.polyfills) : undefined,
es2015Polyfills: options.es2015Polyfills
? resolve(root, options.es2015Polyfills)
: undefined
};
}
export function convertBuildOptions(
buildOptions: WebBuildBuilderOptions,
scriptTargetOverride: ScriptTarget
): any {
const options = buildOptions as any;
return <any>{
...options,
buildOptimizer: options.optimization,
aot: false,
scriptTargetOverride: scriptTargetOverride,
esVersionInFileName: !!scriptTargetOverride,
forkTypeChecker: false,
lazyModules: [] as string[],
assets: [] as string[]
};
}