fs/lib/WriterCollection.js

  1. import AbstractReaderWriter from "./AbstractReaderWriter.js";
  2. import ReaderCollection from "./ReaderCollection.js";
  3. import escapeStringRegExp from "escape-string-regexp";
  4. /**
  5. * Resource Locator WriterCollection
  6. *
  7. * @public
  8. * @class
  9. * @alias @ui5/fs/WriterCollection
  10. * @extends @ui5/fs/AbstractReaderWriter
  11. */
  12. class WriterCollection extends AbstractReaderWriter {
  13. /**
  14. * The constructor.
  15. *
  16. * @param {object} parameters Parameters
  17. * @param {string} parameters.name The collection name
  18. * @param {object.<string, @ui5/fs/AbstractReaderWriter>} parameters.writerMapping
  19. * Mapping of virtual base paths to writers. Path are matched greedy
  20. *
  21. * @example
  22. * new WriterCollection({
  23. * name: "Writer Collection",
  24. * writerMapping: {
  25. * "/": writerA,
  26. * "/my/path/": writerB,
  27. * }
  28. * });
  29. */
  30. constructor({name, writerMapping}) {
  31. super(name);
  32. if (!writerMapping) {
  33. throw new Error(`Cannot create WriterCollection ${this._name}: Missing parameter 'writerMapping'`);
  34. }
  35. const basePaths = Object.keys(writerMapping);
  36. if (!basePaths.length) {
  37. throw new Error(`Cannot create WriterCollection ${this._name}: Empty parameter 'writerMapping'`);
  38. }
  39. // Create a regular expression (which is greedy by nature) from all paths to easily
  40. // find the correct writer for any given resource path
  41. this._basePathRegex = basePaths.sort().reduce((regex, basePath, idx) => {
  42. // Validate base path
  43. if (!basePath) {
  44. throw new Error(`Empty path in path mapping of WriterCollection ${this._name}`);
  45. }
  46. if (!basePath.startsWith("/")) {
  47. throw new Error(
  48. `Missing leading slash in path mapping '${basePath}' of WriterCollection ${this._name}`);
  49. }
  50. if (!basePath.endsWith("/")) {
  51. throw new Error(
  52. `Missing trailing slash in path mapping '${basePath}' of WriterCollection ${this._name}`);
  53. }
  54. return `${regex}(?:${escapeStringRegExp(basePath)})??`;
  55. }, "^(") + ")+.*?$";
  56. this._writerMapping = writerMapping;
  57. this._readerCollection = new ReaderCollection({
  58. name: `Reader collection of writer collection '${this._name}'`,
  59. readers: Object.values(writerMapping)
  60. });
  61. }
  62. /**
  63. * Locates resources by glob.
  64. *
  65. * @private
  66. * @param {string|string[]} pattern glob pattern as string or an array of
  67. * glob patterns for virtual directory structure
  68. * @param {object} options glob options
  69. * @param {@ui5/fs/tracing.Trace} trace Trace instance
  70. * @returns {Promise<@ui5/fs/Resource[]>} Promise resolving to list of resources
  71. */
  72. _byGlob(pattern, options, trace) {
  73. return this._readerCollection._byGlob(pattern, options, trace);
  74. }
  75. /**
  76. * Locates resources by path.
  77. *
  78. * @private
  79. * @param {string} virPath Virtual path
  80. * @param {object} options Options
  81. * @param {@ui5/fs/tracing.Trace} trace Trace instance
  82. * @returns {Promise<@ui5/fs/Resource>} Promise resolving to a single resource
  83. */
  84. _byPath(virPath, options, trace) {
  85. return this._readerCollection._byPath(virPath, options, trace);
  86. }
  87. /**
  88. * Writes the content of a resource to a path.
  89. *
  90. * @private
  91. * @param {@ui5/fs/Resource} resource The Resource to write
  92. * @param {object} [options] Write options, see above
  93. * @returns {Promise<undefined>} Promise resolving once data has been written
  94. */
  95. _write(resource, options) {
  96. const resourcePath = resource.getPath();
  97. const basePathMatch = resourcePath.match(this._basePathRegex);
  98. if (!basePathMatch || basePathMatch.length < 2) {
  99. throw new Error(
  100. `Failed to find a writer for resource with path ${resourcePath} in WriterCollection ${this._name}. ` +
  101. `Base paths handled by this collection are: ${Object.keys(this._writerMapping).join(", ")}`);
  102. }
  103. const writer = this._writerMapping[basePathMatch[1]];
  104. return writer._write(resource, options);
  105. }
  106. }
  107. export default WriterCollection;