gulpfile.js 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593
  1. var gulp = require("gulp");
  2. var uglify = require("gulp-uglify");
  3. var typescript = require("gulp-typescript");
  4. var sourcemaps = require("gulp-sourcemaps");
  5. var srcToVariable = require("gulp-content-to-variable");
  6. var appendSrcToVariable = require("./gulp-appendSrcToVariable");
  7. var addDtsExport = require("./gulp-addDtsExport");
  8. var addModuleExports = require("./gulp-addModuleExports");
  9. var merge2 = require("merge2");
  10. var concat = require("gulp-concat");
  11. var rename = require("gulp-rename");
  12. var cleants = require("gulp-clean-ts-extends");
  13. var changedInPlace = require("gulp-changed-in-place");
  14. var runSequence = require("run-sequence");
  15. var replace = require("gulp-replace");
  16. var uncommentShader = require("./gulp-removeShaderComments");
  17. var expect = require("gulp-expect-file");
  18. var optimisejs = require("gulp-optimize-js");
  19. var webserver = require("gulp-webserver");
  20. var path = require("path");
  21. var sass = require("gulp-sass");
  22. var webpack = require("webpack-stream");
  23. var config = require("./config.json");
  24. var del = require("del");
  25. var karmaServer = require('karma').Server;
  26. var debug = require("gulp-debug");
  27. var includeShadersStream;
  28. var shadersStream;
  29. var workersStream;
  30. var extendsSearchRegex = /var\s__extends[\s\S]+?\}\)\(\);/g;
  31. var decorateSearchRegex = /var\s__decorate[\s\S]+?\};/g;
  32. var referenceSearchRegex = /\/\/\/ <reference.*/g;
  33. /**
  34. * TS configurations shared in the gulp file.
  35. */
  36. var tsConfig = {
  37. noResolve: true,
  38. target: "ES5",
  39. declarationFiles: true,
  40. typescript: require("typescript"),
  41. experimentalDecorators: true,
  42. isolatedModules: false,
  43. noImplicitAny: true,
  44. noImplicitReturns: true,
  45. noImplicitThis: true,
  46. noUnusedLocals: true,
  47. strictNullChecks: true,
  48. strictFunctionTypes: true,
  49. types: []
  50. };
  51. var tsProject = typescript.createProject(tsConfig);
  52. var externalTsConfig = {
  53. noResolve: false,
  54. target: "ES5",
  55. declarationFiles: true,
  56. typescript: require("typescript"),
  57. experimentalDecorators: true,
  58. isolatedModules: false,
  59. noImplicitAny: true,
  60. noImplicitReturns: true,
  61. noImplicitThis: true,
  62. noUnusedLocals: true,
  63. strictNullChecks: true,
  64. types: []
  65. };
  66. var minimist = require("minimist");
  67. var commandLineOptions = minimist(process.argv.slice(2), {
  68. boolean: "public"
  69. });
  70. function processDependency(kind, dependency, filesToLoad) {
  71. if (dependency.dependUpon) {
  72. for (var i = 0; i < dependency.dependUpon.length; i++) {
  73. var dependencyName = dependency.dependUpon[i];
  74. var parent = config.workloads[dependencyName];
  75. processDependency(kind, parent, filesToLoad);
  76. }
  77. }
  78. var content = dependency[kind];
  79. if (!content) {
  80. return;
  81. }
  82. for (var i = 0; i < content.length; i++) {
  83. var file = content[i];
  84. if (filesToLoad.indexOf(file) === -1) {
  85. filesToLoad.push(file);
  86. }
  87. }
  88. }
  89. function determineFilesToProcess(kind) {
  90. var currentConfig = config.build.currentConfig;
  91. var buildConfiguration = config.buildConfigurations[currentConfig];
  92. var filesToLoad = [];
  93. for (var index = 0; index < buildConfiguration.length; index++) {
  94. var dependencyName = buildConfiguration[index];
  95. var dependency = config.workloads[dependencyName];
  96. if (kind === "directFiles" && !dependency) {
  97. filesToLoad.push("../../dist/preview release/" + dependencyName);
  98. }
  99. else if (dependency) {
  100. processDependency(kind, dependency, filesToLoad);
  101. }
  102. }
  103. if (kind === "shaderIncludes") {
  104. for (var index = 0; index < filesToLoad.length; index++) {
  105. filesToLoad[index] = "../../src/Shaders/ShadersInclude/" + filesToLoad[index] + ".fx";
  106. }
  107. } else if (kind === "shaders") {
  108. for (var index = 0; index < filesToLoad.length; index++) {
  109. var name = filesToLoad[index];
  110. filesToLoad[index] = "../../src/Shaders/" + filesToLoad[index] + ".fx";
  111. }
  112. }
  113. return filesToLoad;
  114. }
  115. /*
  116. * Shader Management.
  117. */
  118. function shadersName(filename) {
  119. return path.basename(filename)
  120. .replace(".fragment", "Pixel")
  121. .replace(".vertex", "Vertex")
  122. .replace(".fx", "Shader");
  123. }
  124. function includeShadersName(filename) {
  125. return path.basename(filename).replace(".fx", "");
  126. }
  127. /*
  128. * Main necessary files stream Management.
  129. */
  130. gulp.task("includeShaders", function (cb) {
  131. var filesToProcess = determineFilesToProcess("shaderIncludes");
  132. includeShadersStream = gulp.src(filesToProcess).
  133. pipe(expect.real({ errorOnFailure: true }, filesToProcess)).
  134. pipe(uncommentShader()).
  135. pipe(srcToVariable({
  136. variableName: "BABYLON.Effect.IncludesShadersStore", asMap: true, namingCallback: includeShadersName
  137. }));
  138. cb();
  139. });
  140. gulp.task("shaders", ["includeShaders"], function (cb) {
  141. var filesToProcess = determineFilesToProcess("shaders");
  142. shadersStream = gulp.src(filesToProcess).
  143. pipe(expect.real({ errorOnFailure: true }, filesToProcess)).
  144. pipe(uncommentShader()).
  145. pipe(srcToVariable({
  146. variableName: "BABYLON.Effect.ShadersStore", asMap: true, namingCallback: shadersName
  147. }));
  148. cb();
  149. });
  150. gulp.task("workers", function (cb) {
  151. workersStream = config.workers.map(function (workerDef) {
  152. return gulp.src(workerDef.files).
  153. pipe(expect.real({ errorOnFailure: true }, workerDef.files)).
  154. pipe(uglify()).
  155. pipe(srcToVariable({
  156. variableName: workerDef.variable
  157. }));
  158. });
  159. cb();
  160. });
  161. /**
  162. * Build tasks to concat minify uflify optimise the BJS js in different flavor (workers...).
  163. */
  164. gulp.task("buildWorker", ["workers", "shaders"], function () {
  165. var filesToProcess = determineFilesToProcess("files");
  166. return merge2(
  167. gulp.src(filesToProcess).
  168. pipe(expect.real({ errorOnFailure: true }, filesToProcess)),
  169. shadersStream,
  170. includeShadersStream,
  171. workersStream
  172. )
  173. .pipe(concat(config.build.minWorkerFilename))
  174. .pipe(cleants())
  175. .pipe(replace(extendsSearchRegex, ""))
  176. .pipe(replace(decorateSearchRegex, ""))
  177. .pipe(addModuleExports("BABYLON"))
  178. .pipe(uglify())
  179. .pipe(optimisejs())
  180. .pipe(gulp.dest(config.build.outputDirectory));
  181. });
  182. gulp.task("build", ["shaders"], function () {
  183. var filesToProcess = determineFilesToProcess("files");
  184. var directFilesToProcess = determineFilesToProcess("directFiles");
  185. return merge2(
  186. gulp.src(filesToProcess).
  187. pipe(expect.real({ errorOnFailure: true }, filesToProcess)),
  188. shadersStream,
  189. includeShadersStream,
  190. gulp.src(directFilesToProcess)
  191. )
  192. .pipe(concat(config.build.filename))
  193. .pipe(cleants())
  194. .pipe(replace(extendsSearchRegex, ""))
  195. .pipe(replace(decorateSearchRegex, ""))
  196. .pipe(addModuleExports("BABYLON"))
  197. .pipe(gulp.dest(config.build.outputDirectory))
  198. .pipe(rename(config.build.minFilename))
  199. .pipe(uglify())
  200. .pipe(optimisejs())
  201. .pipe(gulp.dest(config.build.outputDirectory));
  202. });
  203. /*
  204. * Compiles all typescript files and creating a js and a declaration file.
  205. */
  206. gulp.task("typescript-compile", function () {
  207. var tsResult = gulp.src(config.typescript)
  208. .pipe(sourcemaps.init())
  209. .pipe(tsProject());
  210. //If this gulp task is running on travis, file the build!
  211. if (process.env.TRAVIS) {
  212. var error = false;
  213. tsResult.on("error", function () {
  214. error = true;
  215. }).on("end", function () {
  216. if (error) {
  217. console.log("Typescript compile failed");
  218. process.exit(1);
  219. }
  220. });
  221. }
  222. return merge2([
  223. tsResult.dts
  224. .pipe(concat(config.build.declarationFilename))
  225. .pipe(addDtsExport("BABYLON", "babylonjs"))
  226. .pipe(gulp.dest(config.build.outputDirectory)),
  227. tsResult.js
  228. .pipe(sourcemaps.write("./",
  229. {
  230. includeContent: false,
  231. sourceRoot: (filePath) => {
  232. return "";
  233. }
  234. }))
  235. .pipe(gulp.dest(config.build.srcOutputDirectory))
  236. ])
  237. });
  238. /**
  239. * Helper methods to build external library (mat, post processes, ...).
  240. */
  241. var buildExternalLibraries = function (settings) {
  242. var tasks = settings.libraries.map(function (library) {
  243. return buildExternalLibrary(library, settings, false);
  244. });
  245. let mergedTasks = merge2(tasks);
  246. if (settings.build.buildAsModule) {
  247. mergedTasks.on("end", function () {
  248. //generate js file list
  249. let files = settings.libraries.filter(function (lib) {
  250. return !lib.doNotIncludeInBundle;
  251. }).map(function (lib) {
  252. return config.build.outputDirectory + settings.build.distOutputDirectory + lib.output;
  253. });
  254. var outputDirectory = config.build.outputDirectory + settings.build.distOutputDirectory;
  255. let srcTask = gulp.src(files)
  256. .pipe(concat(settings.build.outputFilename + ".js"))
  257. .pipe(replace(extendsSearchRegex, ""))
  258. .pipe(replace(decorateSearchRegex, ""))
  259. .pipe(replace(referenceSearchRegex, ""))
  260. .pipe(addModuleExports(settings.build.moduleDeclaration, true, settings.build.extendsRoot))
  261. .pipe(gulp.dest(outputDirectory))
  262. .pipe(cleants())
  263. .pipe(rename({ extname: ".min.js" }))
  264. .pipe(uglify())
  265. .pipe(optimisejs())
  266. .pipe(gulp.dest(outputDirectory));
  267. let dtsFiles = files.map(function (filename) {
  268. return filename.replace(".js", ".d.ts");
  269. });
  270. let dtsTask = gulp.src(dtsFiles)
  271. .pipe(concat(settings.build.outputFilename + ".module.d.ts"))
  272. .pipe(replace(referenceSearchRegex, ""))
  273. .pipe(addDtsExport(settings.build.moduleDeclaration, settings.build.moduleName, true, settings.build.extendsRoot))
  274. .pipe(gulp.dest(outputDirectory));
  275. return merge2([srcTask, dtsTask]);
  276. });
  277. }
  278. return mergedTasks;
  279. }
  280. var buildExternalLibrary = function (library, settings, watch) {
  281. var tsProcess = gulp.src(library.files, { base: settings.build.srcOutputDirectory })
  282. .pipe(sourcemaps.init())
  283. .pipe(typescript(externalTsConfig));
  284. var includeShader = gulp.src(library.shadersIncludeFiles || [], { base: settings.build.srcOutputDirectory })
  285. .pipe(uncommentShader())
  286. .pipe(appendSrcToVariable("BABYLON.Effect.IncludesShadersStore", includeShadersName, library.output + ".include.fx"))
  287. .pipe(gulp.dest(settings.build.srcOutputDirectory));
  288. var shader = gulp.src(library.shaderFiles || [], { base: settings.build.srcOutputDirectory })
  289. .pipe(uncommentShader())
  290. .pipe(appendSrcToVariable("BABYLON.Effect.ShadersStore", shadersName, library.output + ".fx"))
  291. .pipe(gulp.dest(settings.build.srcOutputDirectory));
  292. var dev = tsProcess.js
  293. .pipe(sourcemaps.write("./", {
  294. includeContent: false,
  295. sourceRoot: (filePath) => {
  296. return "";
  297. }
  298. })).pipe(gulp.dest(settings.build.srcOutputDirectory));
  299. var outputDirectory = config.build.outputDirectory + settings.build.distOutputDirectory;
  300. var css = gulp.src(library.sassFiles || [])
  301. .pipe(sass().on("error", sass.logError))
  302. .pipe(concat(library.output.replace(".js", ".css")))
  303. .pipe(gulp.dest(outputDirectory));
  304. if (watch) {
  305. return merge2([shader, includeShader, dev, css]);
  306. }
  307. else {
  308. /*if (library.bundle) {
  309. // Don't remove extends and decorate functions
  310. var code = merge2([tsProcess.js, shader, includeShader])
  311. .pipe(concat(library.output));
  312. if (library.buildAsModule) {
  313. code = code.pipe(addModuleExports(library.moduleDeclaration, true))
  314. }
  315. code.pipe(gulp.dest(outputDirectory))
  316. .pipe(cleants())
  317. .pipe(rename({ extname: ".min.js" }))
  318. .pipe(uglify())
  319. .pipe(optimisejs())
  320. .pipe(gulp.dest(outputDirectory));
  321. } else {*/
  322. var code = merge2([tsProcess.js, shader, includeShader])
  323. .pipe(concat(library.output))
  324. if (library.buildAsModule) {
  325. code = code.pipe(replace(extendsSearchRegex, ""))
  326. .pipe(replace(decorateSearchRegex, ""))
  327. .pipe(addModuleExports(library.moduleDeclaration, true, library.extendsRoot))
  328. }
  329. code = code.pipe(gulp.dest(outputDirectory))
  330. .pipe(cleants())
  331. .pipe(rename({ extname: ".min.js" }))
  332. .pipe(uglify())
  333. .pipe(optimisejs())
  334. .pipe(gulp.dest(outputDirectory));
  335. /*}*/
  336. var dts = tsProcess.dts
  337. .pipe(concat(library.output))
  338. .pipe(replace(referenceSearchRegex, ""))
  339. .pipe(rename({ extname: ".d.ts" }))
  340. .pipe(gulp.dest(outputDirectory));
  341. var waitAll;
  342. if (library.buildAsModule) {
  343. var dts2 = tsProcess.dts
  344. .pipe(concat(library.output))
  345. .pipe(replace(referenceSearchRegex, ""))
  346. .pipe(addDtsExport(library.moduleDeclaration, library.moduleName, true, library.extendsRoot))
  347. .pipe(rename({ extname: ".module.d.ts" }))
  348. .pipe(gulp.dest(outputDirectory));
  349. waitAll = merge2([dev, code, css, dts, dts2]);
  350. } else {
  351. waitAll = merge2([dev, code, css, dts]);
  352. }
  353. if (library.webpack) {
  354. return waitAll.on("end", function () {
  355. return webpack(require(library.webpack))
  356. .pipe(rename(library.output.replace(".js", library.noBundleInName ? '.js' : ".bundle.js")))
  357. .pipe(addModuleExports(library.moduleDeclaration, false, false, true))
  358. .pipe(uglify())
  359. .pipe(optimisejs())
  360. .pipe(gulp.dest(outputDirectory))
  361. });
  362. }
  363. else {
  364. return waitAll;
  365. }
  366. }
  367. }
  368. /**
  369. * The default task, concat and min the main BJS files.
  370. */
  371. gulp.task("default", function (cb) {
  372. // runSequence("typescript-all", "intellisense", "tests-browserStack", cb);
  373. runSequence("typescript-all", "intellisense", cb);
  374. });
  375. gulp.task("mainBuild", function (cb) {
  376. runSequence("buildWorker", "build", cb);
  377. });
  378. /**
  379. * Build the releasable files.
  380. */
  381. gulp.task("typescript", function (cb) {
  382. runSequence("typescript-compile", "mainBuild", cb);
  383. });
  384. /**
  385. * Dynamic module creation.
  386. */
  387. config.modules.map(function (module) {
  388. gulp.task(module, function () {
  389. return buildExternalLibraries(config[module]);
  390. });
  391. });
  392. gulp.task("typescript-libraries", config.modules, function () {
  393. });
  394. /**
  395. * Dynamic custom configurations.
  396. */
  397. config.buildConfigurations.distributed.map(function (customConfiguration) {
  398. gulp.task(customConfiguration, function (cb) {
  399. config.build.currentConfig = customConfiguration;
  400. config.build.outputDirectory = config.build.outputCustomConfigurationsDirectory + "/" + customConfiguration;
  401. runSequence("typescript-compile", "build", cb);
  402. });
  403. });
  404. gulp.task("typescript-customConfigurations", function (cb) {
  405. runSequence(config.buildConfigurations.distributed, cb);
  406. });
  407. /**
  408. * Custom build with full path file control; used by profile.html
  409. */
  410. gulp.task("build-custom", function (cb) {
  411. runSequence("typescript-compile", "build", cb);
  412. });
  413. /**
  414. * Do it all.
  415. */
  416. gulp.task("typescript-all", function (cb) {
  417. runSequence("typescript", "typescript-libraries", "typescript-customConfigurations", cb);
  418. });
  419. /**
  420. * Watch ts files from typescript .
  421. */
  422. gulp.task("srcTscWatch", function () {
  423. // Reuse The TSC CLI from gulp to enable -w.
  424. process.argv[2] = "-w";
  425. process.argv[3] = "-p";
  426. process.argv[4] = "../../src/tsconfig.json";
  427. require("./node_modules/typescript/lib/tsc.js");
  428. });
  429. /**
  430. * Watch ts files and fire repective tasks.
  431. */
  432. gulp.task("watch", ["srcTscWatch"], function () {
  433. var interval = 1000;
  434. var tasks = [];
  435. config.modules.map(function (module) {
  436. config[module].libraries.map(function (library) {
  437. tasks.push(gulp.watch(library.files, { interval: interval }, function () {
  438. console.log(library.output);
  439. return buildExternalLibrary(library, config[module], true)
  440. .pipe(debug());
  441. }));
  442. tasks.push(gulp.watch(library.shaderFiles, { interval: interval }, function () {
  443. console.log(library.output);
  444. return buildExternalLibrary(library, config[module], true)
  445. .pipe(debug())
  446. }));
  447. tasks.push(gulp.watch(library.sassFiles, { interval: interval }, function () {
  448. console.log(library.output);
  449. return buildExternalLibrary(library, config[module], true)
  450. .pipe(debug())
  451. }));
  452. });
  453. });
  454. return tasks;
  455. });
  456. gulp.task("intellisense", function () {
  457. gulp.src(config.build.intellisenseSources)
  458. .pipe(concat(config.build.intellisenseFile))
  459. .pipe(replace(/^\s*_.*?$/gm, ""))
  460. .pipe(replace(/^\s*private .*?$/gm, ""))
  461. .pipe(replace(/^\s*public _.*?$/gm, ""))
  462. .pipe(replace(/^\s*protected .*?$/gm, ""))
  463. .pipe(replace(/^\s*public static _.*?$/gm, ""))
  464. .pipe(replace(/^\s*static _.*?$/gm, ""))
  465. .pipe(gulp.dest(config.build.playgroundDirectory));
  466. });
  467. /**
  468. * Embedded local dev env management.
  469. */
  470. gulp.task("deployLocalDev", function () {
  471. gulp.src("../../localDev/template/**.*")
  472. .pipe(gulp.dest("../../localDev/src/"));
  473. });
  474. /**
  475. * Embedded webserver for test convenience.
  476. */
  477. gulp.task("webserver", function () {
  478. var options = {
  479. port: 1338,
  480. livereload: false
  481. };
  482. if (commandLineOptions.public) {
  483. options.host = "0.0.0.0";
  484. }
  485. gulp.src("../../.").pipe(webserver(options));
  486. });
  487. /**
  488. * Combine Webserver and Watch as long as vscode does not handle multi tasks.
  489. */
  490. gulp.task("run", ["watch", "webserver"], function () {
  491. });
  492. gulp.task("tests-integration", function (done) {
  493. var kamaServerOptions = {
  494. configFile: __dirname + "/../../tests/validation/karma.conf.js",
  495. singleRun: false
  496. };
  497. var server = new karmaServer(kamaServerOptions, done);
  498. server.start();
  499. });
  500. gulp.task("tests-browserStack", function (done) {
  501. if (!process.env.TRAVIS) {
  502. done();
  503. return;
  504. }
  505. var kamaServerOptions = {
  506. configFile: __dirname + "/../../tests/validation/karma.conf.browserstack.js",
  507. singleRun: true
  508. };
  509. var server = new karmaServer(kamaServerOptions, done);
  510. server.start();
  511. });
  512. gulp.task("clean-JS-MAP", function () {
  513. return del([
  514. "../../src/**/*.js.map", "../../src/**/*.js"
  515. ], { force: true });
  516. });