feat: parse --env.<flags> [WIP]

This commit is contained in:
Igor Randjelovic
2021-03-02 18:50:10 +01:00
parent 60293bb819
commit 0a1ba16436
4 changed files with 150 additions and 1 deletions

View File

@ -0,0 +1,47 @@
import { parseEnvFlags } from '../../src/cli/parseEnvFlags';
describe.only('parseEnvFlags', () => {
it('parses all possible flags', () => {
const res = parseEnvFlags([
'--env', // invalid
'--env.foo',
'--env.externals=ext1',
'--env.externals=ext2',
'--env.externals=ext3',
'--env.externals=ext4',
'--env.externals=ext4',
'--env.externals=/path/to/a/very/long/path with spaces/foo.js',
'--env.externals=~/package.json',
'--env.externals=package.json',
'--env.ios=false',
'--env.android',
'--env.verbose',
'--env.sourceMap',
'--env.appPath=app',
'--env.appResourcesPath=App_Resources',
'--env.num=5',
'--env.float=5.4',
'--env.numArray=3',
'--env.numArray=4',
'--env.numArray=5',
'--no-hmr',
'--not-env-flag',
]);
expect(res).toBeDefined();
expect(res.foo).toBe(true);
expect(res.externals).toBeInstanceOf(Array);
expect(res.externals.length).toBe(8);
expect(res.ios).toBe(false);
expect(res.android).toBe(true);
expect(res.verbose).toBe(true);
expect(res.sourceMap).toBe(true);
expect(res.sourceMap).toBe(true);
expect(res.appPath).toBe('app');
expect(res.appResourcesPath).toBe('App_Resources');
expect(res.num).toBe(5);
expect(res.float).toBe(5.4);
expect(res.numArray).toStrictEqual([3, 4, 5]);
expect(Object.keys(res).length).toBe(11);
});
});

View File

@ -3,8 +3,10 @@
import { redBright, green, greenBright } from 'chalk'; import { redBright, green, greenBright } from 'chalk';
import { program } from 'commander'; import { program } from 'commander';
import dedent from 'ts-dedent'; import dedent from 'ts-dedent';
import webpack from 'webpack';
import path from 'path'; import path from 'path';
import fs from 'fs'; import fs from 'fs';
import { parseEnvFlags } from '../cli/parseEnvFlags';
const defaultConfig = path.resolve( const defaultConfig = path.resolve(
__dirname, __dirname,
@ -20,6 +22,8 @@ function info(message: string) {
console.info(`${tag} ${greenBright(dedent(message))}`); console.info(`${tag} ${greenBright(dedent(message))}`);
} }
program.enablePositionalOptions();
program program
.command('init') .command('init')
.description('Initialize a new webpack.config.js in the current directory.') .description('Initialize a new webpack.config.js in the current directory.')
@ -35,4 +39,63 @@ program
info('Initialized config.'); info('Initialized config.');
}); });
program
.command('build')
.description('Build...')
.option('--env [name]', 'environment options')
.option('--hmr', 'enable HMR')
.option('--no-hmr', 'disable HMR')
.option('--watch', 'watch for changes')
.allowUnknownOption()
.action((options, command) => {
const env = parseEnvFlags(command.args);
// add --env <val> into the env object
// for example if we use --env prod
// we'd have env.env = 'prod'
if (options.env) {
env['env'] = options.env;
}
// const env = {
// platform: 'ios',
// verbose: true,
// appResourcesPath: 'App_Resources',
// appPath: 'src'
// }
const configPath = path.resolve(process.cwd(), 'webpack.config.js');
// todo: validate config exists
// todo: guard against invalid config
let configuration;
try {
configuration = require(configPath)(env);
} catch (ignore) {
console.log(ignore);
}
if (!configuration) {
console.log('No configuration!!!!!');
return;
}
const compiler = webpack(configuration);
// todo: handle --watch flag
// todo: promisify callback?
compiler.watch(
{
ignored: ['platforms'],
},
(err, stats) => {
if (stats) {
console.log(
stats.toString({
colors: true,
})
);
}
// err && console.log(err)
}
);
});
program.parse(process.argv); program.parse(process.argv);

View File

@ -0,0 +1,39 @@
import { IWebpackEnv } from '@nativescript/webpack';
const ENV_FLAG_RE = /--env\.(\w+)(?:=(.+))?/;
export function parseEnvFlags(flags: string[]): IWebpackEnv {
const envFlags = flags.filter((flag) => flag.includes('--env.'));
const env = {};
envFlags.map((flag) => {
let [_, name, v] = ENV_FLAG_RE.exec(flag);
let value: any = v;
// convert --env.foo to --env.foo=true
if (value === undefined) {
value = true;
}
// convert true/false to boolean
if (value === 'true' || value === 'false') {
value = value === 'true';
}
// convert numbers
if (!isNaN(value) && !isNaN(parseFloat(value))) {
value = +value;
}
// duplicate key/name - convert to array
if (name in env && value) {
const orig = Array.isArray(env[name]) ? env[name] : [env[name]];
env[name] = [...orig, value];
} else {
env[name] = value;
}
});
return env;
}

View File

@ -70,7 +70,7 @@ export function getPlatformName(): Platform {
Available platforms: ${Object.keys(platforms).join(', ')} Available platforms: ${Object.keys(platforms).join(', ')}
Use --env=platform=<platform> or --env=android, --env=ios to specify the target platform. Use --env.platform=<platform> or --env.android, --env.ios to specify the target platform.
`); `);
} }