- /*
-
- ESP8266 file system builder
-
- Copyright (C) 2016-2019 by Xose Pérez <xose dot perez at gmail dot com>
-
- This program is free software: you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation, either version 3 of the License, or
- (at your option) any later version.
-
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU General Public License for more details.
-
- You should have received a copy of the GNU General Public License
- along with this program. If not, see <http://www.gnu.org/licenses/>.
-
- */
-
- /*eslint quotes: ['error', 'single']*/
- /*eslint-env es6*/
-
- // -----------------------------------------------------------------------------
- // Dependencies
- // -----------------------------------------------------------------------------
-
- const path = require('path');
-
- const gulp = require('gulp');
- const through = require('through2');
-
- const htmllint = require('gulp-htmllint');
- const csslint = require('gulp-csslint');
-
- const htmlmin = require('html-minifier');
-
- const gzip = require('gulp-gzip');
- const inline = require('gulp-inline-source-html');
- const rename = require('gulp-rename');
- const replace = require('gulp-replace');
-
- // -----------------------------------------------------------------------------
- // Configuration
- // -----------------------------------------------------------------------------
-
- const htmlFolder = 'html/';
- const dataFolder = 'espurna/data/';
- const staticFolder = 'espurna/static/';
-
- // -----------------------------------------------------------------------------
- // Methods
- // -----------------------------------------------------------------------------
-
- var toMinifiedHtml = function(options) {
- return through.obj(function (source, encoding, callback) {
- if (source.isNull()) {
- callback(null, source);
- return;
- }
-
- var contents = source.contents.toString();
- source.contents = Buffer.from(htmlmin.minify(contents, options));
- callback(null, source);
- });
- }
-
- var toHeader = function(name, debug) {
-
- return through.obj(function (source, encoding, callback) {
-
- var parts = source.path.split(path.sep);
- var filename = parts[parts.length - 1];
- var safename = name || filename.split('.').join('_');
-
- // Generate output
- var output = '';
- output += '#define ' + safename + '_len ' + source.contents.length + '\n';
- output += 'const uint8_t ' + safename + '[] PROGMEM = {';
- for (var i=0; i<source.contents.length; i++) {
- if (i > 0) { output += ','; }
- if (0 === (i % 20)) { output += '\n'; }
- output += '0x' + ('00' + source.contents[i].toString(16)).slice(-2);
- }
- output += '\n};';
-
- // clone the contents
- var destination = source.clone();
- destination.path = source.path + '.h';
- destination.contents = Buffer.from(output);
-
- if (debug) {
- console.info('Image ' + filename + ' \tsize: ' + source.contents.length + ' bytes');
- }
-
- callback(null, destination);
-
- });
-
- };
-
- var htmllintReporter = function(filepath, issues) {
- if (issues.length > 0) {
- issues.forEach(function (issue) {
- console.info(
- '[gulp-htmllint] ' +
- filepath + ' [' +
- issue.line + ',' +
- issue.column + ']: ' +
- '(' + issue.code + ') ' +
- issue.msg
- );
- });
- process.exitCode = 1;
- }
- };
-
- // TODO: this is a roughly equivalent port of the gulp-remove-code,
- // which also uses regexp rules to filter in-between specially-formatted comment blocks
-
- var jsRegexp = function(module) {
- return '//\\s*removeIf\\(!' + module + '\\)'
- + '\\s*(\n|\r|.)*?'
- + '//\\s*endRemoveIf\\(!' + module + '\\)';
- }
-
- var cssRegexp = function(module) {
- return '/\\*\\s*removeIf\\(!' + module + '\\)\\s*\\*/'
- + '\\s*(\n|\r|.)*?'
- + '/\\*\\s*endRemoveIf\\(!' + module + '\\)\\s*\\*/';
- }
-
- var htmlRegexp = function(module) {
- return '<!--\\s*removeIf\\(!' + module + '\\)\\s*-->'
- + '\\s*(\n|\r|.)*?'
- + '<!--\\s*endRemoveIf\\(!' + module + '\\)\\s*-->';
- }
-
- var generateRegexps = function(modules, func) {
- var regexps = new Set();
- for (const [module, enabled] of Object.entries(modules)) {
- if (enabled) {
- continue;
- }
-
- const expression = func(module);
- const re = new RegExp(expression, 'gm');
-
- regexps.add(re);
- }
-
- return regexps;
- }
-
- // TODO: use html parser here?
- // TODO: separate js files to include js, html & css and avoid 2 step regexp?
-
- var htmlRemover = function(modules) {
- const regexps = generateRegexps(modules, htmlRegexp);
-
- return through.obj(function (source, _, callback) {
- if (source.isNull()) {
- callback(null, source);
- return;
- }
-
- var contents = source.contents.toString();
- for (var regexp of regexps) {
- contents = contents.replace(regexp, '');
- }
- source.contents = Buffer.from(contents);
- callback(null, source);
- });
- }
-
- var inlineHandler = function(modules) {
- return function(source) {
- if (((source.sourcepath === 'custom.css') || (source.sourcepath === 'custom.js'))) {
- const filter = (source.type === 'css') ? cssRegexp : jsRegexp;
- const regexps = generateRegexps(modules, filter);
-
- var content = source.fileContent;
- for (var regexp of regexps) {
- content = content.replace(regexp, '');
- }
-
- source.fileContent = content;
- return;
- }
-
- if (source.sourcepath === "favicon.ico") {
- source.format = "x-icon";
- return;
- }
-
- if (source.content) {
- return;
- }
-
- // Just ignore the vendored libs, repackaging makes things worse for the size
- const path = source.sourcepath;
- if (path.endsWith('.min.js')) {
- source.compress = false;
- } else if (path.endsWith('.min.css')) {
- source.compress = false;
- }
- };
- }
-
- var buildWebUI = function(module) {
-
- // Declare some modules as optional to remove with
- // removeIf(!name) ...code... endRemoveIf(!name) sections
- // (via gulp-remove-code)
- var modules = {
- 'light': false,
- 'sensor': false,
- 'rfbridge': false,
- 'rfm69': false,
- 'garland': false,
- 'thermostat': false,
- 'lightfox': false,
- 'curtain': false
- };
-
- // Note: only build these when specified as module arg
- var excludeAll = [
- 'rfm69',
- 'lightfox'
- ];
-
- // 'all' to include all *but* excludeAll
- // '<module>' to include a single module
- // 'small' is the default state (all disabled)
- if ('all' === module) {
- Object.keys(modules).
- filter(function(key) {
- return excludeAll.indexOf(key) < 0;
- }).
- forEach(function(key) {
- modules[key] = true;
- });
- } else if ('small' !== module) {
- modules[module] = true;
- }
-
- return gulp.src(htmlFolder + '*.html').
- pipe(htmllint({
- 'failOnError': true,
- 'rules': {
- 'id-class-style': false,
- 'label-req-for': false,
- 'line-end-style': false,
- 'attr-req-value': false
- }
- }, htmllintReporter)).
- pipe(htmlRemover(modules)).
- pipe(inline({handlers: [inlineHandler(modules)]})).
- pipe(toMinifiedHtml({
- collapseWhitespace: true,
- removeComments: true,
- minifyCSS: false,
- minifyJS: false
- })).
- pipe(replace('pure-', 'p-')).
- pipe(gzip({ gzipOptions: { level: 9 } })).
- pipe(rename('index.' + module + '.html.gz')).
- pipe(gulp.dest(dataFolder)).
- pipe(toHeader('webui_image', true)).
- pipe(gulp.dest(staticFolder));
-
- };
-
- // -----------------------------------------------------------------------------
- // Tasks
- // -----------------------------------------------------------------------------
-
- gulp.task('certs', function() {
- gulp.src(dataFolder + 'server.*').
- pipe(toHeader('', false)).
- pipe(gulp.dest(staticFolder));
- });
-
- gulp.task('csslint', function() {
- gulp.src(htmlFolder + '*.css').
- pipe(csslint({ids: false})).
- pipe(csslint.formatter());
- });
-
- gulp.task('webui_small', function() {
- return buildWebUI('small');
- });
-
- gulp.task('webui_sensor', function() {
- return buildWebUI('sensor');
- });
-
- gulp.task('webui_light', function() {
- return buildWebUI('light');
- });
-
- gulp.task('webui_rfbridge', function() {
- return buildWebUI('rfbridge');
- });
-
- gulp.task('webui_rfm69', function() {
- return buildWebUI('rfm69');
- });
-
- gulp.task('webui_lightfox', function() {
- return buildWebUI('lightfox');
- });
-
- gulp.task('webui_garland', function() {
- return buildWebUI('garland');
- });
-
- gulp.task('webui_thermostat', function() {
- return buildWebUI('thermostat');
- });
-
- gulp.task('webui_curtain', function() {
- return buildWebUI('curtain');
- });
-
- gulp.task('webui_all', function() {
- return buildWebUI('all');
- });
-
- gulp.task('webui',
- gulp.parallel(
- 'webui_small',
- 'webui_sensor',
- 'webui_light',
- 'webui_rfbridge',
- 'webui_rfm69',
- 'webui_lightfox',
- 'webui_garland',
- 'webui_thermostat',
- 'webui_curtain',
- 'webui_all'
- )
- );
-
- gulp.task('default', gulp.series('webui'));
|