1 | import { EventEmitter } from '@angular/core';
|
2 | import { Observable, Subject } from 'rxjs';
|
3 | import { mergeMap, finalize } from 'rxjs/operators';
|
4 | import { UploadStatus } from './interfaces';
|
5 | export function humanizeBytes(bytes) {
|
6 | if (bytes === 0) {
|
7 | return '0 Byte';
|
8 | }
|
9 | const k = 1024;
|
10 | const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB', 'PB'];
|
11 | const i = Math.floor(Math.log(bytes) / Math.log(k));
|
12 | return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];
|
13 | }
|
14 | export class NgUploaderService {
|
15 | constructor(concurrency = Number.POSITIVE_INFINITY, contentTypes = ['*'], maxUploads = Number.POSITIVE_INFINITY, maxFileSize = Number.POSITIVE_INFINITY) {
|
16 | this.queue = [];
|
17 | this.serviceEvents = new EventEmitter();
|
18 | this.uploadScheduler = new Subject();
|
19 | this.subs = [];
|
20 | this.contentTypes = contentTypes;
|
21 | this.maxUploads = maxUploads;
|
22 | this.maxFileSize = maxFileSize;
|
23 | this.uploadScheduler
|
24 | .pipe(mergeMap(upload => this.startUpload(upload), concurrency))
|
25 | .subscribe(uploadOutput => this.serviceEvents.emit(uploadOutput));
|
26 | }
|
27 | handleFiles(incomingFiles) {
|
28 | const allowedIncomingFiles = [].reduce.call(incomingFiles, (acc, checkFile, i) => {
|
29 | const futureQueueLength = acc.length + this.queue.length + 1;
|
30 | if (this.isContentTypeAllowed(checkFile.type) &&
|
31 | futureQueueLength <= this.maxUploads &&
|
32 | this.isFileSizeAllowed(checkFile.size)) {
|
33 | acc = acc.concat(checkFile);
|
34 | }
|
35 | else {
|
36 | const rejectedFile = this.makeUploadFile(checkFile, i);
|
37 | this.serviceEvents.emit({ type: 'rejected', file: rejectedFile });
|
38 | }
|
39 | return acc;
|
40 | }, []);
|
41 | this.queue.push(...[].map.call(allowedIncomingFiles, (file, i) => {
|
42 | const uploadFile = this.makeUploadFile(file, i);
|
43 | this.serviceEvents.emit({ type: 'addedToQueue', file: uploadFile });
|
44 | return uploadFile;
|
45 | }));
|
46 | this.serviceEvents.emit({ type: 'allAddedToQueue' });
|
47 | }
|
48 | initInputEvents(input) {
|
49 | return input.subscribe((event) => {
|
50 | switch (event.type) {
|
51 | case 'uploadFile':
|
52 | const uploadFileIndex = this.queue.findIndex(file => file === event.file);
|
53 | if (uploadFileIndex !== -1 && event.file) {
|
54 | this.uploadScheduler.next({ file: this.queue[uploadFileIndex], event: event });
|
55 | }
|
56 | break;
|
57 | case 'uploadAll':
|
58 | const files = this.queue.filter(file => file.progress.status === UploadStatus.Queue);
|
59 | files.forEach(file => this.uploadScheduler.next({ file: file, event: event }));
|
60 | break;
|
61 | case 'cancel':
|
62 | const id = event.id || null;
|
63 | if (!id) {
|
64 | return;
|
65 | }
|
66 | const subs = this.subs.filter(sub => sub.id === id);
|
67 | subs.forEach(sub => {
|
68 | if (sub.sub) {
|
69 | sub.sub.unsubscribe();
|
70 | const fileIndex = this.queue.findIndex(file => file.id === id);
|
71 | if (fileIndex !== -1) {
|
72 | this.queue[fileIndex].progress.status = UploadStatus.Cancelled;
|
73 | this.serviceEvents.emit({ type: 'cancelled', file: this.queue[fileIndex] });
|
74 | }
|
75 | }
|
76 | });
|
77 | break;
|
78 | case 'cancelAll':
|
79 | this.subs.forEach(sub => {
|
80 | if (sub.sub) {
|
81 | sub.sub.unsubscribe();
|
82 | }
|
83 | const file = this.queue.find(uploadFile => uploadFile.id === sub.id);
|
84 | if (file) {
|
85 | file.progress.status = UploadStatus.Cancelled;
|
86 | this.serviceEvents.emit({ type: 'cancelled', file: file });
|
87 | }
|
88 | });
|
89 | break;
|
90 | case 'remove':
|
91 | if (!event.id) {
|
92 | return;
|
93 | }
|
94 | const i = this.queue.findIndex(file => file.id === event.id);
|
95 | if (i !== -1) {
|
96 | const file = this.queue[i];
|
97 | this.queue.splice(i, 1);
|
98 | this.serviceEvents.emit({ type: 'removed', file: file });
|
99 | }
|
100 | break;
|
101 | case 'removeAll':
|
102 | if (this.queue.length) {
|
103 | this.queue = [];
|
104 | this.serviceEvents.emit({ type: 'removedAll' });
|
105 | }
|
106 | break;
|
107 | }
|
108 | });
|
109 | }
|
110 | startUpload(upload) {
|
111 | return new Observable(observer => {
|
112 | const sub = this.uploadFile(upload.file, upload.event)
|
113 | .pipe(finalize(() => {
|
114 | if (!observer.closed) {
|
115 | observer.complete();
|
116 | }
|
117 | }))
|
118 | .subscribe(output => {
|
119 | observer.next(output);
|
120 | }, err => {
|
121 | observer.error(err);
|
122 | observer.complete();
|
123 | }, () => {
|
124 | observer.complete();
|
125 | });
|
126 | this.subs.push({ id: upload.file.id, sub: sub });
|
127 | });
|
128 | }
|
129 | uploadFile(file, event) {
|
130 | return new Observable(observer => {
|
131 | const url = event.url || '';
|
132 | const method = event.method || 'POST';
|
133 | const data = event.data || {};
|
134 | const headers = event.headers || {};
|
135 | const xhr = new XMLHttpRequest();
|
136 | const time = new Date().getTime();
|
137 | let progressStartTime = (file.progress.data && file.progress.data.startTime) || time;
|
138 | let speed = 0;
|
139 | let eta = null;
|
140 | xhr.upload.addEventListener('progress', (e) => {
|
141 | if (e.lengthComputable) {
|
142 | const percentage = Math.round((e.loaded * 100) / e.total);
|
143 | const diff = new Date().getTime() - time;
|
144 | speed = Math.round((e.loaded / diff) * 1000);
|
145 | progressStartTime = (file.progress.data && file.progress.data.startTime) || new Date().getTime();
|
146 | eta = Math.ceil((e.total - e.loaded) / speed);
|
147 | file.progress = {
|
148 | status: UploadStatus.Uploading,
|
149 | data: {
|
150 | percentage: percentage,
|
151 | speed: speed,
|
152 | speedHuman: `${humanizeBytes(speed)}/s`,
|
153 | startTime: progressStartTime,
|
154 | endTime: null,
|
155 | eta: eta,
|
156 | etaHuman: this.secondsToHuman(eta)
|
157 | }
|
158 | };
|
159 | observer.next({ type: 'uploading', file: file });
|
160 | }
|
161 | }, false);
|
162 | xhr.upload.addEventListener('error', (e) => {
|
163 | observer.error(e);
|
164 | observer.complete();
|
165 | });
|
166 | xhr.onreadystatechange = () => {
|
167 | if (xhr.readyState === XMLHttpRequest.DONE) {
|
168 | const speedAverage = Math.round((file.size / (new Date().getTime() - progressStartTime)) * 1000);
|
169 | file.progress = {
|
170 | status: UploadStatus.Done,
|
171 | data: {
|
172 | percentage: 100,
|
173 | speed: speedAverage,
|
174 | speedHuman: `${humanizeBytes(speedAverage)}/s`,
|
175 | startTime: progressStartTime,
|
176 | endTime: new Date().getTime(),
|
177 | eta: eta,
|
178 | etaHuman: this.secondsToHuman(eta || 0)
|
179 | }
|
180 | };
|
181 | file.responseStatus = xhr.status;
|
182 | try {
|
183 | file.response = JSON.parse(xhr.response);
|
184 | }
|
185 | catch (e) {
|
186 | file.response = xhr.response;
|
187 | }
|
188 | file.responseHeaders = this.parseResponseHeaders(xhr.getAllResponseHeaders());
|
189 | observer.next({ type: 'done', file: file });
|
190 | observer.complete();
|
191 | }
|
192 | };
|
193 | xhr.open(method, url, true);
|
194 | xhr.withCredentials = event.withCredentials ? true : false;
|
195 | try {
|
196 | const uploadFile = file.nativeFile;
|
197 | const uploadIndex = this.queue.findIndex(outFile => outFile.nativeFile === uploadFile);
|
198 | if (this.queue[uploadIndex].progress.status === UploadStatus.Cancelled) {
|
199 | observer.complete();
|
200 | }
|
201 | Object.keys(headers).forEach(key => xhr.setRequestHeader(key, headers[key]));
|
202 | let bodyToSend;
|
203 | if (event.includeWebKitFormBoundary !== false) {
|
204 | Object.keys(data).forEach(key => file.form.append(key, data[key]));
|
205 | file.form.append(event.fieldName || 'file', uploadFile, uploadFile.name);
|
206 | bodyToSend = file.form;
|
207 | }
|
208 | else {
|
209 | bodyToSend = uploadFile;
|
210 | }
|
211 | this.serviceEvents.emit({ type: 'start', file: file });
|
212 | xhr.send(bodyToSend);
|
213 | }
|
214 | catch (e) {
|
215 | observer.complete();
|
216 | }
|
217 | return () => {
|
218 | xhr.abort();
|
219 | };
|
220 | });
|
221 | }
|
222 | secondsToHuman(sec) {
|
223 | return new Date(sec * 1000).toISOString().substr(11, 8);
|
224 | }
|
225 | generateId() {
|
226 | return Math.random().toString(36).substring(7);
|
227 | }
|
228 | setContentTypes(contentTypes) {
|
229 | if (typeof contentTypes !== 'undefined' && contentTypes instanceof Array) {
|
230 | if (contentTypes.find((type) => type === '*') !== undefined) {
|
231 | this.contentTypes = ['*'];
|
232 | }
|
233 | else {
|
234 | this.contentTypes = contentTypes;
|
235 | }
|
236 | return;
|
237 | }
|
238 | this.contentTypes = ['*'];
|
239 | }
|
240 | allContentTypesAllowed() {
|
241 | return this.contentTypes.find((type) => type === '*') !== undefined;
|
242 | }
|
243 | isContentTypeAllowed(mimetype) {
|
244 | if (this.allContentTypesAllowed()) {
|
245 | return true;
|
246 | }
|
247 | return this.contentTypes.find((type) => type === mimetype) !== undefined;
|
248 | }
|
249 | isFileSizeAllowed(fileSize) {
|
250 | if (!this.maxFileSize) {
|
251 | return true;
|
252 | }
|
253 | return fileSize <= this.maxFileSize;
|
254 | }
|
255 | makeUploadFile(file, index) {
|
256 | return {
|
257 | fileIndex: index,
|
258 | id: this.generateId(),
|
259 | name: file.name,
|
260 | size: file.size,
|
261 | type: file.type,
|
262 | form: new FormData(),
|
263 | progress: {
|
264 | status: UploadStatus.Queue,
|
265 | data: {
|
266 | percentage: 0,
|
267 | speed: 0,
|
268 | speedHuman: `${humanizeBytes(0)}/s`,
|
269 | startTime: null,
|
270 | endTime: null,
|
271 | eta: null,
|
272 | etaHuman: null
|
273 | }
|
274 | },
|
275 | lastModifiedDate: new Date(file.lastModified),
|
276 | sub: undefined,
|
277 | nativeFile: file
|
278 | };
|
279 | }
|
280 | parseResponseHeaders(httpHeaders) {
|
281 | if (!httpHeaders) {
|
282 | return;
|
283 | }
|
284 | return httpHeaders
|
285 | .split('\n')
|
286 | .map((x) => x.split(/: */, 2))
|
287 | .filter((x) => x[0])
|
288 | .reduce((acc, x) => {
|
289 | acc[x[0]] = x[1];
|
290 | return acc;
|
291 | }, {});
|
292 | }
|
293 | }
|
294 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ngx-uploader.class.js","sourceRoot":"../../../projects/ngx-uploader/src/","sources":["lib/ngx-uploader.class.ts"],"names":[],"mappings":"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","sourcesContent":["import { EventEmitter } from '@angular/core';\nimport { Observable, Subject, Subscription } from 'rxjs';\nimport { mergeMap, finalize } from 'rxjs/operators';\nimport { UploadFile, UploadOutput, UploadInput, UploadStatus, BlobFile } from './interfaces';\n\nexport function humanizeBytes(bytes: number): string {\n  if (bytes === 0) {\n    return '0 Byte';\n  }\n\n  const k = 1024;\n  const sizes: string[] = ['Bytes', 'KB', 'MB', 'GB', 'TB', 'PB'];\n  const i: number = Math.floor(Math.log(bytes) / Math.log(k));\n\n  return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];\n}\n\nexport class NgUploaderService {\n  queue: UploadFile[];\n  serviceEvents: EventEmitter<UploadOutput>;\n  uploadScheduler: Subject<{ file: UploadFile; event: UploadInput }>;\n  subs: { id: string; sub: Subscription }[];\n  contentTypes: string[];\n  maxUploads: number;\n  maxFileSize: number;\n\n  constructor(\n    concurrency: number = Number.POSITIVE_INFINITY,\n    contentTypes: string[] = ['*'],\n    maxUploads: number = Number.POSITIVE_INFINITY,\n    maxFileSize: number = Number.POSITIVE_INFINITY\n  ) {\n    this.queue = [];\n    this.serviceEvents = new EventEmitter<UploadOutput>();\n    this.uploadScheduler = new Subject();\n    this.subs = [];\n    this.contentTypes = contentTypes;\n    this.maxUploads = maxUploads;\n    this.maxFileSize = maxFileSize;\n\n    this.uploadScheduler\n      .pipe(mergeMap(upload => this.startUpload(upload), concurrency))\n      .subscribe(uploadOutput => this.serviceEvents.emit(uploadOutput));\n  }\n\n  handleFiles(incomingFiles: FileList): void {\n    const allowedIncomingFiles: File[] = [].reduce.call(\n      incomingFiles,\n      (acc: File[], checkFile: File, i: number) => {\n        const futureQueueLength = acc.length + this.queue.length + 1;\n        if (\n          this.isContentTypeAllowed(checkFile.type) &&\n          futureQueueLength <= this.maxUploads &&\n          this.isFileSizeAllowed(checkFile.size)\n        ) {\n          acc = acc.concat(checkFile);\n        } else {\n          const rejectedFile: UploadFile = this.makeUploadFile(checkFile, i);\n          this.serviceEvents.emit({ type: 'rejected', file: rejectedFile });\n        }\n\n        return acc;\n      },\n      []\n    );\n\n    this.queue.push(\n      ...[].map.call(allowedIncomingFiles, (file: File, i: number) => {\n        const uploadFile: UploadFile = this.makeUploadFile(file, i);\n        this.serviceEvents.emit({ type: 'addedToQueue', file: uploadFile });\n        return uploadFile;\n      })\n    );\n\n    this.serviceEvents.emit({ type: 'allAddedToQueue' });\n  }\n\n  initInputEvents(input: EventEmitter<UploadInput>): Subscription {\n    return input.subscribe((event: UploadInput) => {\n      switch (event.type) {\n        case 'uploadFile':\n          const uploadFileIndex = this.queue.findIndex(file => file === event.file);\n          if (uploadFileIndex !== -1 && event.file) {\n            this.uploadScheduler.next({ file: this.queue[uploadFileIndex], event: event });\n          }\n          break;\n        case 'uploadAll':\n          const files = this.queue.filter(file => file.progress.status === UploadStatus.Queue);\n          files.forEach(file => this.uploadScheduler.next({ file: file, event: event }));\n          break;\n        case 'cancel':\n          const id = event.id || null;\n          if (!id) {\n            return;\n          }\n          const subs = this.subs.filter(sub => sub.id === id);\n          subs.forEach(sub => {\n            if (sub.sub) {\n              sub.sub.unsubscribe();\n              const fileIndex = this.queue.findIndex(file => file.id === id);\n              if (fileIndex !== -1) {\n                this.queue[fileIndex].progress.status = UploadStatus.Cancelled;\n                this.serviceEvents.emit({ type: 'cancelled', file: this.queue[fileIndex] });\n              }\n            }\n          });\n          break;\n        case 'cancelAll':\n          this.subs.forEach(sub => {\n            if (sub.sub) {\n              sub.sub.unsubscribe();\n            }\n\n            const file = this.queue.find(uploadFile => uploadFile.id === sub.id);\n            if (file) {\n              file.progress.status = UploadStatus.Cancelled;\n              this.serviceEvents.emit({ type: 'cancelled', file: file });\n            }\n          });\n          break;\n        case 'remove':\n          if (!event.id) {\n            return;\n          }\n\n          const i = this.queue.findIndex(file => file.id === event.id);\n          if (i !== -1) {\n            const file = this.queue[i];\n            this.queue.splice(i, 1);\n            this.serviceEvents.emit({ type: 'removed', file: file });\n          }\n          break;\n        case 'removeAll':\n          if (this.queue.length) {\n            this.queue = [];\n            this.serviceEvents.emit({ type: 'removedAll' });\n          }\n          break;\n      }\n    });\n  }\n\n  startUpload(upload: { file: UploadFile; event: UploadInput }): Observable<UploadOutput> {\n    return new Observable(observer => {\n      const sub = this.uploadFile(upload.file, upload.event)\n        .pipe(\n          finalize(() => {\n            if (!observer.closed) {\n              observer.complete();\n            }\n          })\n        )\n        .subscribe(\n          output => {\n            observer.next(output);\n          },\n          err => {\n            observer.error(err);\n            observer.complete();\n          },\n          () => {\n            observer.complete();\n          }\n        );\n\n      this.subs.push({ id: upload.file.id, sub: sub });\n    });\n  }\n\n  uploadFile(file: UploadFile, event: UploadInput): Observable<UploadOutput> {\n    return new Observable(observer => {\n      const url = event.url || '';\n      const method = event.method || 'POST';\n      const data = event.data || {};\n      const headers = event.headers || {};\n\n      const xhr = new XMLHttpRequest();\n      const time: number = new Date().getTime();\n      let progressStartTime: number = (file.progress.data && file.progress.data.startTime) || time;\n      let speed = 0;\n      let eta: number | null = null;\n\n      xhr.upload.addEventListener(\n        'progress',\n        (e: ProgressEvent) => {\n          if (e.lengthComputable) {\n            const percentage = Math.round((e.loaded * 100) / e.total);\n            const diff = new Date().getTime() - time;\n            speed = Math.round((e.loaded / diff) * 1000);\n            progressStartTime = (file.progress.data && file.progress.data.startTime) || new Date().getTime();\n            eta = Math.ceil((e.total - e.loaded) / speed);\n\n            file.progress = {\n              status: UploadStatus.Uploading,\n              data: {\n                percentage: percentage,\n                speed: speed,\n                speedHuman: `${humanizeBytes(speed)}/s`,\n                startTime: progressStartTime,\n                endTime: null,\n                eta: eta,\n                etaHuman: this.secondsToHuman(eta)\n              }\n            };\n\n            observer.next({ type: 'uploading', file: file });\n          }\n        },\n        false\n      );\n\n      xhr.upload.addEventListener('error', (e: Event) => {\n        observer.error(e);\n        observer.complete();\n      });\n\n      xhr.onreadystatechange = () => {\n        if (xhr.readyState === XMLHttpRequest.DONE) {\n          const speedAverage = Math.round((file.size / (new Date().getTime() - progressStartTime)) * 1000);\n          file.progress = {\n            status: UploadStatus.Done,\n            data: {\n              percentage: 100,\n              speed: speedAverage,\n              speedHuman: `${humanizeBytes(speedAverage)}/s`,\n              startTime: progressStartTime,\n              endTime: new Date().getTime(),\n              eta: eta,\n              etaHuman: this.secondsToHuman(eta || 0)\n            }\n          };\n\n          file.responseStatus = xhr.status;\n\n          try {\n            file.response = JSON.parse(xhr.response);\n          } catch (e) {\n            file.response = xhr.response;\n          }\n\n          file.responseHeaders = this.parseResponseHeaders(xhr.getAllResponseHeaders());\n\n          observer.next({ type: 'done', file: file });\n\n          observer.complete();\n        }\n      };\n\n      xhr.open(method, url, true);\n      xhr.withCredentials = event.withCredentials ? true : false;\n\n      try {\n        const uploadFile = <BlobFile>file.nativeFile;\n        const uploadIndex = this.queue.findIndex(outFile => outFile.nativeFile === uploadFile);\n\n        if (this.queue[uploadIndex].progress.status === UploadStatus.Cancelled) {\n          observer.complete();\n        }\n\n        Object.keys(headers).forEach(key => xhr.setRequestHeader(key, headers[key]));\n\n        let bodyToSend: FormData | BlobFile;\n\n        if (event.includeWebKitFormBoundary !== false) {\n          Object.keys(data).forEach(key => file.form.append(key, data[key]));\n          file.form.append(event.fieldName || 'file', uploadFile, uploadFile.name);\n          bodyToSend = file.form;\n        } else {\n          bodyToSend = uploadFile;\n        }\n\n        this.serviceEvents.emit({ type: 'start', file: file });\n        xhr.send(bodyToSend);\n      } catch (e) {\n        observer.complete();\n      }\n\n      return () => {\n        xhr.abort();\n      };\n    });\n  }\n\n  secondsToHuman(sec: number): string {\n    return new Date(sec * 1000).toISOString().substr(11, 8);\n  }\n\n  generateId(): string {\n    return Math.random().toString(36).substring(7);\n  }\n\n  setContentTypes(contentTypes: string[]): void {\n    if (typeof contentTypes !== 'undefined' && contentTypes instanceof Array) {\n      if (contentTypes.find((type: string) => type === '*') !== undefined) {\n        this.contentTypes = ['*'];\n      } else {\n        this.contentTypes = contentTypes;\n      }\n      return;\n    }\n    this.contentTypes = ['*'];\n  }\n\n  allContentTypesAllowed(): boolean {\n    return this.contentTypes.find((type: string) => type === '*') !== undefined;\n  }\n\n  isContentTypeAllowed(mimetype: string): boolean {\n    if (this.allContentTypesAllowed()) {\n      return true;\n    }\n    return this.contentTypes.find((type: string) => type === mimetype) !== undefined;\n  }\n\n  isFileSizeAllowed(fileSize: number): boolean {\n    if (!this.maxFileSize) {\n      return true;\n    }\n    return fileSize <= this.maxFileSize;\n  }\n\n  makeUploadFile(file: File, index: number): UploadFile {\n    return {\n      fileIndex: index,\n      id: this.generateId(),\n      name: file.name,\n      size: file.size,\n      type: file.type,\n      form: new FormData(),\n      progress: {\n        status: UploadStatus.Queue,\n        data: {\n          percentage: 0,\n          speed: 0,\n          speedHuman: `${humanizeBytes(0)}/s`,\n          startTime: null,\n          endTime: null,\n          eta: null,\n          etaHuman: null\n        }\n      },\n      lastModifiedDate: new Date(file.lastModified),\n      sub: undefined,\n      nativeFile: file\n    };\n  }\n\n  private parseResponseHeaders(httpHeaders: string): { [key: string]: string } {\n    if (!httpHeaders) {\n      return;\n    }\n\n    return httpHeaders\n      .split('\\n')\n      .map((x: string) => x.split(/: */, 2))\n      .filter((x: string[]) => x[0])\n      .reduce((acc: { [key: string]: string }, x: string[]) => {\n        acc[x[0]] = x[1];\n        return acc;\n      }, {});\n  }\n}\n"]} |
\ | No newline at end of file |