server/lib/server.js

  1. import express from "express";
  2. import portscanner from "portscanner";
  3. import MiddlewareManager from "./middleware/MiddlewareManager.js";
  4. import {createReaderCollection} from "@ui5/fs/resourceFactory";
  5. import ReaderCollectionPrioritized from "@ui5/fs/ReaderCollectionPrioritized";
  6. /**
  7. * @public
  8. * @module @ui5/server
  9. */
  10. /**
  11. * Returns a promise resolving by starting the server.
  12. *
  13. * @param {object} app The express application object
  14. * @param {number} port Desired port to listen to
  15. * @param {boolean} changePortIfInUse If true and the port is already in use, an unused port is searched
  16. * @param {boolean} acceptRemoteConnections If true, listens to remote connections and not only to localhost connections
  17. * @returns {Promise<object>} Returns an object containing server related information like (selected port, protocol)
  18. * @private
  19. */
  20. function _listen(app, port, changePortIfInUse, acceptRemoteConnections) {
  21. return new Promise(function(resolve, reject) {
  22. const options = {};
  23. if (!acceptRemoteConnections) {
  24. options.host = "localhost";
  25. }
  26. const host = options.host || "127.0.0.1";
  27. let portMax;
  28. if (changePortIfInUse) {
  29. portMax = port + 30;
  30. } else {
  31. portMax = port;
  32. }
  33. portscanner.findAPortNotInUse(port, portMax, host, function(error, foundPort) {
  34. if (error) {
  35. reject(error);
  36. return;
  37. }
  38. if (!foundPort) {
  39. if (changePortIfInUse) {
  40. const error = new Error(
  41. `EADDRINUSE: Could not find available ports between ${port} and ${portMax}.`);
  42. error.code = "EADDRINUSE";
  43. error.errno = "EADDRINUSE";
  44. error.address = host;
  45. error.port = portMax;
  46. reject(error);
  47. return;
  48. } else {
  49. const error = new Error(`EADDRINUSE: Port ${port} is already in use.`);
  50. error.code = "EADDRINUSE";
  51. error.errno = "EADDRINUSE";
  52. error.address = host;
  53. error.port = portMax;
  54. reject(error);
  55. return;
  56. }
  57. }
  58. options.port = foundPort;
  59. const server = app.listen(options, function() {
  60. resolve({port: options.port, server});
  61. });
  62. server.on("error", function(err) {
  63. reject(err);
  64. });
  65. });
  66. });
  67. }
  68. /**
  69. * Adds SSL support to an express application.
  70. *
  71. * @param {object} parameters
  72. * @param {object} parameters.app The original express application
  73. * @param {string} parameters.key Path to private key to be used for https
  74. * @param {string} parameters.cert Path to certificate to be used for for https
  75. * @returns {Promise<object>} The express application with SSL support
  76. * @private
  77. */
  78. async function _addSsl({app, key, cert}) {
  79. // Using spdy as http2 server as the native http2 implementation
  80. // from Node v8.4.0 doesn't seem to work with express
  81. const {default: spdy} = await import("spdy");
  82. return spdy.createServer({cert, key}, app);
  83. }
  84. /**
  85. * SAP target CSP middleware options
  86. *
  87. * @public
  88. * @typedef {object} module:@ui5/server.SAPTargetCSPOptions
  89. * @property {string} [defaultPolicy="sap-target-level-1"]
  90. * @property {string} [defaultPolicyIsReportOnly=true]
  91. * @property {string} [defaultPolicy2="sap-target-level-3"]
  92. * @property {string} [defaultPolicy2IsReportOnly=true]
  93. * @property {string[]} [ignorePaths=["test-resources/sap/ui/qunit/testrunner.html"]]
  94. */
  95. /**
  96. * Start a server for the given project (sub-)tree.
  97. *
  98. * @public
  99. * @param {@ui5/project/graph/ProjectGraph} graph Project graph
  100. * @param {object} options Options
  101. * @param {number} options.port Port to listen to
  102. * @param {boolean} [options.changePortIfInUse=false] If true, change the port if it is already in use
  103. * @param {boolean} [options.h2=false] Whether HTTP/2 should be used - defaults to <code>http</code>
  104. * @param {string} [options.key] Path to private key to be used for https
  105. * @param {string} [options.cert] Path to certificate to be used for for https
  106. * @param {boolean} [options.simpleIndex=false] Use a simplified view for the server directory listing
  107. * @param {boolean} [options.acceptRemoteConnections=false] If true, listens to remote connections and
  108. * not only to localhost connections
  109. * @param {boolean|module:@ui5/server.SAPTargetCSPOptions} [options.sendSAPTargetCSP=false]
  110. * If set to <code>true</code> or an object, then the default (or configured)
  111. * set of security policies that SAP and UI5 aim for (AKA 'target policies'),
  112. * are send for any requested <code>*.html</code> file
  113. * @param {boolean} [options.serveCSPReports=false] Enable CSP reports serving for request url
  114. * '/.ui5/csp/csp-reports.json'
  115. * @returns {Promise<object>} Promise resolving once the server is listening.
  116. * It resolves with an object containing the <code>port</code>,
  117. * <code>h2</code>-flag and a <code>close</code> function,
  118. * which can be used to stop the server.
  119. */
  120. export async function serve(graph, {
  121. port: requestedPort, changePortIfInUse = false, h2 = false, key, cert,
  122. acceptRemoteConnections = false, sendSAPTargetCSP = false, simpleIndex = false, serveCSPReports = false
  123. }) {
  124. const rootProject = graph.getRoot();
  125. const readers = [];
  126. await graph.traverseBreadthFirst(async function({project: dep}) {
  127. if (dep.getName() === rootProject.getName()) {
  128. // Ignore root project
  129. return;
  130. }
  131. readers.push(dep.getReader({style: "runtime"}));
  132. });
  133. const dependencies = createReaderCollection({
  134. name: `Dependency reader collection for project ${rootProject.getName()}`,
  135. readers
  136. });
  137. const rootReader = rootProject.getReader({style: "runtime"});
  138. // TODO change to ReaderCollection once duplicates are sorted out
  139. const combo = new ReaderCollectionPrioritized({
  140. name: "server - prioritize workspace over dependencies",
  141. readers: [rootReader, dependencies]
  142. });
  143. const resources = {
  144. rootProject: rootReader,
  145. dependencies: dependencies,
  146. all: combo
  147. };
  148. const middlewareManager = new MiddlewareManager({
  149. graph,
  150. rootProject,
  151. resources,
  152. options: {
  153. sendSAPTargetCSP,
  154. serveCSPReports,
  155. simpleIndex
  156. }
  157. });
  158. let app = express();
  159. await middlewareManager.applyMiddleware(app);
  160. if (h2) {
  161. app = await _addSsl({app, key, cert});
  162. }
  163. const {port, server} = await _listen(app, requestedPort, changePortIfInUse, acceptRemoteConnections);
  164. return {
  165. h2,
  166. port,
  167. close: function(callback) {
  168. server.close(callback);
  169. }
  170. };
  171. }