A pixel art painter geared specifically at NES pixel art. Includes export for .chr binary file as well as palette and namespace data.
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.

179 lines
4.8KB

  1. const package = require("./package.json");
  2. const vendors = require("./vendors.json");
  3. const exec = require('child_process').execSync;
  4. const fs = require("fs");
  5. const path = require("path");
  6. const express = require("express");
  7. const app = express();
  8. const http = require("http");
  9. const https = require("https");
  10. const sass = require("sass");
  11. const SASS_PATH = path.join(__dirname, "sass");
  12. const SASS_FILE = "style.scss";
  13. const watcher = require("chokidar").watch(SASS_PATH, {ignored: /[\/\\]\./, persistent: true});
  14. // --------------------------------------------------------
  15. // Environment options for the server.
  16. var production = process.env.PRODUCTION || false;
  17. var forceCSSRegen = process.env.FORCECSSREGEN || false;
  18. // NOTE: The default ports are blocked by default on linux without some hocus pocus.
  19. var port = process.env.NESPORT || 80;
  20. var portSSL = process.env.NESPORTSSL || 443;
  21. var sslKeyPath = process.env.SSLKEYPATH || null;
  22. var sslCertPath = process.env.SSLCERTPATH || null;
  23. var sslCaPath = process.env.SSLCAPATH || null;
  24. var css_output = ""; // Used to hold dynamic css.
  25. // -------------------------------------------------------
  26. // Simple helper function
  27. function debounce(func, delay, scope){
  28. var timeout = null;
  29. return function(){
  30. var context = scope || this;
  31. var args = arguments;
  32. clearTimeout(timeout);
  33. timeout = setTimeout(function(){
  34. func.apply(context, args);
  35. }, delay);
  36. };
  37. }
  38. var generateCSS = debounce(function(src, cb){
  39. sass.render({file: src}, (err, res)=>{
  40. if (err){
  41. cb("Failed to generate css - " + err.toString());
  42. } else {
  43. css_output = res.css.toString();
  44. cb();
  45. }
  46. });
  47. }, 1000);
  48. // -------------------------------------------------------
  49. // Configuring the current version of the application.
  50. function GitClean(){
  51. try {
  52. exec("git diff --quiet HEAD");
  53. } catch(e) {
  54. return false;
  55. }
  56. return true;
  57. }
  58. function GenVersion(){
  59. var v = package.version;
  60. var clean = GitClean();
  61. // Testing for a GIT repo... if not in a production environment.
  62. if (production === false){
  63. try{
  64. var res = exec("git rev-parse --abbrev-ref HEAD").toString();
  65. v += "-[" + res.trim();
  66. res = exec("git rev-parse HEAD").toString();
  67. v += ":" + res.substring(0, 5) + ((clean) ? "" : "-X") + "]";
  68. } catch(e) {
  69. if (v !== package.version){
  70. v += "]"; // If v doesn't match package.version, then assume that the first git call worked.
  71. }
  72. }
  73. }
  74. return v;
  75. }
  76. // ---------------------------------------------------
  77. // Configuring the express server.
  78. app.set('views', path.join(__dirname, "/views"));
  79. app.engine('html', require('ejs').renderFile);
  80. app.set('view engine', 'html');
  81. app.use("/app", express.static(path.join(__dirname, "/app")));
  82. vendors.modules.forEach((m) => {
  83. var p = path.normalize(path.join(__dirname, vendors.default_path, m.path));
  84. var url = path.join(vendors.url_path, m.url_path);
  85. app.use(url, express.static(p));
  86. });
  87. app.use("/app/css/nespaint.css", function(req, res){
  88. res.set("Content-Type", "text/css");
  89. res.send(new Buffer.from(css_output));
  90. });
  91. app.get('/', function(req, res){
  92. res.render('index.html', {version:GenVersion(), author:package.author});
  93. });
  94. // ----------------------------------------------------
  95. // Watching for any needed file updates (to minimize the need to restart the server.
  96. watcher.on('ready', () => {
  97. generateCSS(path.join(SASS_PATH, SASS_FILE), (err) => {
  98. if (err){
  99. console.log("ERROR: " + err);
  100. exit();
  101. } else {
  102. startServer();
  103. }
  104. });
  105. });
  106. watcher.on('change', (fpath) => {
  107. generateCSS(path.join(SASS_PATH, SASS_FILE), (err) => {
  108. if (err)
  109. console.log("WARNING: " + err);
  110. });
  111. });
  112. // --------------------------------------------------
  113. // Announce app version!
  114. console.log("NESPaint (v" + GenVersion() + ") Server");
  115. // --------------------------------------------------
  116. // KICK THE PIG!
  117. function startServer(){
  118. // Check if given SSL key and cert(s). If so, attempt to start an HTTPS server and
  119. // reroute HTTP requests to HTTPS.
  120. if (sslKeyPath !== null && sslCertPath !== null){
  121. try {
  122. var options = {
  123. key: fs.readFileSync(sslKeyPath),
  124. cert: fs.readFileSync(sslCertPath)
  125. }
  126. if (sslCaPath !== null)
  127. options.ca = fs.readFileSync(sslCaPath);
  128. app.use(function(req, res, next){
  129. if (req.secure){
  130. next();
  131. } else {
  132. res.redirect('https://' + req.headers.host + req.url);
  133. }
  134. });
  135. https.createServer(options, app).listen(portSSL, () => {
  136. console.log("HTTPS Listening on port " + portSSL + "!");
  137. });
  138. } catch (e) {
  139. console.log("WARNING: Failed to initialize HTTPS server. \"" + e.toString() + "\"");
  140. }
  141. }
  142. // Start the HTTP server.
  143. http.createServer(app).listen(port, () => {
  144. console.log("HTTP Listening on port " + port + "!");
  145. });
  146. }