Mirror of espurna firmware for wireless switches and more
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

274 lines
7.5 KiB

7 years ago
7 years ago
6 years ago
7 years ago
7 years ago
7 years ago
6 years ago
6 years ago
6 years ago
6 years ago
7 years ago
7 years ago
5 years ago
5 years ago
  1. /*
  2. ESP8266 file system builder
  3. Copyright (C) 2016-2019 by Xose Pérez <xose dot perez at gmail dot com>
  4. This program is free software: you can redistribute it and/or modify
  5. it under the terms of the GNU General Public License as published by
  6. the Free Software Foundation, either version 3 of the License, or
  7. (at your option) any later version.
  8. This program is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. GNU General Public License for more details.
  12. You should have received a copy of the GNU General Public License
  13. along with this program. If not, see <http://www.gnu.org/licenses/>.
  14. */
  15. /*eslint quotes: ['error', 'single']*/
  16. /*eslint-env es6*/
  17. // -----------------------------------------------------------------------------
  18. // Dependencies
  19. // -----------------------------------------------------------------------------
  20. const gulp = require('gulp');
  21. const through = require('through2');
  22. const htmlmin = require('gulp-htmlmin');
  23. const inline = require('gulp-inline');
  24. const inlineImages = require('gulp-css-base64');
  25. const favicon = require('gulp-base64-favicon');
  26. const crass = require('gulp-crass');
  27. const htmllint = require('gulp-htmllint');
  28. const csslint = require('gulp-csslint');
  29. const rename = require('gulp-rename');
  30. const replace = require('gulp-replace');
  31. const remover = require('gulp-remove-code');
  32. const gzip = require('gulp-gzip');
  33. const path = require('path');
  34. const terser = require('terser')
  35. // -----------------------------------------------------------------------------
  36. // Configuration
  37. // -----------------------------------------------------------------------------
  38. const htmlFolder = 'html/';
  39. const configFolder = 'espurna/config/';
  40. const dataFolder = 'espurna/data/';
  41. const staticFolder = 'espurna/static/';
  42. // -----------------------------------------------------------------------------
  43. // Methods
  44. // -----------------------------------------------------------------------------
  45. var toHeader = function(name, debug) {
  46. return through.obj(function (source, encoding, callback) {
  47. var parts = source.path.split(path.sep);
  48. var filename = parts[parts.length - 1];
  49. var safename = name || filename.split('.').join('_');
  50. // Generate output
  51. var output = '';
  52. output += '#define ' + safename + '_len ' + source.contents.length + '\n';
  53. output += 'const uint8_t ' + safename + '[] PROGMEM = {';
  54. for (var i=0; i<source.contents.length; i++) {
  55. if (i > 0) { output += ','; }
  56. if (0 === (i % 20)) { output += '\n'; }
  57. output += '0x' + ('00' + source.contents[i].toString(16)).slice(-2);
  58. }
  59. output += '\n};';
  60. // clone the contents
  61. var destination = source.clone();
  62. destination.path = source.path + '.h';
  63. destination.contents = Buffer.from(output);
  64. if (debug) {
  65. console.info('Image ' + filename + ' \tsize: ' + source.contents.length + ' bytes');
  66. }
  67. callback(null, destination);
  68. });
  69. };
  70. var htmllintReporter = function(filepath, issues) {
  71. if (issues.length > 0) {
  72. issues.forEach(function (issue) {
  73. console.info(
  74. '[gulp-htmllint] ' +
  75. filepath + ' [' +
  76. issue.line + ',' +
  77. issue.column + ']: ' +
  78. '(' + issue.code + ') ' +
  79. issue.msg
  80. );
  81. });
  82. process.exitCode = 1;
  83. }
  84. };
  85. var compressJs = function() {
  86. return through.obj(function (source, encoding, callback) {
  87. if (source.isNull()) {
  88. callback(null, source);
  89. return;
  90. }
  91. if (source.path.endsWith("custom.js")) {
  92. var result = terser.minify(source.contents.toString());
  93. source.contents = Buffer.from(result.code);
  94. this.push(source);
  95. callback();
  96. return;
  97. }
  98. callback(null, source);
  99. return;
  100. });
  101. };
  102. var buildWebUI = function(module) {
  103. // Declare some modules as optional to remove with
  104. // removeIf(!name) ...code... endRemoveIf(!name) sections
  105. // (via gulp-remove-code)
  106. var modules = {
  107. 'light': false,
  108. 'sensor': false,
  109. 'rfbridge': false,
  110. 'rfm69': false,
  111. 'garland': false,
  112. 'thermostat': false,
  113. 'lightfox': false,
  114. 'curtain': false
  115. };
  116. // Note: only build these when specified as module arg
  117. var excludeAll = [
  118. 'rfm69',
  119. 'lightfox'
  120. ];
  121. // 'all' to include all *but* excludeAll
  122. // '<module>' to include a single module
  123. // 'small' is the default state (all disabled)
  124. if ('all' === module) {
  125. Object.keys(modules).
  126. filter(function(key) {
  127. return excludeAll.indexOf(key) < 0;
  128. }).
  129. forEach(function(key) {
  130. modules[key] = true;
  131. });
  132. } else if ('small' !== module) {
  133. modules[module] = true;
  134. }
  135. return gulp.src(htmlFolder + '*.html').
  136. pipe(remover(modules)).
  137. pipe(htmllint({
  138. 'failOnError': true,
  139. 'rules': {
  140. 'id-class-style': false,
  141. 'label-req-for': false,
  142. 'line-end-style': false,
  143. }
  144. }, htmllintReporter)).
  145. pipe(favicon()).
  146. pipe(inline({
  147. base: htmlFolder,
  148. js: [function() { return remover(modules); }, compressJs],
  149. css: [crass, inlineImages],
  150. disabledTypes: ['svg', 'img']
  151. })).
  152. pipe(remover(modules)).
  153. pipe(htmlmin({
  154. collapseWhitespace: true,
  155. removeComments: true,
  156. minifyCSS: true,
  157. minifyJS: false
  158. })).
  159. pipe(replace('pure-', 'p-')).
  160. pipe(gzip({ gzipOptions: { level: 9 } })).
  161. pipe(rename('index.' + module + '.html.gz')).
  162. pipe(gulp.dest(dataFolder)).
  163. pipe(toHeader('webui_image', true)).
  164. pipe(gulp.dest(staticFolder));
  165. };
  166. // -----------------------------------------------------------------------------
  167. // Tasks
  168. // -----------------------------------------------------------------------------
  169. gulp.task('certs', function() {
  170. gulp.src(dataFolder + 'server.*').
  171. pipe(toHeader(debug=false)).
  172. pipe(gulp.dest(staticFolder));
  173. });
  174. gulp.task('csslint', function() {
  175. gulp.src(htmlFolder + '*.css').
  176. pipe(csslint({ids: false})).
  177. pipe(csslint.formatter());
  178. });
  179. gulp.task('webui_small', function() {
  180. return buildWebUI('small');
  181. });
  182. gulp.task('webui_sensor', function() {
  183. return buildWebUI('sensor');
  184. });
  185. gulp.task('webui_light', function() {
  186. return buildWebUI('light');
  187. });
  188. gulp.task('webui_rfbridge', function() {
  189. return buildWebUI('rfbridge');
  190. });
  191. gulp.task('webui_rfm69', function() {
  192. return buildWebUI('rfm69');
  193. });
  194. gulp.task('webui_lightfox', function() {
  195. return buildWebUI('lightfox');
  196. });
  197. gulp.task('webui_garland', function() {
  198. return buildWebUI('garland');
  199. });
  200. gulp.task('webui_thermostat', function() {
  201. return buildWebUI('thermostat');
  202. });
  203. gulp.task('webui_curtain', function() {
  204. return buildWebUI('curtain');
  205. });
  206. gulp.task('webui_all', function() {
  207. return buildWebUI('all');
  208. });
  209. gulp.task('webui',
  210. gulp.parallel(
  211. 'webui_small',
  212. 'webui_sensor',
  213. 'webui_light',
  214. 'webui_rfbridge',
  215. 'webui_rfm69',
  216. 'webui_lightfox',
  217. 'webui_garland',
  218. 'webui_thermostat',
  219. 'webui_curtain',
  220. 'webui_all'
  221. )
  222. );
  223. gulp.task('default', gulp.series('webui'));