server/lib/middleware/MiddlewareUtil.js

  1. import parseurl from "parseurl";
  2. import mime from "mime-types";
  3. import {
  4. createReaderCollection,
  5. createReaderCollectionPrioritized,
  6. createResource,
  7. createFilterReader,
  8. createLinkReader,
  9. createFlatReader
  10. } from "@ui5/fs/resourceFactory";
  11. /**
  12. * Convenience functions for UI5 Server middleware.
  13. * An instance of this class is passed to every standard UI5 Server middleware.
  14. * Custom middleware that define a specification version >= 2.0 will also receive an instance
  15. * of this class as part of the parameters of their create-middleware function.
  16. *
  17. * The set of functions that can be accessed by a custom middleware depends on the specification
  18. * version defined for the extension.
  19. *
  20. * @public
  21. * @class
  22. * @alias @ui5/server/middleware/MiddlewareUtil
  23. * @hideconstructor
  24. */
  25. class MiddlewareUtil {
  26. /**
  27. *
  28. * @param {object} parameters
  29. * @param {@ui5/project/graph/ProjectGraph} parameters.graph Relevant ProjectGraph
  30. * @param {@ui5/project/specifications/Project} parameters.project Project that is being served
  31. * @public
  32. */
  33. constructor({graph, project}) {
  34. if (!graph) {
  35. throw new Error(`Missing parameter "graph"`);
  36. }
  37. if (!project) {
  38. throw new Error(`Missing parameter "project"`);
  39. }
  40. this._graph = graph;
  41. this._project = project;
  42. }
  43. /**
  44. * Returns the [pathname]{@link https://developer.mozilla.org/en-US/docs/Web/API/URL/pathname}
  45. * of a given request. Any escape sequences will be decoded.
  46. * </br></br>
  47. * This method is only available to custom middleware extensions defining
  48. * <b>Specification Version 2.0 and above</b>.
  49. *
  50. * @param {object} req Request object
  51. * @returns {string} [Pathname]{@link https://developer.mozilla.org/en-US/docs/Web/API/URL/pathname}
  52. * of the given request
  53. * @public
  54. */
  55. getPathname(req) {
  56. let {pathname} = parseurl(req);
  57. pathname = decodeURIComponent(pathname);
  58. return pathname;
  59. }
  60. /**
  61. * MIME Info
  62. *
  63. * @example
  64. * const mimeInfo = {
  65. * "type": "text/html",
  66. * "charset": "utf-8",
  67. * "contentType": "text/html; charset=utf-8"
  68. * };
  69. *
  70. * @public
  71. * @typedef {object} MimeInfo
  72. * @property {string} type Detected content-type for the given resource path
  73. * @property {string} charset Default charset for the detected content-type
  74. * @property {string} contentType Calculated content-type header value
  75. * @memberof @ui5/server/middleware/MiddlewareUtil
  76. */
  77. /**
  78. * Returns MIME information derived from a given resource path.
  79. * </br></br>
  80. * This method is only available to custom middleware extensions defining
  81. * <b>Specification Version 2.0 and above</b>.
  82. *
  83. * @param {object} resourcePath
  84. * @returns {@ui5/server/middleware/MiddlewareUtil.MimeInfo}
  85. * @public
  86. */
  87. getMimeInfo(resourcePath) {
  88. const type = mime.lookup(resourcePath) || "application/octet-stream";
  89. const charset = mime.charset(type);
  90. return {
  91. type,
  92. charset,
  93. contentType: type + (charset ? "; charset=" + charset : "")
  94. };
  95. }
  96. /**
  97. * Specification Version-dependent [Project]{@link @ui5/project/specifications/Project} interface.
  98. * For details on individual functions, see [Project]{@link @ui5/project/specifications/Project}
  99. *
  100. * @public
  101. * @typedef {object} @ui5/server/middleware/MiddlewareUtil~ProjectInterface
  102. * @property {Function} getType Get the project type
  103. * @property {Function} getName Get the project name
  104. * @property {Function} getVersion Get the project version
  105. * @property {Function} getNamespace Get the project namespace
  106. * @property {Function} getRootReader Get the project rootReader
  107. * @property {Function} getReader Get the project reader, defaulting to "runtime" style instead of "buildtime"
  108. * @property {Function} getRootPath Get the local File System path of the project's root directory
  109. * @property {Function} getSourcePath Get the local File System path of the project's source directory
  110. * @property {Function} getCustomConfiguration Get the project Custom Configuration
  111. * @property {Function} isFrameworkProject Check whether the project is a UI5-Framework project
  112. * @property {Function} getFrameworkName Get the project's framework name configuration
  113. * @property {Function} getFrameworkVersion Get the project's framework version configuration
  114. * @property {Function} getFrameworkDependencies Get the project's framework dependencies configuration
  115. */
  116. /**
  117. * Retrieve a single project from the dependency graph
  118. *
  119. * </br></br>
  120. * This method is only available to custom server middleware extensions defining
  121. * <b>Specification Version 3.0 and above</b>.
  122. *
  123. * @param {string|@ui5/fs/Resource} [projectNameOrResource]
  124. * Name of the project to retrieve or a Resource instance to retrieve the associated project for.
  125. * Defaults to the name of the current root project
  126. * @returns {@ui5/server/middleware/MiddlewareUtil~ProjectInterface|undefined}
  127. * Specification Version-dependent interface to the Project instance or <code>undefined</code>
  128. * if the project name is unknown or the provided resource is not associated with any project.
  129. * @public
  130. */
  131. getProject(projectNameOrResource) {
  132. if (projectNameOrResource) {
  133. if (typeof projectNameOrResource === "string" || projectNameOrResource instanceof String) {
  134. // A project name has been provided
  135. return this._graph.getProject(projectNameOrResource);
  136. } else {
  137. // A Resource instance has been provided
  138. return projectNameOrResource.getProject();
  139. }
  140. }
  141. // No parameter has been provided, default to the root project
  142. return this._project;
  143. }
  144. /**
  145. * Retrieve a list of direct dependencies of a given project from the dependency graph.
  146. * Note that this list does not include transitive dependencies.
  147. *
  148. * </br></br>
  149. * This method is only available to custom server middleware extensions defining
  150. * <b>Specification Version 3.0 and above</b>.
  151. *
  152. * @param {string} [projectName] Name of the project to retrieve.
  153. * Defaults to the name of the current root project
  154. * @returns {string[]} Names of all direct dependencies
  155. * @throws {Error} If the requested project is unknown to the graph
  156. * @public
  157. */
  158. getDependencies(projectName) {
  159. return this._graph.getDependencies(projectName || this._project.getName());
  160. }
  161. /**
  162. * Specification Version-dependent set of [@ui5/fs/resourceFactory]{@link @ui5/fs/resourceFactory}
  163. * functions provided to middleware.
  164. * For details on individual functions, see [@ui5/fs/resourceFactory]{@link @ui5/fs/resourceFactory}
  165. *
  166. * @public
  167. * @typedef {object} @ui5/server/middleware/MiddlewareUtil~resourceFactory
  168. * @property {Function} createResource Creates a [Resource]{@link @ui5/fs/Resource}.
  169. * Accepts the same parameters as the [Resource]{@link @ui5/fs/Resource} constructor.
  170. * @property {Function} createReaderCollection Creates a reader collection:
  171. * [ReaderCollection]{@link @ui5/fs/ReaderCollection}
  172. * @property {Function} createReaderCollectionPrioritized Creates a prioritized reader collection:
  173. * [ReaderCollectionPrioritized]{@link @ui5/fs/ReaderCollectionPrioritized}
  174. * @property {Function} createFilterReader
  175. * Create a [Filter-Reader]{@link @ui5/fs/readers/Filter} with the given reader.
  176. * @property {Function} createLinkReader
  177. * Create a [Link-Reader]{@link @ui5/fs/readers/Filter} with the given reader.
  178. * @property {Function} createFlatReader Create a [Link-Reader]{@link @ui5/fs/readers/Link}
  179. * where all requests are prefixed with <code>/resources/<namespace></code>.
  180. */
  181. /**
  182. * Provides limited access to [@ui5/fs/resourceFactory]{@link @ui5/fs/resourceFactory} functions
  183. *
  184. * </br></br>
  185. * This attribute is only available to custom server middleware extensions defining
  186. * <b>Specification Version 3.0 and above</b>.
  187. *
  188. * @type {@ui5/server/middleware/MiddlewareUtil~resourceFactory}
  189. * @public
  190. */
  191. resourceFactory = {
  192. createResource,
  193. createReaderCollection,
  194. createReaderCollectionPrioritized,
  195. createFilterReader,
  196. createLinkReader,
  197. createFlatReader,
  198. };
  199. /**
  200. * Get an interface to an instance of this class that only provides those functions
  201. * that are supported by the given custom middleware extension specification version.
  202. *
  203. * @param {@ui5/project/specifications/SpecificationVersion} specVersion
  204. * SpecVersionComparator instance of the custom server middleware
  205. * @returns {object} An object with bound instance methods supported by the given specification version
  206. */
  207. getInterface(specVersion) {
  208. if (specVersion.lt("2.0")) {
  209. // Custom middleware defining specVersion <2.0 does not have access to any MiddlewareUtil API
  210. return undefined;
  211. }
  212. const baseInterface = {};
  213. bindFunctions(this, baseInterface, [
  214. "getPathname", "getMimeInfo"
  215. ]);
  216. if (specVersion.gte("3.0")) {
  217. // getProject function, returning an interfaced project instance
  218. baseInterface.getProject = (projectName) => {
  219. const project = this.getProject(projectName);
  220. const baseProjectInterface = {};
  221. bindFunctions(project, baseProjectInterface, [
  222. "getType", "getName", "getVersion", "getNamespace",
  223. "getRootReader", "getRootPath", "getSourcePath",
  224. "getCustomConfiguration", "isFrameworkProject", "getFrameworkName",
  225. "getFrameworkVersion", "getFrameworkDependencies"
  226. ]);
  227. // Project#getReader defaults to style "buildtime". However ui5-server uses
  228. // style "runtime". The main difference is that for some project types (like applications)
  229. // the /resources/<namespace> path prefix is omitted for "runtime". Also, no builder resource-
  230. // exclude configuration is applied.
  231. // Therefore default to style "runtime" here so that custom middleware will commonly work with
  232. // the same paths as ui5-server and no unexpected builder-excludes.
  233. baseProjectInterface.getReader = function(options = {style: "runtime"}) {
  234. return project.getReader(options);
  235. };
  236. return baseProjectInterface;
  237. };
  238. // getDependencies function, returning an array of project names
  239. baseInterface.getDependencies = (projectName) => {
  240. return this.getDependencies(projectName);
  241. };
  242. baseInterface.resourceFactory = Object.create(null);
  243. [
  244. // Once new functions get added, extract this array into a variable
  245. // and enhance based on spec version once new functions get added
  246. "createResource", "createReaderCollection", "createReaderCollectionPrioritized",
  247. "createFilterReader", "createLinkReader", "createFlatReader",
  248. ].forEach((factoryFunction) => {
  249. baseInterface.resourceFactory[factoryFunction] = this.resourceFactory[factoryFunction];
  250. });
  251. }
  252. return baseInterface;
  253. }
  254. }
  255. function bindFunctions(sourceObject, targetObject, funcNames) {
  256. funcNames.forEach((funcName) => {
  257. targetObject[funcName] = sourceObject[funcName].bind(sourceObject);
  258. });
  259. }
  260. export default MiddlewareUtil;