2020-11-17 05:58:06 +01:00
|
|
|
#!/usr/bin/env -S node --trace-warnings
|
|
|
|
|
2020-11-17 18:38:48 +01:00
|
|
|
const fs = require('fs');
|
2020-11-17 05:58:06 +01:00
|
|
|
const esbuild = require('esbuild');
|
|
|
|
const log = require('@vladmandic/pilogger');
|
|
|
|
|
|
|
|
// keeps esbuild service instance cached
|
|
|
|
let es;
|
|
|
|
|
|
|
|
// common configuration
|
|
|
|
const common = {
|
|
|
|
minifyWhitespace: true,
|
2020-11-17 23:42:44 +01:00
|
|
|
minifySyntax: true,
|
2020-11-17 05:58:06 +01:00
|
|
|
bundle: true,
|
|
|
|
sourcemap: true,
|
|
|
|
logLevel: 'error',
|
|
|
|
target: 'es2018',
|
|
|
|
};
|
|
|
|
|
2020-11-17 18:38:48 +01:00
|
|
|
const tfjs = {
|
|
|
|
platform: 'browser',
|
|
|
|
format: 'esm',
|
|
|
|
metafile: 'dist/tfjs.esm.json',
|
|
|
|
entryPoints: ['src/tf.js'],
|
|
|
|
outfile: 'dist/tfjs.esm.js',
|
|
|
|
external: ['fs', 'buffer', 'util'],
|
|
|
|
};
|
|
|
|
|
2020-11-17 05:58:06 +01:00
|
|
|
// all build targets
|
|
|
|
const config = {
|
|
|
|
iifeBundle: {
|
|
|
|
platform: 'browser',
|
|
|
|
format: 'iife',
|
|
|
|
globalName: 'Human',
|
|
|
|
metafile: 'dist/human.json',
|
|
|
|
entryPoints: ['src/human.js'],
|
|
|
|
outfile: 'dist/human.js',
|
|
|
|
external: ['fs', 'buffer', 'util'],
|
|
|
|
},
|
|
|
|
esmBundle: {
|
|
|
|
platform: 'browser',
|
|
|
|
format: 'esm',
|
|
|
|
metafile: 'dist/human.esm.json',
|
|
|
|
entryPoints: ['src/human.js'],
|
|
|
|
outfile: 'dist/human.esm.js',
|
|
|
|
external: ['fs', 'buffer', 'util'],
|
|
|
|
},
|
|
|
|
esmNoBundle: {
|
|
|
|
platform: 'browser',
|
|
|
|
format: 'esm',
|
|
|
|
metafile: 'dist/human.esm-nobundle.json',
|
|
|
|
entryPoints: ['src/human.js'],
|
|
|
|
outfile: 'dist/human.esm-nobundle.js',
|
|
|
|
external: ['fs', 'buffer', 'util', '@tensorflow'],
|
|
|
|
},
|
|
|
|
nodeBundle: {
|
|
|
|
platform: 'node',
|
|
|
|
format: 'cjs',
|
|
|
|
metafile: 'dist/human.node.json',
|
|
|
|
entryPoints: ['src/human.js'],
|
|
|
|
outfile: 'dist/human.node.js',
|
|
|
|
},
|
|
|
|
nodeNoBundle: {
|
|
|
|
platform: 'node',
|
|
|
|
format: 'cjs',
|
|
|
|
metafile: 'dist/human.node-nobundle.json',
|
|
|
|
entryPoints: ['src/human.js'],
|
|
|
|
outfile: 'dist/human.node-nobundle.js',
|
|
|
|
external: ['@tensorflow'],
|
|
|
|
},
|
|
|
|
demo: {
|
|
|
|
platform: 'browser',
|
|
|
|
format: 'esm',
|
|
|
|
metafile: 'dist/demo-browser-index.json',
|
|
|
|
entryPoints: ['demo/browser.js'],
|
|
|
|
outfile: 'dist/demo-browser-index.js',
|
|
|
|
external: ['fs', 'buffer', 'util'],
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2020-11-17 18:38:48 +01:00
|
|
|
async function getStats(metafile) {
|
|
|
|
const stats = {};
|
|
|
|
if (!fs.existsSync(metafile)) return stats;
|
|
|
|
const data = fs.readFileSync(metafile);
|
|
|
|
const json = JSON.parse(data);
|
|
|
|
if (json && json.inputs && json.outputs) {
|
|
|
|
for (const [key, val] of Object.entries(json.inputs)) {
|
|
|
|
if (key.startsWith('node_modules')) {
|
|
|
|
stats.modules = (stats.modules || 0) + 1;
|
|
|
|
stats.moduleBytes = (stats.moduleBytes || 0) + val.bytes;
|
|
|
|
} else {
|
|
|
|
stats.imports = (stats.imports || 0) + 1;
|
|
|
|
stats.importBytes = (stats.importBytes || 0) + val.bytes;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const files = [];
|
|
|
|
for (const [key, val] of Object.entries(json.outputs)) {
|
|
|
|
if (!key.endsWith('.map')) {
|
|
|
|
// stats.outputs += 1;
|
|
|
|
files.push(key);
|
|
|
|
stats.outputBytes = (stats.outputBytes || 0) + val.bytes;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
stats.outputFiles = files.join(', ');
|
|
|
|
}
|
|
|
|
return stats;
|
|
|
|
}
|
|
|
|
|
2020-11-17 05:58:06 +01:00
|
|
|
// rebuild on file change
|
|
|
|
async function build(f, msg) {
|
2020-11-17 18:38:48 +01:00
|
|
|
log.info('Build: file', msg, f, 'target:', common.target);
|
2020-11-17 05:58:06 +01:00
|
|
|
if (!es) es = await esbuild.startService();
|
|
|
|
// common build options
|
|
|
|
try {
|
2020-11-17 18:38:48 +01:00
|
|
|
// rebuild tfjs
|
|
|
|
if (f.endsWith('tf.js') || !module.parent) {
|
|
|
|
await es.build({ ...common, ...tfjs });
|
|
|
|
const stats = await getStats(tfjs.metafile);
|
|
|
|
log.state('Build:', stats);
|
|
|
|
}
|
2020-11-17 05:58:06 +01:00
|
|
|
// rebuild all targets
|
|
|
|
for (const [target, options] of Object.entries(config)) {
|
|
|
|
await es.build({ ...common, ...options });
|
2020-11-17 18:38:48 +01:00
|
|
|
const stats = await getStats(options.metafile, target);
|
|
|
|
log.state('Build:', stats);
|
2020-11-17 05:58:06 +01:00
|
|
|
}
|
|
|
|
if (!module.parent) process.exit(0);
|
|
|
|
} catch (err) {
|
|
|
|
// catch errors and print where it occured
|
|
|
|
log.error('Build error', JSON.stringify(err.errors || err, null, 2));
|
|
|
|
if (!module.parent) process.exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!module.parent) {
|
|
|
|
log.header();
|
|
|
|
build('all', 'startup');
|
|
|
|
} else {
|
|
|
|
exports.build = build;
|
|
|
|
}
|