fs/lib/ReaderCollection.js

  1. import AbstractReader from "./AbstractReader.js";
  2. /**
  3. * Resource Locator ReaderCollection
  4. *
  5. * @public
  6. * @class
  7. * @alias @ui5/fs/ReaderCollection
  8. * @extends @ui5/fs/AbstractReader
  9. */
  10. class ReaderCollection extends AbstractReader {
  11. /**
  12. * The constructor.
  13. *
  14. * @param {object} parameters Parameters
  15. * @param {string} parameters.name The collection name
  16. * @param {@ui5/fs/AbstractReader[]} [parameters.readers]
  17. * List of resource readers (all tried in parallel).
  18. * If none are provided, the collection will never return any results.
  19. */
  20. constructor({name, readers}) {
  21. super(name);
  22. // Remove any undefined (empty) readers from array
  23. this._readers = readers.filter(($) => $);
  24. }
  25. /**
  26. * Locates resources by glob.
  27. *
  28. * @private
  29. * @param {string|string[]} pattern glob pattern as string or an array of
  30. * glob patterns for virtual directory structure
  31. * @param {object} options glob options
  32. * @param {@ui5/fs/tracing.Trace} trace Trace instance
  33. * @returns {Promise<@ui5/fs/Resource[]>} Promise resolving to list of resources
  34. */
  35. _byGlob(pattern, options, trace) {
  36. return Promise.all(this._readers.map(function(resourceLocator) {
  37. return resourceLocator._byGlob(pattern, options, trace);
  38. })).then((result) => {
  39. trace.collection(this._name);
  40. return Array.prototype.concat.apply([], result);
  41. });
  42. }
  43. /**
  44. * Locates resources by path.
  45. *
  46. * @private
  47. * @param {string} virPath Virtual path
  48. * @param {object} options Options
  49. * @param {@ui5/fs/tracing.Trace} trace Trace instance
  50. * @returns {Promise<@ui5/fs/Resource|null>}
  51. * Promise resolving to a single resource or <code>null</code> if no resource is found
  52. */
  53. _byPath(virPath, options, trace) {
  54. const that = this;
  55. const resourceLocatorCount = this._readers.length;
  56. let resolveCount = 0;
  57. if (resourceLocatorCount === 0) {
  58. // Short-circuit if there are no readers (Promise.race does not settle for empty arrays)
  59. trace.collection(that._name);
  60. return Promise.resolve(null);
  61. }
  62. // Using Promise.race to deliver files that can be found as fast as possible
  63. return Promise.race(this._readers.map(function(resourceLocator) {
  64. return resourceLocator._byPath(virPath, options, trace).then(function(resource) {
  65. return new Promise(function(resolve, reject) {
  66. trace.collection(that._name);
  67. resolveCount++;
  68. if (resource) {
  69. resource.pushCollection(that._name);
  70. resolve(resource);
  71. } else if (resolveCount === resourceLocatorCount) {
  72. resolve(null);
  73. }
  74. });
  75. });
  76. }));
  77. }
  78. }
  79. export default ReaderCollection;