szurubooru/client/build.js

199 lines
5.6 KiB
JavaScript
Raw Normal View History

'use strict';
const fs = require('fs');
2016-03-27 21:10:41 +02:00
const glob = require('glob');
const path = require('path');
const util = require('util');
2016-03-31 23:18:08 +02:00
const execSync = require('child_process').execSync;
2016-04-06 20:38:45 +02:00
const camelcase = require('camelcase');
function convertKeysToCamelCase(input) {
let result = {};
Object.keys(input).map((key, _) => {
const value = input[key];
if (value !== null && value.constructor == Object) {
result[camelcase(key)] = convertKeysToCamelCase(value);
} else {
result[camelcase(key)] = value;
}
});
return result;
}
2016-03-31 23:18:08 +02:00
function readTextFile(path) {
return fs.readFileSync(path, 'utf-8');
}
function writeFile(path, content) {
return fs.writeFileSync(path, content);
}
2016-03-31 23:18:08 +02:00
function getVersion() {
return execSync('git describe --always --dirty --long --tags').toString();
}
function getConfig() {
2016-04-06 20:38:45 +02:00
const yaml = require('js-yaml');
const merge = require('merge');
const camelcaseKeys = require('camelcase-keys');
2016-04-06 20:38:45 +02:00
function parseConfigFile(path) {
let result = yaml.load(readTextFile(path, 'utf-8'));
2016-04-06 20:38:45 +02:00
return convertKeysToCamelCase(result);
}
2016-04-06 20:38:45 +02:00
let config = parseConfigFile('../config.yaml.dist');
try {
2016-04-06 20:38:45 +02:00
const localConfig = parseConfigFile('../config.yaml');
config = merge.recursive(config, localConfig);
} catch (e) {
console.warn('Local config does not exist, ignoring');
}
config.canSendMails = !!config.smtp.host;
2016-04-06 20:38:45 +02:00
delete config.secret;
delete config.smtp;
delete config.database;
2016-03-31 23:18:08 +02:00
config.meta = {
version: getVersion(),
buildDate: new Date().toUTCString(),
};
return config;
}
function copyFile(source, target) {
fs.createReadStream(source).pipe(fs.createWriteStream(target));
}
function minifyJs(path) {
return require('uglify-js').minify(path).code;
}
function minifyCss(css) {
return require('csso').minify(css);
}
function minifyHtml(html) {
return require('html-minifier').minify(html, {
removeComments: true,
collapseWhitespace: true,
conservativeCollapse: true,
}).trim();
}
function bundleHtml(config) {
const underscore = require('underscore');
const babelify = require('babelify');
const baseHtml = readTextFile('./html/index.htm', 'utf-8');
const finalHtml = baseHtml
.replace(
/(<title>)(.*)(<\/title>)/,
util.format('$1%s$3', config.name));
writeFile('./public/index.htm', minifyHtml(finalHtml));
glob('./html/**/*.tpl', {}, (er, files) => {
let compiledTemplateJs = '\'use strict\'\n';
compiledTemplateJs += 'let _ = require(\'underscore\');';
compiledTemplateJs += 'let templates = {};';
2016-03-27 21:10:41 +02:00
for (const file of files) {
const name = path.basename(file, '.tpl').replace(/_/g, '-');
const templateText = minifyHtml(readTextFile(file, 'utf-8'));
const functionText = underscore.template(
templateText, {variable: 'ctx'}).source;
compiledTemplateJs += `templates['${name}'] = ${functionText};`;
2016-03-27 21:10:41 +02:00
}
compiledTemplateJs += 'module.exports = templates;';
writeFile('./js/.templates.autogen.js', compiledTemplateJs);
2016-03-27 21:10:41 +02:00
console.info('Bundled HTML');
});
}
function bundleCss() {
2016-04-13 20:06:34 +02:00
const stylus = require('stylus');
glob('./css/**/*.styl', {}, (er, files) => {
let css = '';
for (const file of files) {
2016-04-13 20:06:34 +02:00
css += stylus.render(
readTextFile(file), {filename: file});
}
writeFile('./public/css/app.min.css', minifyCss(css));
console.info('Bundled CSS');
});
}
function bundleJs(config) {
const browserify = require('browserify');
const external = [
'underscore',
'superagent',
'mousetrap',
'js-cookie',
'page',
'nprogress',
];
function writeJsBundle(b, path, message, compress) {
let outputFile = fs.createWriteStream(path);
b.bundle().pipe(outputFile);
outputFile.on('finish', function() {
if (compress) {
writeFile(path, minifyJs(path));
}
console.info(message);
});
}
glob('./js/**/*.js', {}, (er, files) => {
if (!process.argv.includes('--no-vendor-js')) {
let b = browserify();
for (let lib of external) {
b.require(lib);
}
if (config.transpile) {
b.add(require.resolve('babel-polyfill'));
}
writeJsBundle(
b, './public/js/vendor.min.js', 'Bundled vendor JS', true);
}
if (!process.argv.includes('--no-app-js')) {
let outputFile = fs.createWriteStream('./public/js/app.min.js');
let b = browserify({debug: config.debug});
if (config.transpile) {
b = b.transform('babelify');
}
writeJsBundle(
b.external(external).add(files),
'./public/js/app.min.js',
'Bundled app JS',
!config.debug);
}
});
}
function bundleConfig(config) {
writeFile(
2016-04-01 18:45:25 +02:00
'./js/.config.autogen.json', JSON.stringify(config));
}
function bundleBinaryAssets() {
copyFile('./img/favicon.png', './public/img/favicon.png');
copyFile('./img/404.png', './public/img/404.png');
2016-03-30 16:06:51 +02:00
}
const config = getConfig();
bundleConfig(config);
bundleBinaryAssets();
if (!process.argv.includes('--no-html')) {
bundleHtml(config);
}
if (!process.argv.includes('--no-css')) {
bundleCss();
}
if (!process.argv.includes('--no-js')) {
bundleJs(config);
}