salix/gulpfile.js

567 lines
16 KiB
JavaScript
Raw Normal View History

require('require-yaml');
const gulp = require('gulp');
const exec = require('child_process').exec;
const PluginError = require('plugin-error');
const argv = require('minimist')(process.argv.slice(2));
const log = require('fancy-log');
const request = require('request');
const e2eConfig = require('./e2e/helpers/config.js');
2017-10-18 04:41:17 +00:00
// Configuration
let isWindows = /^win/.test(process.platform);
if (argv.NODE_ENV)
process.env.NODE_ENV = argv.NODE_ENV;
let langs = ['es', 'en'];
let srcDir = './front';
2018-12-27 11:54:16 +00:00
let modulesDir = './modules';
2019-01-25 22:02:29 +00:00
let buildDir = 'dist';
2019-02-11 21:57:48 +00:00
let containerId = 'salix-db';
let dataSources = require('./loopback/server/datasources.json');
let dbConf = dataSources.vn;
2019-02-11 22:01:32 +00:00
if (process.env.DB_HOST)
dbConf.host = process.env.DB_HOST;
2018-02-03 21:53:02 +00:00
let backSources = [
'!node_modules',
'loopback',
'modules/*/back/**',
2019-02-06 15:08:23 +00:00
'modules/*/back/*',
'back',
'print'
];
// Development
2017-05-16 10:37:48 +00:00
const localesRoutes = gulp.parallel(locales, routes);
localesRoutes.description = `Builds locales and routes`;
const front = gulp.series(clean, gulp.parallel(localesRoutes, watch, webpackDevServer));
front.description = `Starts frontend service`;
2018-01-31 11:17:17 +00:00
function backOnly(done) {
2018-12-27 11:54:16 +00:00
let app = require(`./loopback/server/server`);
2019-01-25 22:02:29 +00:00
app.start();
app.on('started', done);
}
backOnly.description = `Starts backend service`;
function backWatch(done) {
const nodemon = require('gulp-nodemon');
2019-01-28 11:14:22 +00:00
// XXX: Workaround to avoid nodemon bug
// https://github.com/remy/nodemon/issues/1346
2019-01-29 20:00:27 +00:00
let commands = ['node --inspect ./node_modules/gulp/bin/gulp.js'];
if (!isWindows) commands.unshift('sleep 1');
nodemon({
2019-01-29 20:00:27 +00:00
exec: commands.join(' && '),
2019-02-18 11:55:22 +00:00
ext: 'js html css json',
args: ['backOnly'],
watch: backSources,
done: done
});
}
backWatch.description = `Starts backend in waching mode`;
const back = gulp.series(dockerStart, backWatch);
back.description = `Starts backend and database service`;
const defaultTask = gulp.parallel(front, back);
defaultTask.description = `Starts all application services`;
// Backend tests
2019-02-13 09:00:56 +00:00
async function backTestOnce() {
2019-02-11 21:57:48 +00:00
let bootOptions;
if (argv['random'])
bootOptions = {dataSources};
let app = require(`./loopback/server/server`);
2019-02-11 21:57:48 +00:00
app.boot(bootOptions);
2019-02-11 21:57:48 +00:00
await new Promise((resolve, reject) => {
const jasmine = require('gulp-jasmine');
2019-02-11 21:57:48 +00:00
let options = {errorOnFail: false};
2019-02-11 21:57:48 +00:00
if (argv.junit) {
const reporters = require('jasmine-reporters');
options.reporter = new reporters.JUnitXmlReporter();
}
2019-02-11 21:57:48 +00:00
let backSpecFiles = [
'back/**/*.spec.js',
'loopback/**/*.spec.js',
'modules/*/back/**/*.spec.js'
];
gulp.src(backSpecFiles)
.pipe(jasmine(options))
.on('end', resolve)
.resume();
});
await app.disconnect();
}
2019-02-13 09:00:56 +00:00
backTestOnce.description = `Runs the backend tests once, can receive --junit arg to save reports on a xml file`;
async function backTestDockerOnce() {
let containerId = await docker();
await backTestOnce();
if (argv['random'])
await execP(`docker rm -fv ${containerId}`);
}
backTestDockerOnce.description = `Runs backend tests using in site container once`;
2019-02-11 21:57:48 +00:00
async function backTestDocker() {
let containerId = await docker();
2019-02-13 09:00:56 +00:00
await backTest();
2019-02-11 21:57:48 +00:00
if (argv['random'])
await execP(`docker rm -fv ${containerId}`);
}
2019-02-13 13:01:37 +00:00
backTestDocker.description = `Runs backend tests restoring fixtures first`;
2019-01-24 11:01:35 +00:00
function backTest(done) {
const nodemon = require('gulp-nodemon');
nodemon({
2019-02-11 21:57:48 +00:00
exec: ['node ./node_modules/gulp/bin/gulp.js'],
2019-02-13 09:00:56 +00:00
args: ['backTestOnce'],
watch: backSources,
2019-01-24 11:01:35 +00:00
done: done
});
}
backTest.description = `Watches for changes in modules to execute backTest task`;
// End to end tests
function e2eOnly() {
2019-02-06 15:08:23 +00:00
require('@babel/register')({presets: ['@babel/preset-env']});
require('@babel/polyfill');
const jasmine = require('gulp-jasmine');
2019-02-06 15:08:23 +00:00
const SpecReporter = require('jasmine-spec-reporter').SpecReporter;
const createNightmare = require('./e2e/helpers/nightmare');
2018-09-05 11:01:21 +00:00
if (argv.show || argv.s)
process.env.E2E_SHOW = true;
2019-05-23 07:30:29 +00:00
process.env.ELECTRON_DISABLE_SECURITY_WARNINGS = true;
2018-09-05 11:01:21 +00:00
2019-02-06 15:08:23 +00:00
const specFiles = [
2019-06-19 07:03:45 +00:00
`${__dirname}/e2e/paths/01*/*[sS]pec.js`,
`${__dirname}/e2e/paths/02*/*[sS]pec.js`,
`${__dirname}/e2e/paths/03*/*[sS]pec.js`,
`${__dirname}/e2e/paths/04*/*[sS]pec.js`,
`${__dirname}/e2e/paths/05*/*[sS]pec.js`,
`${__dirname}/e2e/paths/06*/*[sS]pec.js`,
`${__dirname}/e2e/paths/07*/*[sS]pec.js`,
`${__dirname}/e2e/paths/08*/*[sS]pec.js`,
`${__dirname}/e2e/paths/09*/*[sS]pec.js`,
2019-02-06 15:08:23 +00:00
`${__dirname}/e2e/paths/**/*[sS]pec.js`,
`${__dirname}/e2e/helpers/extensions.js`
];
return gulp.src(specFiles).pipe(jasmine({
errorOnFail: false,
timeout: 10000,
reporter: [
new SpecReporter({
spec: {
displayStacktrace: 'summary',
displaySuccessful: true,
displayFailedSpec: true,
displaySpecDuration: true,
}
})
]
})
.on('jasmineDone', function() {
const nightmare = createNightmare();
nightmare.end(() => {});
}));
}
e2eOnly.description = `Runs the e2e tests only`;
2018-02-01 07:48:54 +00:00
async function backendStatus() {
const milliseconds = 250;
return new Promise(resolve => {
let timer;
let attempts = 1;
timer = setInterval(() => {
const url = `${e2eConfig.url}/api/Applications/status`;
request.get(url, (err, res) => {
if (err || attempts > 100) // 250ms * 100 => 25s timeout
throw new Error('Could not connect to backend');
else if (res && res.body == 'true') {
clearInterval(timer);
resolve(attempts);
} else
attempts++;
});
}, milliseconds);
});
}
backendStatus.description = `Performs a simple requests to check the backend status`;
e2e = gulp.series(docker, async function isBackendReady() {
const attempts = await backendStatus();
log(`Backend ready after ${attempts} attempt(s)`);
return attempts;
}, e2eOnly);
e2e.description = `Restarts database and runs the e2e tests`;
function smokesOnly() {
2018-03-09 19:04:03 +00:00
const jasmine = require('gulp-jasmine');
2019-01-04 14:51:03 +00:00
return gulp.src('./e2e/smokes-tests.js')
.pipe(jasmine({reporter: 'none'}));
}
smokesOnly.description = `Runs the smokes tests only`;
2018-03-09 19:04:03 +00:00
smokes = gulp.series(docker, smokesOnly);
smokes.description = `Restarts database and runs the smokes tests`;
function install() {
const install = require('gulp-install');
const print = require('gulp-print');
2019-01-22 10:16:14 +00:00
let packageFiles = ['front/package.json', 'print/package.json'];
return gulp.src(packageFiles)
.pipe(print(filepath => {
return `Installing packages in ${filepath}`;
}))
.pipe(install({
npm: ['--no-package-lock']
}));
}
install.description = `Installs node dependencies in all directories`;
2018-06-12 09:24:43 +00:00
const i = gulp.series(install);
i.description = `Alias for the 'install' task`;
// Deployment
2018-06-12 09:24:43 +00:00
2019-01-25 22:02:29 +00:00
const build = gulp.series(clean, gulp.parallel(localesRoutes, webpack));
build.description = `Generates binaries and configuration files`;
function clean() {
const del = require('del');
const files = [
2018-12-27 15:10:55 +00:00
`${buildDir}/*`
];
return del(files, {force: true});
}
clean.description = `Cleans all files generated by the 'build' task`;
// Webpack
function webpack(done) {
const webpackCompile = require('webpack');
const merge = require('webpack-merge');
let wpConfig = require('./webpack.config.js');
wpConfig = merge(wpConfig, {});
let compiler = webpackCompile(wpConfig);
compiler.run(function(err, stats) {
if (err) throw new PluginError('webpack', err);
log('[webpack]', stats.toString(wpConfig.stats));
done();
});
}
webpack.description = `Transpiles application into files`;
function webpackDevServer(done) {
const webpack = require('webpack');
const merge = require('webpack-merge');
const WebpackDevServer = require('webpack-dev-server');
let wpConfig = require('./webpack.config.js');
wpConfig = merge(wpConfig, {});
let devServer = wpConfig.devServer;
for (let entryName in wpConfig.entry) {
let entry = wpConfig.entry[entryName];
if (!Array.isArray(entry))
entry = [entry];
let wdsAssets = [
`webpack-dev-server/client?http://localhost:${devServer.port}/`,
`webpack/hot/dev-server`
];
wpConfig.entry[entryName] = wdsAssets.concat(entry);
}
let compiler = webpack(wpConfig);
new WebpackDevServer(compiler, wpConfig.devServer)
.listen(devServer.port, devServer.host, function(err) {
if (err) throw new PluginError('webpack-dev-server', err);
// TODO: Keep the server alive or continue?
done();
});
}
webpackDevServer.description = `Transpiles application into memory`;
// Locale
2018-12-27 11:54:16 +00:00
let localeFiles = [
`${srcDir}/**/locale/*.yml`,
`${modulesDir}/*/front/**/locale/*.yml`
];
/**
* Mixes all locale files into one JSON file per module and language. It looks
* recursively in all project directories for locale folders with per language
* yaml translation files.
*
* @return {Stream} The merged gulp streams
*/
function locales() {
const mergeJson = require('gulp-merge-json');
const gulpFile = require('gulp-file');
2018-02-04 14:46:46 +00:00
const yaml = require('gulp-yaml');
2018-02-05 18:34:04 +00:00
const merge = require('merge-stream');
const fs = require('fs-extra');
2018-02-05 18:34:04 +00:00
let streams = [];
2018-12-27 11:54:16 +00:00
let localePaths = [];
let modules = fs.readdirSync(modulesDir);
for (let mod of modules)
2018-12-27 11:54:16 +00:00
localePaths[mod] = `${modulesDir}/${mod}`;
2019-01-25 22:02:29 +00:00
let baseMods = ['core', 'salix'];
2018-12-27 11:54:16 +00:00
for (let mod of baseMods)
localePaths[mod] = `${srcDir}/${mod}`;
for (let mod in localePaths) {
let path = localePaths[mod];
for (let lang of langs) {
2018-12-27 11:54:16 +00:00
let localeFiles = `${path}/**/locale/${lang}.yml`;
2017-03-01 08:55:17 +00:00
streams.push(gulp.src(localeFiles)
2018-02-04 14:46:46 +00:00
.pipe(yaml())
.pipe(mergeJson({fileName: `${lang}.json`}))
2017-03-01 08:55:17 +00:00
.pipe(gulp.dest(`${buildDir}/locale/${mod}`)));
}
}
for (let mod in localePaths) {
for (let lang of langs) {
let file = `${buildDir}/locale/${mod}/${lang}.json`;
if (fs.existsSync(file)) continue;
streams.push(gulpFile('en.json', '{}', {src: true})
.pipe(gulp.dest(`${buildDir}/locale/${mod}`)));
}
}
return merge(streams);
}
locales.description = `Generates client locale files`;
// Routes
2018-12-27 11:54:16 +00:00
let routeFiles = `${modulesDir}/*/front/routes.json`;
function routes() {
2018-02-05 18:34:04 +00:00
const concat = require('gulp-concat');
const wrap = require('gulp-wrap');
return gulp.src(routeFiles)
.pipe(concat('routes.js', {newLine: ','}))
.pipe(wrap('var routes = [<%=contents%>\n];'))
.pipe(gulp.dest(buildDir));
}
routes.description = 'Merges all module routes file into one file';
// Watch
function watch(done) {
gulp.watch(routeFiles, gulp.series(routes));
gulp.watch(localeFiles, gulp.series(locales));
done();
}
watch.description = `Watches for changes in routes and locale files`;
2017-10-18 04:41:17 +00:00
2018-02-01 15:01:49 +00:00
// Docker
/**
* Builds the database image and runs a container. It only rebuilds the
* image when fixtures have been modified or when the day on which the
* image was built is different to today. Some workarounds have been used
* to avoid a bug with OverlayFS driver on MacOS.
*/
async function docker() {
let d = new Date();
let pad = v => v < 10 ? '0' + v : v;
let stamp = `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}`;
await execP(`docker build --build-arg STAMP=${stamp} -t salix-db ./db`);
2019-02-11 21:57:48 +00:00
let dockerArgs = `--name ${containerId} -p 3306:${dbConf.port}`;
if (argv['random'])
dockerArgs = '-p 3306';
else {
try {
await execP(`docker rm -fv ${containerId}`);
} catch (e) {}
}
2019-01-09 12:04:49 +00:00
let runChown = process.platform != 'linux';
2019-01-26 19:30:45 +00:00
if (argv['run-chown']) runChown = true;
2019-01-26 19:26:14 +00:00
2019-02-11 21:57:48 +00:00
let result = await execP(`docker run --env RUN_CHOWN=${runChown} -d ${dockerArgs} salix-db`);
containerId = result.stdout;
if (argv['random']) {
let inspect = await execP(`docker inspect -f "{{json .NetworkSettings.Ports}}" ${containerId}`);
let ports = JSON.parse(inspect.stdout);
dbConf.port = ports['3306/tcp'][0]['HostPort'];
}
if (runChown) await dockerWait();
2019-02-11 21:57:48 +00:00
return containerId;
}
docker.description = `Builds the database image and runs a container`;
/**
* Does the minium effort to start the database container, if it doesn't exists
* calls the 'docker' task, if it is started does nothing. Keep in mind that when
* you do not rebuild the docker you may be using an outdated version of it.
* See the 'docker' task for more info.
*/
async function dockerStart() {
let state;
try {
2019-02-11 21:57:48 +00:00
let result = await execP(`docker inspect -f "{{json .State}}" ${containerId}`);
state = JSON.parse(result.stdout);
} catch (err) {
return await docker();
}
switch (state.Status) {
case 'running':
return;
case 'exited':
2019-02-11 21:57:48 +00:00
await execP(`docker start ${containerId}`);
await dockerWait();
2019-01-04 12:32:04 +00:00
return;
default:
throw new Error(`Unknown docker status: ${state.Status}`);
}
}
dockerStart.description = `Starts the database container`;
function dockerWait() {
return new Promise((resolve, reject) => {
const mysql = require('mysql2');
let interval = 100;
let elapsedTime = 0;
2019-02-11 23:44:39 +00:00
let maxInterval = 4 * 60 * 1000;
2019-02-11 21:57:48 +00:00
let myConf = {
user: dbConf.username,
password: dbConf.password,
host: dbConf.host,
2019-02-27 16:04:32 +00:00
port: dbConf.port
2019-02-11 21:57:48 +00:00
};
log('Waiting for MySQL init process...');
checker();
async function checker() {
elapsedTime += interval;
let state;
try {
2019-02-11 21:57:48 +00:00
let result = await execP(`docker container inspect -f "{{json .State}}" ${containerId}`);
state = JSON.parse(result.stdout);
} catch (err) {
return reject(new Error(err.message));
}
if (state.Status === 'exited')
return reject(new Error('Docker exited, please see the docker logs for more info'));
2019-02-11 21:57:48 +00:00
let conn = mysql.createConnection(myConf);
conn.on('error', () => {});
conn.connect(err => {
conn.destroy();
2019-02-11 23:39:39 +00:00
if (!err) {
log('MySQL process ready.');
return resolve();
}
if (elapsedTime >= maxInterval)
2019-02-11 23:44:39 +00:00
reject(new Error(`MySQL not initialized whithin ${elapsedTime / 1000} secs`));
else
setTimeout(checker, interval);
});
}
});
}
dockerWait.description = `Waits until database service is ready`;
// Helpers
2018-02-01 11:52:58 +00:00
/**
* Promisified version of exec().
*
* @param {String} command The exec command
* @return {Promise} The promise
*/
function execP(command) {
return new Promise((resolve, reject) => {
exec(command, (err, stdout, stderr) => {
if (err)
reject(err);
else {
resolve({
stdout: stdout,
stderr: stderr
});
}
});
2018-02-01 11:52:58 +00:00
});
}
2018-02-01 11:52:58 +00:00
module.exports = {
default: defaultTask,
front,
back,
backOnly,
backWatch,
2019-02-13 09:00:56 +00:00
backTestOnce,
backTestDockerOnce,
backTest,
2019-02-11 21:57:48 +00:00
backTestDocker,
e2e,
2019-02-11 21:57:48 +00:00
e2eOnly,
smokes,
2019-02-11 21:57:48 +00:00
smokesOnly,
2019-01-09 14:41:15 +00:00
i,
2019-02-11 21:57:48 +00:00
install,
build,
clean,
webpack,
webpackDevServer,
routes,
2019-02-11 21:57:48 +00:00
locales,
localesRoutes,
watch,
docker,
dockerStart,
dockerWait,
backendStatus
};