@dereekb/dbx-firebase 12.5.0 → 12.5.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -103,45 +103,52 @@ export function storageFileUploadFiles(input) {
103
103
  // emit new progress event
104
104
  _emitEvent();
105
105
  }
106
- function _markFileUploadDone(fileIndex, error) {
106
+ function _markFileUploadDone(fileIndex, finalError) {
107
107
  doneFileIndexes.add(fileIndex); // add to done file indexes
108
108
  activeFileIndexes.delete(fileIndex); // remove from active file indexes if it exists
109
109
  incompleteFileFileIndexes.delete(fileIndex); // remove from incomplete file indexes
110
110
  // update details
111
111
  allFilesAndDetails[fileIndex].endTime = new Date();
112
+ const error = finalError ?? allFilesAndDetails[fileIndex].error;
112
113
  allFilesAndDetails[fileIndex].success = !error;
113
114
  allFilesAndDetails[fileIndex].error = error;
114
115
  }
115
116
  function updateUploadProgress(input) {
116
- const { index: fileIndex, nextProgress, fileUploadTaskDone, error } = input;
117
- let nextOverallProgress = latestOverallProgress;
118
- const nextProgressPercent = fileUploadTaskDone ? 100 : (nextProgress?.progress ?? 0) * 100;
119
- // update the overall progress percentage
120
- if (nextProgressPercent) {
121
- // update the overall percentage
122
- const previousProgress = allFilesAndLatestProgress[fileIndex];
123
- const previousProgressPercent = previousProgress?.progress != null ? previousProgress.progress * 100 : 0;
124
- const progressPercentChange = nextProgressPercent - previousProgressPercent;
125
- // increase overall progress by the change
126
- nextOverallProgress += progressPercentChange * overallProgressPerCompletedFile;
127
- }
128
- // update the file progress
129
- if (nextProgress) {
130
- // update the latest FileProgress
131
- allFilesAndLatestProgress[fileIndex] = nextProgress;
132
- // only set fileRef once
133
- if (!allFilesAndDetails[fileIndex].fileRef) {
134
- allFilesAndDetails[fileIndex].fileRef = nextProgress.fileRef;
117
+ const { index: fileIndex, nextProgress, fileUploadTaskDone: inputFileUploadTaskDone, nonProgressError } = input;
118
+ const error = nonProgressError ?? nextProgress?.error;
119
+ const fileUploadTaskDone = inputFileUploadTaskDone ?? Boolean(error);
120
+ // the task may already be done, as after a progress-related error the complete task can get called.
121
+ const isTaskAlreadyDone = allFilesAndDetails[fileIndex].endTime != null;
122
+ if (!isTaskAlreadyDone) {
123
+ let nextOverallProgress = latestOverallProgress;
124
+ const nextProgressPercent = fileUploadTaskDone ? 100 : (nextProgress?.progress ?? 0) * 100;
125
+ // update the overall progress percentage
126
+ if (nextProgressPercent) {
127
+ // update the overall percentage
128
+ const previousProgress = allFilesAndLatestProgress[fileIndex];
129
+ const previousProgressPercent = previousProgress?.progress != null ? previousProgress.progress * 100 : 0;
130
+ const progressPercentChange = nextProgressPercent - previousProgressPercent;
131
+ // increase overall progress by the change
132
+ nextOverallProgress += progressPercentChange * overallProgressPerCompletedFile;
135
133
  }
134
+ // update the file progress
135
+ if (nextProgress) {
136
+ // update the latest FileProgress
137
+ allFilesAndLatestProgress[fileIndex] = nextProgress;
138
+ // only set fileRef once
139
+ if (!allFilesAndDetails[fileIndex].fileRef) {
140
+ allFilesAndDetails[fileIndex].fileRef = nextProgress.fileRef;
141
+ }
142
+ }
143
+ // if complete, update the indexes and details
144
+ if (fileUploadTaskDone) {
145
+ _markFileUploadDone(fileIndex, error);
146
+ }
147
+ // update the overall progress
148
+ latestOverallProgress = nextOverallProgress;
149
+ // emit the event to send it
150
+ _emitEvent(nextProgress);
136
151
  }
137
- // if complete, update the indexes and details
138
- if (fileUploadTaskDone) {
139
- _markFileUploadDone(fileIndex, error);
140
- }
141
- // update the overall progress
142
- latestOverallProgress = nextOverallProgress;
143
- // emit the event to send it
144
- _emitEvent(nextProgress);
145
152
  }
146
153
  function _emitEvent(nextProgress) {
147
154
  const isComplete = incompleteFileFileIndexes.size === 0;
@@ -200,7 +207,7 @@ export function storageFileUploadFiles(input) {
200
207
  // error occurred, update the progress with the error
201
208
  updateUploadProgress({
202
209
  index,
203
- error,
210
+ nonProgressError: error,
204
211
  fileUploadTaskDone: true
205
212
  });
206
213
  // always resolve, never reject
@@ -241,7 +248,6 @@ export function storageFileUploadFiles(input) {
241
248
  retriesAllowed: 0 // no retries allowed
242
249
  }).then(() => {
243
250
  // all tasks are finished. Complete the subscriber.
244
- console.log('complete');
245
251
  subscriber.complete();
246
252
  });
247
253
  }).pipe(shareReplay(1));
@@ -251,4 +257,4 @@ export function storageFileUploadFiles(input) {
251
257
  };
252
258
  return instance;
253
259
  }
254
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"storagefile.upload.handler.js","sourceRoot":"","sources":["../../../../../../../../packages/dbx-firebase/src/lib/modules/storagefile/container/storagefile.upload.handler.ts"],"names":[],"mappings":"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","sourcesContent":["import { catchError, map, Observable, of, shareReplay } from 'rxjs';\nimport { DbxFirebaseStorageFileUploadStoreFileProgress } from '../store';\nimport { DbxFirebaseStorageService } from '../../../storage/firebase.storage.service';\nimport { IndexNumber, Maybe, PercentDecimal, PercentNumber, PromiseOrValue, runAsyncTasksForValues, separateValues } from '@dereekb/util';\nimport { MultiSubscriptionObject } from '@dereekb/rxjs';\nimport { FirebaseStorageAccessorFile, StorageCustomMetadata, StoragePathInput, StorageUploadOptions, StorageUploadTask } from '@dereekb/firebase';\n\n/**\n * Creates a new observable for uploading a file.\n */\nexport type StorageFileUploadHandlerFunction = (file: File) => Promise<StorageFileUploadHandlerInstance>;\n\nexport interface StorageFileUploadHandlerInstance extends Pick<StorageUploadTask<FirebaseStorageAccessorFile>, 'taskRef' | 'pause' | 'resume' | 'cancel'> {\n  /**\n   * The upload observable.\n   *\n   * Must be subscribed to in order for the upload to begin.\n   */\n  readonly upload: Observable<DbxFirebaseStorageFileUploadStoreFileProgress>;\n}\n\n/**\n * Handles uploading files.\n */\nexport interface StorageFileUploadHandler {\n  /**\n   * Uploads a file, and returns the file progress as it is uploading.\n   */\n  readonly uploadFile: StorageFileUploadHandlerFunction;\n}\n\n/**\n * Configuration for a single file upload.\n */\nexport interface StorageFileUploadConfig {\n  /**\n   * Path for where to upload the file to\n   */\n  readonly storagePath: StoragePathInput;\n  /**\n   * Upload options for the file.\n   *\n   * Resumable is not supported.\n   */\n  readonly uploadOptions?: StorageFileUploadConfigOptions;\n  /**\n   * Custom metadata for the file.\n   *\n   * Is merged with uploadOptions's metadata.\n   */\n  readonly customMetadata?: StorageCustomMetadata;\n}\n\n/**\n * StorageFileUploadConfig upload options.\n */\nexport type StorageFileUploadConfigOptions = Omit<StorageUploadOptions, 'resumable'>;\n\n/**\n * Function used to generate file names for the uploaded files.\n *\n * If not set, the file name will be used as is.\n */\nexport type StorageFileUploadConfigFactory = (file: File) => PromiseOrValue<StorageFileUploadConfig>;\n\n/**\n * Configuration for StorageFileUploadHandler().\n */\nexport interface StorageFileUploadHandlerConfig {\n  readonly storageService: DbxFirebaseStorageService;\n  readonly storageFileUploadConfigFactory: StorageFileUploadConfigFactory;\n}\n\n/**\n * Default implementation of StorageFileUploadHandler.\n */\nexport function storageFileUploadHandler(config: StorageFileUploadHandlerConfig): StorageFileUploadHandler {\n  const { storageService, storageFileUploadConfigFactory } = config;\n\n  let resumable: Maybe<StorageUploadTask>;\n\n  return {\n    uploadFile: async (file) => {\n      const storageFileUploadConfig = await storageFileUploadConfigFactory(file);\n\n      const { storagePath } = storageFileUploadConfig;\n      const storageAccessorFile = storageService.file(storagePath);\n\n      const upload = new Observable<DbxFirebaseStorageFileUploadStoreFileProgress>((x) => {\n        if (storageAccessorFile.uploadResumable) {\n          resumable = storageAccessorFile.uploadResumable(file);\n\n          // subscribe to the event by piping this observable to it\n          resumable\n            .streamSnapshotEvents()\n            .pipe(\n              map((x) => {\n                const { bytesTransferred, totalBytes } = x;\n\n                const progress: DbxFirebaseStorageFileUploadStoreFileProgress = {\n                  file,\n                  fileRef: storageAccessorFile,\n                  uploadRef: resumable,\n                  bytesTransferred,\n                  totalBytes,\n                  progress: bytesTransferred / totalBytes\n                };\n\n                return progress;\n              })\n            )\n            .pipe(\n              catchError((error) => {\n                // if an error occurs, catch it and emit it as a progress\n                const progress: DbxFirebaseStorageFileUploadStoreFileProgress = {\n                  file,\n                  fileRef: storageAccessorFile,\n                  uploadRef: resumable,\n                  error,\n                  failed: true\n                };\n\n                return of(progress);\n              })\n            )\n            .subscribe(x);\n        } else {\n          throw new Error('uploadResumable() function was unavailable.');\n        }\n      }).pipe(shareReplay(1));\n\n      const instance: StorageFileUploadHandlerInstance = {\n        upload,\n        taskRef: storageAccessorFile,\n        pause: () => resumable?.pause() ?? false,\n        resume: () => resumable?.resume() ?? false,\n        cancel: () => resumable?.cancel() ?? false\n      };\n\n      return instance;\n    }\n  };\n}\n\n// MARK: Upload Files\nexport interface StorageFileUploadFilesInput {\n  readonly uploadHandler: StorageFileUploadHandler;\n  /**\n   * Files to upload\n   */\n  readonly files: File[];\n  /**\n   * The number of max parallel uploads to perform at a time.\n   *\n   * Defaults to 3\n   */\n  readonly maxParallelUploads?: number;\n}\n\nexport interface StorageFileUploadFilesInstance {\n  /**\n   * Cancels the upload of the remaining files.\n   */\n  cancel(): void;\n  /**\n   * The upload observable.\n   *\n   * Must be subscribed to in order for the upload to begin.\n   */\n  readonly upload: Observable<StorageFileUploadFilesEvent>;\n}\n\nexport interface StorageFileUploadFilesEvent {\n  /**\n   * All files being uploaded\n   */\n  readonly allFiles: File[];\n  /**\n   * Returns true if all files have been uploaded.\n   *\n   * The result value should be available.\n   */\n  readonly isComplete: boolean;\n  /**\n   * Returns true if the upload was canceled.\n   */\n  readonly isCanceled?: Maybe<boolean>;\n  /**\n   * The overall progress of all files being uploaded.\n   */\n  readonly overallProgress: PercentNumber;\n  /**\n   * The upload progress that triggered this event.\n   */\n  readonly uploadProgress?: Maybe<DbxFirebaseStorageFileUploadStoreFileProgress>;\n  /**\n   * The final result.\n   *\n   * Set when the final file has been uploaded or failed.\n   */\n  readonly result?: StorageFileUploadFilesFinalResult;\n  /**\n   * The number of files that are still uploading or queued for upload.\n   */\n  readonly incompleteFileCount: number;\n  /**\n   * The number of files that are active.\n   */\n  readonly activeFileCount: number;\n  /**\n   * The number of files that are done.\n   */\n  readonly doneFileCount: number;\n}\n\nexport interface StorageFileUploadFilesFinalResult {\n  readonly startTime: Date;\n  readonly endTime: Date;\n  readonly fileResults: StorageFileUploadFilesFinalFileResult[];\n  readonly successFileResults: StorageFileUploadFilesFinalFileResult[];\n  readonly errorFileResults: StorageFileUploadFilesFinalFileResult[];\n}\n\nexport interface StorageFileUploadFilesFinalFileResult {\n  /**\n   * The start time of the file upload.\n   */\n  readonly startTime: Date;\n  /**\n   * The end time of the file upload, or when it failed or was canceled.\n   */\n  readonly endTime: Date;\n  /**\n   * The file that was uploaded.\n   */\n  readonly file: File;\n  /**\n   * The accessor file for the file, if available.\n   *\n   * Is generally available if success is true.\n   */\n  readonly fileRef?: Maybe<FirebaseStorageAccessorFile>;\n  /**\n   * True if the file was uploaded successfully.\n   */\n  readonly success: boolean;\n  /**\n   * Error if the file failed to upload.\n   */\n  readonly error?: Maybe<unknown>;\n  /**\n   * True if the file upload was cancelled.\n   */\n  readonly canceled?: Maybe<boolean>;\n}\n\n/**\n * Uploads files using the provided upload handler and files.\n *\n * An observable is returned that emits the latest file events from any file that is being uploaded.\n *\n * @param input\n * @returns\n */\nexport function storageFileUploadFiles(input: StorageFileUploadFilesInput): StorageFileUploadFilesInstance {\n  const { uploadHandler, files, maxParallelUploads: inputMaxParallelUploads } = input;\n  const maxParallelTasks = inputMaxParallelUploads ?? 3;\n\n  const multiUploadsSubscriptionObject = new MultiSubscriptionObject();\n\n  // begin the upload for each file\n  const allFiles = Array.from(files);\n\n  // unsubscribe from all previous uploads\n  multiUploadsSubscriptionObject.unsub();\n\n  interface UpdateUploadProgressInput {\n    /**\n     * The file index number.\n     */\n    readonly index: IndexNumber;\n    /**\n     * The next progress event, if applicable.\n     */\n    readonly nextProgress?: Maybe<DbxFirebaseStorageFileUploadStoreFileProgress>;\n    /**\n     * An error that occured, if applicable.\n     */\n    readonly error?: Maybe<unknown>;\n    /**\n     * Passed as true when the upload task is done.\n     *\n     * Does not specify whether or not success was achieved or not.\n     */\n    readonly fileUploadTaskDone?: boolean;\n    /**\n     * True if the upload was canceled.\n     */\n    readonly canceled?: boolean;\n  }\n\n  interface FileUploadDetails {\n    readonly file: File;\n    /**\n     * The current upload instance for the file.\n     *\n     * Set if the file is currently uploading.\n     */\n    uploadInstance?: StorageFileUploadHandlerInstance;\n    fileRef?: Maybe<FirebaseStorageAccessorFile>;\n    startTime?: Date;\n    endTime?: Date;\n    success?: boolean;\n    canceled?: Maybe<boolean>;\n    error?: Maybe<unknown>;\n  }\n\n  const allFilesAndLatestProgress: Maybe<DbxFirebaseStorageFileUploadStoreFileProgress>[] = new Array(allFiles.length);\n  const allFilesAndDetails: FileUploadDetails[] = allFiles.map((file) => ({ file }));\n  const overallProgressPerCompletedFile: PercentDecimal = (1 / allFilesAndLatestProgress.length) as PercentDecimal;\n\n  /**\n   * Once set, any new file upload task that hits this will return an cancel failure.\n   */\n  let flaggedCancel = false;\n\n  const cancel = () => {\n    flaggedCancel = true;\n  };\n\n  const upload = new Observable<StorageFileUploadFilesEvent>((subscriber) => {\n    const overallStartTime = new Date();\n\n    const incompleteFileFileIndexes = new Set<IndexNumber>(allFiles.map((_, index) => index));\n    const activeFileIndexes = new Set<IndexNumber>();\n    const doneFileIndexes = new Set<IndexNumber>();\n    let latestOverallProgress: PercentNumber = 0;\n\n    function onStartFileUpload(index: IndexNumber, uploadInstance: StorageFileUploadHandlerInstance) {\n      activeFileIndexes.add(index);\n      allFilesAndDetails[index].startTime = new Date();\n      allFilesAndDetails[index].uploadInstance = uploadInstance;\n      allFilesAndDetails[index].fileRef = uploadInstance.taskRef;\n    }\n\n    function onStartFileUploadFlaggedCancelled(index: IndexNumber) {\n      allFilesAndDetails[index].startTime = new Date();\n\n      // immediately mark it done\n      _markFileUploadDone(index, true);\n\n      // emit new progress event\n      _emitEvent();\n    }\n\n    function _markFileUploadDone(fileIndex: IndexNumber, error?: Maybe<unknown>) {\n      doneFileIndexes.add(fileIndex); // add to done file indexes\n      activeFileIndexes.delete(fileIndex); // remove from active file indexes if it exists\n      incompleteFileFileIndexes.delete(fileIndex); // remove from incomplete file indexes\n\n      // update details\n      allFilesAndDetails[fileIndex].endTime = new Date();\n      allFilesAndDetails[fileIndex].success = !error;\n      allFilesAndDetails[fileIndex].error = error;\n    }\n\n    function updateUploadProgress(input: UpdateUploadProgressInput) {\n      const { index: fileIndex, nextProgress, fileUploadTaskDone, error } = input;\n\n      let nextOverallProgress = latestOverallProgress;\n      const nextProgressPercent = fileUploadTaskDone ? 100 : (nextProgress?.progress ?? 0) * 100;\n\n      // update the overall progress percentage\n      if (nextProgressPercent) {\n        // update the overall percentage\n        const previousProgress = allFilesAndLatestProgress[fileIndex];\n        const previousProgressPercent = previousProgress?.progress != null ? previousProgress.progress * 100 : 0;\n        const progressPercentChange = nextProgressPercent - previousProgressPercent;\n\n        // increase overall progress by the change\n        nextOverallProgress += progressPercentChange * overallProgressPerCompletedFile;\n      }\n\n      // update the file progress\n      if (nextProgress) {\n        // update the latest FileProgress\n        allFilesAndLatestProgress[fileIndex] = nextProgress;\n\n        // only set fileRef once\n        if (!allFilesAndDetails[fileIndex].fileRef) {\n          allFilesAndDetails[fileIndex].fileRef = nextProgress.fileRef;\n        }\n      }\n\n      // if complete, update the indexes and details\n      if (fileUploadTaskDone) {\n        _markFileUploadDone(fileIndex, error);\n      }\n\n      // update the overall progress\n      latestOverallProgress = nextOverallProgress;\n\n      // emit the event to send it\n      _emitEvent(nextProgress);\n    }\n\n    function _emitEvent(nextProgress?: Maybe<DbxFirebaseStorageFileUploadStoreFileProgress>) {\n      const isComplete = incompleteFileFileIndexes.size === 0;\n\n      let overallProgress = latestOverallProgress;\n      let result: Maybe<StorageFileUploadFilesFinalResult> = undefined;\n\n      if (isComplete) {\n        overallProgress = 100; // set to 100%\n        const overallEndTime = new Date();\n\n        const fileResults = allFiles.map((file, index) => {\n          const result: StorageFileUploadFilesFinalFileResult = {\n            startTime: allFilesAndDetails[index].startTime as Date,\n            endTime: allFilesAndDetails[index].endTime as Date,\n            file,\n            fileRef: allFilesAndDetails[index].fileRef,\n            success: !allFilesAndDetails[index].error,\n            error: allFilesAndDetails[index].error\n          };\n\n          return result;\n        });\n\n        const { included: successFileResults, excluded: errorFileResults } = separateValues(fileResults, (x) => x.success);\n\n        // all are done, set the result on the next event\n        result = {\n          startTime: overallStartTime,\n          endTime: overallEndTime,\n          successFileResults,\n          errorFileResults,\n          fileResults\n        };\n      }\n\n      const nextEvent: StorageFileUploadFilesEvent = {\n        allFiles,\n        isComplete,\n        overallProgress,\n        uploadProgress: nextProgress,\n        incompleteFileCount: incompleteFileFileIndexes.size,\n        activeFileCount: activeFileIndexes.size,\n        doneFileCount: doneFileIndexes.size,\n        result\n      };\n\n      subscriber.next(nextEvent);\n    }\n\n    async function runUploadTaskForFile([file, index]: readonly [File, IndexNumber]) {\n      if (flaggedCancel) {\n        onStartFileUploadFlaggedCancelled(index);\n        return Promise.resolve();\n      }\n\n      return new Promise<void>((resolve) => {\n        const updateFileUploadProgress = (nextProgress: DbxFirebaseStorageFileUploadStoreFileProgress) => {\n          // update the progress\n          updateUploadProgress({\n            index,\n            nextProgress\n          });\n        };\n\n        const updateFileUploadProgressWithUncaughtError = (error: unknown) => {\n          // error occurred, update the progress with the error\n          updateUploadProgress({\n            index,\n            error,\n            fileUploadTaskDone: true\n          });\n\n          // always resolve, never reject\n          resolve();\n        };\n\n        const completeFileUploadProgress = () => {\n          updateUploadProgress({\n            index,\n            fileUploadTaskDone: true\n          });\n\n          resolve();\n        };\n\n        // upload the file, subscribe to the progress\n        try {\n          uploadHandler\n            .uploadFile(file)\n            .then((uploadInstance) => {\n              // add to active file indexes\n              onStartFileUpload(index, uploadInstance);\n\n              const uploadSubscription = uploadInstance.upload.subscribe({\n                next: updateFileUploadProgress,\n                error: updateFileUploadProgressWithUncaughtError,\n                complete: completeFileUploadProgress\n              });\n\n              multiUploadsSubscriptionObject.addSubs(uploadSubscription);\n            })\n            .catch(updateFileUploadProgressWithUncaughtError);\n        } catch (error) {\n          updateFileUploadProgressWithUncaughtError(error);\n        }\n      });\n    }\n\n    // run upload task for each file\n    const fileTuples = allFiles.map((file, index) => [file, index] as const);\n\n    runAsyncTasksForValues(fileTuples, runUploadTaskForFile, {\n      maxParallelTasks,\n      retriesAllowed: 0 // no retries allowed\n    }).then(() => {\n      // all tasks are finished. Complete the subscriber.\n      console.log('complete');\n      subscriber.complete();\n    });\n  }).pipe(shareReplay(1));\n\n  const instance: StorageFileUploadFilesInstance = {\n    cancel,\n    upload\n  };\n\n  return instance;\n}\n"]}
260
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"storagefile.upload.handler.js","sourceRoot":"","sources":["../../../../../../../../packages/dbx-firebase/src/lib/modules/storagefile/container/storagefile.upload.handler.ts"],"names":[],"mappings":"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","sourcesContent":["import { catchError, map, Observable, of, shareReplay } from 'rxjs';\nimport { DbxFirebaseStorageFileUploadStoreFileProgress } from '../store';\nimport { DbxFirebaseStorageService } from '../../../storage/firebase.storage.service';\nimport { IndexNumber, Maybe, PercentDecimal, PercentNumber, PromiseOrValue, runAsyncTasksForValues, separateValues } from '@dereekb/util';\nimport { MultiSubscriptionObject } from '@dereekb/rxjs';\nimport { FirebaseStorageAccessorFile, StorageCustomMetadata, StoragePathInput, StorageUploadOptions, StorageUploadTask } from '@dereekb/firebase';\n\n/**\n * Creates a new observable for uploading a file.\n */\nexport type StorageFileUploadHandlerFunction = (file: File) => Promise<StorageFileUploadHandlerInstance>;\n\nexport interface StorageFileUploadHandlerInstance extends Pick<StorageUploadTask<FirebaseStorageAccessorFile>, 'taskRef' | 'pause' | 'resume' | 'cancel'> {\n  /**\n   * The upload observable.\n   *\n   * Must be subscribed to in order for the upload to begin.\n   */\n  readonly upload: Observable<DbxFirebaseStorageFileUploadStoreFileProgress>;\n}\n\n/**\n * Handles uploading files.\n */\nexport interface StorageFileUploadHandler {\n  /**\n   * Uploads a file, and returns the file progress as it is uploading.\n   */\n  readonly uploadFile: StorageFileUploadHandlerFunction;\n}\n\n/**\n * Configuration for a single file upload.\n */\nexport interface StorageFileUploadConfig {\n  /**\n   * Path for where to upload the file to\n   */\n  readonly storagePath: StoragePathInput;\n  /**\n   * Upload options for the file.\n   *\n   * Resumable is not supported.\n   */\n  readonly uploadOptions?: StorageFileUploadConfigOptions;\n  /**\n   * Custom metadata for the file.\n   *\n   * Is merged with uploadOptions's metadata.\n   */\n  readonly customMetadata?: StorageCustomMetadata;\n}\n\n/**\n * StorageFileUploadConfig upload options.\n */\nexport type StorageFileUploadConfigOptions = Omit<StorageUploadOptions, 'resumable'>;\n\n/**\n * Function used to generate file names for the uploaded files.\n *\n * If not set, the file name will be used as is.\n */\nexport type StorageFileUploadConfigFactory = (file: File) => PromiseOrValue<StorageFileUploadConfig>;\n\n/**\n * Configuration for StorageFileUploadHandler().\n */\nexport interface StorageFileUploadHandlerConfig {\n  readonly storageService: DbxFirebaseStorageService;\n  readonly storageFileUploadConfigFactory: StorageFileUploadConfigFactory;\n}\n\n/**\n * Default implementation of StorageFileUploadHandler.\n */\nexport function storageFileUploadHandler(config: StorageFileUploadHandlerConfig): StorageFileUploadHandler {\n  const { storageService, storageFileUploadConfigFactory } = config;\n\n  let resumable: Maybe<StorageUploadTask>;\n\n  return {\n    uploadFile: async (file) => {\n      const storageFileUploadConfig = await storageFileUploadConfigFactory(file);\n\n      const { storagePath } = storageFileUploadConfig;\n      const storageAccessorFile = storageService.file(storagePath);\n\n      const upload = new Observable<DbxFirebaseStorageFileUploadStoreFileProgress>((x) => {\n        if (storageAccessorFile.uploadResumable) {\n          resumable = storageAccessorFile.uploadResumable(file);\n\n          // subscribe to the event by piping this observable to it\n          resumable\n            .streamSnapshotEvents()\n            .pipe(\n              map((x) => {\n                const { bytesTransferred, totalBytes } = x;\n\n                const progress: DbxFirebaseStorageFileUploadStoreFileProgress = {\n                  file,\n                  fileRef: storageAccessorFile,\n                  uploadRef: resumable,\n                  bytesTransferred,\n                  totalBytes,\n                  progress: bytesTransferred / totalBytes\n                };\n\n                return progress;\n              })\n            )\n            .pipe(\n              catchError((error) => {\n                // if an error occurs, catch it and emit it as a progress\n                const progress: DbxFirebaseStorageFileUploadStoreFileProgress = {\n                  file,\n                  fileRef: storageAccessorFile,\n                  uploadRef: resumable,\n                  error,\n                  failed: true\n                };\n\n                return of(progress);\n              })\n            )\n            .subscribe(x);\n        } else {\n          throw new Error('uploadResumable() function was unavailable.');\n        }\n      }).pipe(shareReplay(1));\n\n      const instance: StorageFileUploadHandlerInstance = {\n        upload,\n        taskRef: storageAccessorFile,\n        pause: () => resumable?.pause() ?? false,\n        resume: () => resumable?.resume() ?? false,\n        cancel: () => resumable?.cancel() ?? false\n      };\n\n      return instance;\n    }\n  };\n}\n\n// MARK: Upload Files\nexport interface StorageFileUploadFilesInput {\n  readonly uploadHandler: StorageFileUploadHandler;\n  /**\n   * Files to upload\n   */\n  readonly files: File[];\n  /**\n   * The number of max parallel uploads to perform at a time.\n   *\n   * Defaults to 3\n   */\n  readonly maxParallelUploads?: number;\n}\n\nexport interface StorageFileUploadFilesInstance {\n  /**\n   * Cancels the upload of the remaining files.\n   */\n  cancel(): void;\n  /**\n   * The upload observable.\n   *\n   * Must be subscribed to in order for the upload to begin.\n   */\n  readonly upload: Observable<StorageFileUploadFilesEvent>;\n}\n\nexport interface StorageFileUploadFilesEvent {\n  /**\n   * All files being uploaded\n   */\n  readonly allFiles: File[];\n  /**\n   * Returns true if all files have been uploaded.\n   *\n   * The result value should be available.\n   */\n  readonly isComplete: boolean;\n  /**\n   * Returns true if the upload was canceled.\n   */\n  readonly isCanceled?: Maybe<boolean>;\n  /**\n   * The overall progress of all files being uploaded.\n   */\n  readonly overallProgress: PercentNumber;\n  /**\n   * The upload progress that triggered this event.\n   */\n  readonly uploadProgress?: Maybe<DbxFirebaseStorageFileUploadStoreFileProgress>;\n  /**\n   * The final result.\n   *\n   * Set when the final file has been uploaded or failed.\n   */\n  readonly result?: StorageFileUploadFilesFinalResult;\n  /**\n   * The number of files that are still uploading or queued for upload.\n   */\n  readonly incompleteFileCount: number;\n  /**\n   * The number of files that are active.\n   */\n  readonly activeFileCount: number;\n  /**\n   * The number of files that are done.\n   */\n  readonly doneFileCount: number;\n}\n\nexport interface StorageFileUploadFilesFinalResult {\n  readonly startTime: Date;\n  readonly endTime: Date;\n  readonly fileResults: StorageFileUploadFilesFinalFileResult[];\n  readonly successFileResults: StorageFileUploadFilesFinalFileResult[];\n  readonly errorFileResults: StorageFileUploadFilesFinalFileResult[];\n}\n\nexport interface StorageFileUploadFilesFinalFileResult {\n  /**\n   * The start time of the file upload.\n   */\n  readonly startTime: Date;\n  /**\n   * The end time of the file upload, or when it failed or was canceled.\n   */\n  readonly endTime: Date;\n  /**\n   * The file that was uploaded.\n   */\n  readonly file: File;\n  /**\n   * The accessor file for the file, if available.\n   *\n   * Is generally available if success is true.\n   */\n  readonly fileRef?: Maybe<FirebaseStorageAccessorFile>;\n  /**\n   * True if the file was uploaded successfully.\n   */\n  readonly success: boolean;\n  /**\n   * Error if the file failed to upload.\n   */\n  readonly error?: Maybe<unknown>;\n  /**\n   * True if the file upload was cancelled.\n   */\n  readonly canceled?: Maybe<boolean>;\n}\n\n/**\n * Uploads files using the provided upload handler and files.\n *\n * An observable is returned that emits the latest file events from any file that is being uploaded.\n *\n * @param input\n * @returns\n */\nexport function storageFileUploadFiles(input: StorageFileUploadFilesInput): StorageFileUploadFilesInstance {\n  const { uploadHandler, files, maxParallelUploads: inputMaxParallelUploads } = input;\n  const maxParallelTasks = inputMaxParallelUploads ?? 3;\n\n  const multiUploadsSubscriptionObject = new MultiSubscriptionObject();\n\n  // begin the upload for each file\n  const allFiles = Array.from(files);\n\n  // unsubscribe from all previous uploads\n  multiUploadsSubscriptionObject.unsub();\n\n  interface UpdateUploadProgressInput {\n    /**\n     * The file index number.\n     */\n    readonly index: IndexNumber;\n    /**\n     * The next progress event, if applicable.\n     */\n    readonly nextProgress?: Maybe<DbxFirebaseStorageFileUploadStoreFileProgress>;\n    /**\n     * An error that occured outside of the nextProgress, if applicable.\n     *\n     * These are typically \"uncaught\" errors.\n     */\n    readonly nonProgressError?: Maybe<unknown>;\n    /**\n     * Passed as true when the upload task is done.\n     *\n     * This is also inferred as true when a progress error occurs for an upload.\n     *\n     * Does not specify whether or not success was achieved or not.\n     */\n    readonly fileUploadTaskDone?: boolean;\n    /**\n     * True if the upload was canceled.\n     */\n    readonly canceled?: boolean;\n  }\n\n  interface FileUploadDetails {\n    readonly file: File;\n    /**\n     * The current upload instance for the file.\n     *\n     * Set if the file is currently uploading.\n     */\n    uploadInstance?: StorageFileUploadHandlerInstance;\n    fileRef?: Maybe<FirebaseStorageAccessorFile>;\n    startTime?: Date;\n    endTime?: Date;\n    success?: boolean;\n    canceled?: Maybe<boolean>;\n    error?: Maybe<unknown>;\n  }\n\n  const allFilesAndLatestProgress: Maybe<DbxFirebaseStorageFileUploadStoreFileProgress>[] = new Array(allFiles.length);\n  const allFilesAndDetails: FileUploadDetails[] = allFiles.map((file) => ({ file }));\n  const overallProgressPerCompletedFile: PercentDecimal = (1 / allFilesAndLatestProgress.length) as PercentDecimal;\n\n  /**\n   * Once set, any new file upload task that hits this will return an cancel failure.\n   */\n  let flaggedCancel = false;\n\n  const cancel = () => {\n    flaggedCancel = true;\n  };\n\n  const upload = new Observable<StorageFileUploadFilesEvent>((subscriber) => {\n    const overallStartTime = new Date();\n\n    const incompleteFileFileIndexes = new Set<IndexNumber>(allFiles.map((_, index) => index));\n    const activeFileIndexes = new Set<IndexNumber>();\n    const doneFileIndexes = new Set<IndexNumber>();\n    let latestOverallProgress: PercentNumber = 0;\n\n    function onStartFileUpload(index: IndexNumber, uploadInstance: StorageFileUploadHandlerInstance) {\n      activeFileIndexes.add(index);\n      allFilesAndDetails[index].startTime = new Date();\n      allFilesAndDetails[index].uploadInstance = uploadInstance;\n      allFilesAndDetails[index].fileRef = uploadInstance.taskRef;\n    }\n\n    function onStartFileUploadFlaggedCancelled(index: IndexNumber) {\n      allFilesAndDetails[index].startTime = new Date();\n\n      // immediately mark it done\n      _markFileUploadDone(index, true);\n\n      // emit new progress event\n      _emitEvent();\n    }\n\n    function _markFileUploadDone(fileIndex: IndexNumber, finalError?: Maybe<unknown>) {\n      doneFileIndexes.add(fileIndex); // add to done file indexes\n      activeFileIndexes.delete(fileIndex); // remove from active file indexes if it exists\n      incompleteFileFileIndexes.delete(fileIndex); // remove from incomplete file indexes\n\n      // update details\n      allFilesAndDetails[fileIndex].endTime = new Date();\n\n      const error = finalError ?? allFilesAndDetails[fileIndex].error;\n      allFilesAndDetails[fileIndex].success = !error;\n      allFilesAndDetails[fileIndex].error = error;\n    }\n\n    function updateUploadProgress(input: UpdateUploadProgressInput) {\n      const { index: fileIndex, nextProgress, fileUploadTaskDone: inputFileUploadTaskDone, nonProgressError } = input;\n      const error = nonProgressError ?? nextProgress?.error;\n      const fileUploadTaskDone = inputFileUploadTaskDone ?? Boolean(error);\n\n      // the task may already be done, as after a progress-related error the complete task can get called.\n      const isTaskAlreadyDone = allFilesAndDetails[fileIndex].endTime != null;\n\n      if (!isTaskAlreadyDone) {\n        let nextOverallProgress = latestOverallProgress;\n        const nextProgressPercent = fileUploadTaskDone ? 100 : (nextProgress?.progress ?? 0) * 100;\n\n        // update the overall progress percentage\n        if (nextProgressPercent) {\n          // update the overall percentage\n          const previousProgress = allFilesAndLatestProgress[fileIndex];\n          const previousProgressPercent = previousProgress?.progress != null ? previousProgress.progress * 100 : 0;\n          const progressPercentChange = nextProgressPercent - previousProgressPercent;\n\n          // increase overall progress by the change\n          nextOverallProgress += progressPercentChange * overallProgressPerCompletedFile;\n        }\n\n        // update the file progress\n        if (nextProgress) {\n          // update the latest FileProgress\n          allFilesAndLatestProgress[fileIndex] = nextProgress;\n\n          // only set fileRef once\n          if (!allFilesAndDetails[fileIndex].fileRef) {\n            allFilesAndDetails[fileIndex].fileRef = nextProgress.fileRef;\n          }\n        }\n\n        // if complete, update the indexes and details\n        if (fileUploadTaskDone) {\n          _markFileUploadDone(fileIndex, error);\n        }\n\n        // update the overall progress\n        latestOverallProgress = nextOverallProgress;\n\n        // emit the event to send it\n        _emitEvent(nextProgress);\n      }\n    }\n\n    function _emitEvent(nextProgress?: Maybe<DbxFirebaseStorageFileUploadStoreFileProgress>) {\n      const isComplete = incompleteFileFileIndexes.size === 0;\n\n      let overallProgress = latestOverallProgress;\n      let result: Maybe<StorageFileUploadFilesFinalResult> = undefined;\n\n      if (isComplete) {\n        overallProgress = 100; // set to 100%\n        const overallEndTime = new Date();\n\n        const fileResults = allFiles.map((file, index) => {\n          const result: StorageFileUploadFilesFinalFileResult = {\n            startTime: allFilesAndDetails[index].startTime as Date,\n            endTime: allFilesAndDetails[index].endTime as Date,\n            file,\n            fileRef: allFilesAndDetails[index].fileRef,\n            success: !allFilesAndDetails[index].error,\n            error: allFilesAndDetails[index].error\n          };\n\n          return result;\n        });\n\n        const { included: successFileResults, excluded: errorFileResults } = separateValues(fileResults, (x) => x.success);\n\n        // all are done, set the result on the next event\n        result = {\n          startTime: overallStartTime,\n          endTime: overallEndTime,\n          successFileResults,\n          errorFileResults,\n          fileResults\n        };\n      }\n\n      const nextEvent: StorageFileUploadFilesEvent = {\n        allFiles,\n        isComplete,\n        overallProgress,\n        uploadProgress: nextProgress,\n        incompleteFileCount: incompleteFileFileIndexes.size,\n        activeFileCount: activeFileIndexes.size,\n        doneFileCount: doneFileIndexes.size,\n        result\n      };\n\n      subscriber.next(nextEvent);\n    }\n\n    async function runUploadTaskForFile([file, index]: readonly [File, IndexNumber]) {\n      if (flaggedCancel) {\n        onStartFileUploadFlaggedCancelled(index);\n        return Promise.resolve();\n      }\n\n      return new Promise<void>((resolve) => {\n        const updateFileUploadProgress = (nextProgress: DbxFirebaseStorageFileUploadStoreFileProgress) => {\n          // update the progress\n          updateUploadProgress({\n            index,\n            nextProgress\n          });\n        };\n\n        const updateFileUploadProgressWithUncaughtError = (error: unknown) => {\n          // error occurred, update the progress with the error\n          updateUploadProgress({\n            index,\n            nonProgressError: error,\n            fileUploadTaskDone: true\n          });\n\n          // always resolve, never reject\n          resolve();\n        };\n\n        const completeFileUploadProgress = () => {\n          updateUploadProgress({\n            index,\n            fileUploadTaskDone: true\n          });\n\n          resolve();\n        };\n\n        // upload the file, subscribe to the progress\n        try {\n          uploadHandler\n            .uploadFile(file)\n            .then((uploadInstance) => {\n              // add to active file indexes\n              onStartFileUpload(index, uploadInstance);\n\n              const uploadSubscription = uploadInstance.upload.subscribe({\n                next: updateFileUploadProgress,\n                error: updateFileUploadProgressWithUncaughtError,\n                complete: completeFileUploadProgress\n              });\n\n              multiUploadsSubscriptionObject.addSubs(uploadSubscription);\n            })\n            .catch(updateFileUploadProgressWithUncaughtError);\n        } catch (error) {\n          updateFileUploadProgressWithUncaughtError(error);\n        }\n      });\n    }\n\n    // run upload task for each file\n    const fileTuples = allFiles.map((file, index) => [file, index] as const);\n\n    runAsyncTasksForValues(fileTuples, runUploadTaskForFile, {\n      maxParallelTasks,\n      retriesAllowed: 0 // no retries allowed\n    }).then(() => {\n      // all tasks are finished. Complete the subscriber.\n      subscriber.complete();\n    });\n  }).pipe(shareReplay(1));\n\n  const instance: StorageFileUploadFilesInstance = {\n    cancel,\n    upload\n  };\n\n  return instance;\n}\n"]}
@@ -5462,45 +5462,52 @@ function storageFileUploadFiles(input) {
5462
5462
  // emit new progress event
5463
5463
  _emitEvent();
5464
5464
  }
5465
- function _markFileUploadDone(fileIndex, error) {
5465
+ function _markFileUploadDone(fileIndex, finalError) {
5466
5466
  doneFileIndexes.add(fileIndex); // add to done file indexes
5467
5467
  activeFileIndexes.delete(fileIndex); // remove from active file indexes if it exists
5468
5468
  incompleteFileFileIndexes.delete(fileIndex); // remove from incomplete file indexes
5469
5469
  // update details
5470
5470
  allFilesAndDetails[fileIndex].endTime = new Date();
5471
+ const error = finalError ?? allFilesAndDetails[fileIndex].error;
5471
5472
  allFilesAndDetails[fileIndex].success = !error;
5472
5473
  allFilesAndDetails[fileIndex].error = error;
5473
5474
  }
5474
5475
  function updateUploadProgress(input) {
5475
- const { index: fileIndex, nextProgress, fileUploadTaskDone, error } = input;
5476
- let nextOverallProgress = latestOverallProgress;
5477
- const nextProgressPercent = fileUploadTaskDone ? 100 : (nextProgress?.progress ?? 0) * 100;
5478
- // update the overall progress percentage
5479
- if (nextProgressPercent) {
5480
- // update the overall percentage
5481
- const previousProgress = allFilesAndLatestProgress[fileIndex];
5482
- const previousProgressPercent = previousProgress?.progress != null ? previousProgress.progress * 100 : 0;
5483
- const progressPercentChange = nextProgressPercent - previousProgressPercent;
5484
- // increase overall progress by the change
5485
- nextOverallProgress += progressPercentChange * overallProgressPerCompletedFile;
5486
- }
5487
- // update the file progress
5488
- if (nextProgress) {
5489
- // update the latest FileProgress
5490
- allFilesAndLatestProgress[fileIndex] = nextProgress;
5491
- // only set fileRef once
5492
- if (!allFilesAndDetails[fileIndex].fileRef) {
5493
- allFilesAndDetails[fileIndex].fileRef = nextProgress.fileRef;
5476
+ const { index: fileIndex, nextProgress, fileUploadTaskDone: inputFileUploadTaskDone, nonProgressError } = input;
5477
+ const error = nonProgressError ?? nextProgress?.error;
5478
+ const fileUploadTaskDone = inputFileUploadTaskDone ?? Boolean(error);
5479
+ // the task may already be done, as after a progress-related error the complete task can get called.
5480
+ const isTaskAlreadyDone = allFilesAndDetails[fileIndex].endTime != null;
5481
+ if (!isTaskAlreadyDone) {
5482
+ let nextOverallProgress = latestOverallProgress;
5483
+ const nextProgressPercent = fileUploadTaskDone ? 100 : (nextProgress?.progress ?? 0) * 100;
5484
+ // update the overall progress percentage
5485
+ if (nextProgressPercent) {
5486
+ // update the overall percentage
5487
+ const previousProgress = allFilesAndLatestProgress[fileIndex];
5488
+ const previousProgressPercent = previousProgress?.progress != null ? previousProgress.progress * 100 : 0;
5489
+ const progressPercentChange = nextProgressPercent - previousProgressPercent;
5490
+ // increase overall progress by the change
5491
+ nextOverallProgress += progressPercentChange * overallProgressPerCompletedFile;
5494
5492
  }
5493
+ // update the file progress
5494
+ if (nextProgress) {
5495
+ // update the latest FileProgress
5496
+ allFilesAndLatestProgress[fileIndex] = nextProgress;
5497
+ // only set fileRef once
5498
+ if (!allFilesAndDetails[fileIndex].fileRef) {
5499
+ allFilesAndDetails[fileIndex].fileRef = nextProgress.fileRef;
5500
+ }
5501
+ }
5502
+ // if complete, update the indexes and details
5503
+ if (fileUploadTaskDone) {
5504
+ _markFileUploadDone(fileIndex, error);
5505
+ }
5506
+ // update the overall progress
5507
+ latestOverallProgress = nextOverallProgress;
5508
+ // emit the event to send it
5509
+ _emitEvent(nextProgress);
5495
5510
  }
5496
- // if complete, update the indexes and details
5497
- if (fileUploadTaskDone) {
5498
- _markFileUploadDone(fileIndex, error);
5499
- }
5500
- // update the overall progress
5501
- latestOverallProgress = nextOverallProgress;
5502
- // emit the event to send it
5503
- _emitEvent(nextProgress);
5504
5511
  }
5505
5512
  function _emitEvent(nextProgress) {
5506
5513
  const isComplete = incompleteFileFileIndexes.size === 0;
@@ -5559,7 +5566,7 @@ function storageFileUploadFiles(input) {
5559
5566
  // error occurred, update the progress with the error
5560
5567
  updateUploadProgress({
5561
5568
  index,
5562
- error,
5569
+ nonProgressError: error,
5563
5570
  fileUploadTaskDone: true
5564
5571
  });
5565
5572
  // always resolve, never reject
@@ -5600,7 +5607,6 @@ function storageFileUploadFiles(input) {
5600
5607
  retriesAllowed: 0 // no retries allowed
5601
5608
  }).then(() => {
5602
5609
  // all tasks are finished. Complete the subscriber.
5603
- console.log('complete');
5604
5610
  subscriber.complete();
5605
5611
  });
5606
5612
  }).pipe(shareReplay(1));