speechrecorderng 3.4.3 → 3.4.5
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.
- package/esm2020/lib/action/action.mjs +3 -3
- package/esm2020/lib/audio/audio_display.mjs +9 -11
- package/esm2020/lib/audio/audio_player.mjs +4 -4
- package/esm2020/lib/audio/capture/capture.mjs +120 -106
- package/esm2020/lib/audio/io/stream.mjs +1 -1
- package/esm2020/lib/audio/net_audio_buffer.mjs +1 -1
- package/esm2020/lib/audio/playback/array_audio_buffer_source_node.mjs +2 -2
- package/esm2020/lib/audio/playback/audio_source_worklet_module_loader.mjs +2 -2
- package/esm2020/lib/audio/playback/inddb_audio_buffer_source_node.mjs +1 -1
- package/esm2020/lib/audio/ui/audio_canvas_layer_comp.mjs +7 -7
- package/esm2020/lib/audio/ui/audio_display_control.mjs +4 -4
- package/esm2020/lib/audio/ui/audio_display_scroll_pane.mjs +4 -4
- package/esm2020/lib/audio/ui/audiosignal.mjs +4 -4
- package/esm2020/lib/audio/ui/container.mjs +6 -6
- package/esm2020/lib/audio/ui/livelevel.mjs +4 -4
- package/esm2020/lib/audio/ui/scroll_pane_horizontal.mjs +4 -4
- package/esm2020/lib/audio/ui/sonagram.mjs +4 -4
- package/esm2020/lib/db/inddb.mjs +4 -4
- package/esm2020/lib/net/uploader.mjs +31 -28
- package/esm2020/lib/speechrecorder/project/project.service.mjs +7 -7
- package/esm2020/lib/speechrecorder/recordings/basic_recording.service.mjs +53 -51
- package/esm2020/lib/speechrecorder/recordings/recordings.service.mjs +124 -116
- package/esm2020/lib/speechrecorder/script/script.service.mjs +4 -4
- package/esm2020/lib/speechrecorder/session/audiorecorder.mjs +26 -27
- package/esm2020/lib/speechrecorder/session/basicrecorder.mjs +1 -1
- package/esm2020/lib/speechrecorder/session/controlpanel.mjs +34 -32
- package/esm2020/lib/speechrecorder/session/progress.mjs +4 -4
- package/esm2020/lib/speechrecorder/session/prompting.mjs +16 -16
- package/esm2020/lib/speechrecorder/session/recorder_combi_pane.mjs +4 -4
- package/esm2020/lib/speechrecorder/session/recording_list.mjs +4 -4
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-meta.component.mjs +4 -4
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-navi.component.mjs +6 -6
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-u-i.component.mjs +20 -19
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-view.component.mjs +43 -41
- package/esm2020/lib/speechrecorder/session/recordingfile/recordingfile-service.mjs +147 -137
- package/esm2020/lib/speechrecorder/session/session.service.mjs +15 -12
- package/esm2020/lib/speechrecorder/session/session_finished_dialog.mjs +4 -4
- package/esm2020/lib/speechrecorder/session/sessionmanager.mjs +4 -4
- package/esm2020/lib/speechrecorder/session/warning_bar.mjs +4 -4
- package/esm2020/lib/speechrecorder/spruploader.mjs +4 -4
- package/esm2020/lib/speechrecorder/startstopsignal/startstopsignal.mjs +1 -2
- package/esm2020/lib/speechrecorder/startstopsignal/ui/simpletrafficlight.mjs +4 -4
- package/esm2020/lib/speechrecorderng.component.mjs +6 -6
- package/esm2020/lib/speechrecorderng.module.mjs +5 -5
- package/esm2020/lib/spr.config.mjs +4 -4
- package/esm2020/lib/spr.module.version.mjs +2 -2
- package/esm2020/lib/ui/canvas_layer_comp.mjs +4 -4
- package/esm2020/lib/ui/message_dialog.mjs +4 -4
- package/esm2020/lib/ui/recordingitem_display.mjs +8 -8
- package/esm2020/lib/ui/responsive_component.mjs +4 -4
- package/esm2020/lib/utils/scrollIntoViewToBottom.mjs +4 -4
- package/fesm2015/speechrecorderng.mjs +744 -701
- package/fesm2015/speechrecorderng.mjs.map +1 -1
- package/fesm2020/speechrecorderng.mjs +744 -701
- package/fesm2020/speechrecorderng.mjs.map +1 -1
- package/lib/action/action.d.ts +1 -1
- package/lib/audio/audio_display.d.ts +1 -3
- package/lib/audio/capture/capture.d.ts +2 -0
- package/lib/audio/io/stream.d.ts +1 -1
- package/lib/net/uploader.d.ts +6 -6
- package/lib/speechrecorder/project/project.service.d.ts +4 -4
- package/lib/speechrecorder/script/script.service.d.ts +2 -2
- package/lib/speechrecorder/session/audiorecorder.d.ts +1 -2
- package/lib/speechrecorder/session/basicrecorder.d.ts +1 -1
- package/lib/speechrecorder/session/controlpanel.d.ts +1 -1
- package/lib/speechrecorder/session/session.service.d.ts +2 -2
- package/lib/spr.module.version.d.ts +1 -1
- package/package.json +14 -14
|
@@ -216,7 +216,7 @@ export class Uploader {
|
|
|
216
216
|
si += v.size;
|
|
217
217
|
}
|
|
218
218
|
else if (typeof v === 'string') {
|
|
219
|
-
// encode to
|
|
219
|
+
// encode to UTF-8 to get upload size
|
|
220
220
|
si += this.te.encode().length;
|
|
221
221
|
}
|
|
222
222
|
});
|
|
@@ -254,34 +254,37 @@ export class Uploader {
|
|
|
254
254
|
// pipe(timeout()) is not the same as xhr.timeout
|
|
255
255
|
let uploadedUpload = null;
|
|
256
256
|
//console.debug("Post upload: "+ul)
|
|
257
|
-
this.http.post(ul.url, ul.data, { withCredentials: this.withCredentials }).pipe(timeout(timeoVal)).subscribe(
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
// The backend returned an unsuccessful response code.
|
|
267
|
-
// The response body may contain clues as to what went wrong,
|
|
268
|
-
console.error(`Upload error: Server returned code ${err.status}`);
|
|
269
|
-
}
|
|
270
|
-
this.processError(ul);
|
|
271
|
-
}, () => {
|
|
272
|
-
//console.debug('Upload complete method called')
|
|
273
|
-
if (uploadedUpload) {
|
|
274
|
-
if (this.DEBUG_DELAY > 0) {
|
|
275
|
-
window.setTimeout(() => {
|
|
276
|
-
this.uploadDone(ul);
|
|
277
|
-
}, this.DEBUG_DELAY);
|
|
257
|
+
this.http.post(ul.url, ul.data, { withCredentials: this.withCredentials }).pipe(timeout(timeoVal)).subscribe({
|
|
258
|
+
next: (data) => {
|
|
259
|
+
uploadedUpload = ul;
|
|
260
|
+
//console.debug('Next method called for upload: '+uploadedUpload)
|
|
261
|
+
},
|
|
262
|
+
error: (err) => {
|
|
263
|
+
if (err.error instanceof Error) {
|
|
264
|
+
// A client-side or network error occurred. Handle it accordingly.
|
|
265
|
+
console.error('Upload error occurred:', err.error.message);
|
|
278
266
|
}
|
|
279
267
|
else {
|
|
280
|
-
|
|
268
|
+
// The backend returned an unsuccessful response code.
|
|
269
|
+
// The response body may contain clues as to what went wrong,
|
|
270
|
+
console.error(`Upload error: Server returned code ${err.status}`);
|
|
271
|
+
}
|
|
272
|
+
this.processError(ul);
|
|
273
|
+
}, complete: () => {
|
|
274
|
+
//console.debug('Upload complete method called')
|
|
275
|
+
if (uploadedUpload) {
|
|
276
|
+
if (this.DEBUG_DELAY > 0) {
|
|
277
|
+
window.setTimeout(() => {
|
|
278
|
+
this.uploadDone(ul);
|
|
279
|
+
}, this.DEBUG_DELAY);
|
|
280
|
+
}
|
|
281
|
+
else {
|
|
282
|
+
this.uploadDone(uploadedUpload);
|
|
283
|
+
}
|
|
284
|
+
}
|
|
285
|
+
else {
|
|
286
|
+
console.error('Upload post complete, but upload not set in next method!');
|
|
281
287
|
}
|
|
282
|
-
}
|
|
283
|
-
else {
|
|
284
|
-
console.error('Upload post complete, but upload not set in next method!');
|
|
285
288
|
}
|
|
286
289
|
});
|
|
287
290
|
}
|
|
@@ -296,7 +299,7 @@ export class Uploader {
|
|
|
296
299
|
// set retry timer
|
|
297
300
|
this.retryTimerId = window.setTimeout(() => {
|
|
298
301
|
this.retryTimerRunning = false;
|
|
299
|
-
//console.debug("Upload retry timer
|
|
302
|
+
//console.debug("Upload retry timer expired. Continue processing...")
|
|
300
303
|
this.process();
|
|
301
304
|
}, this.RETRY_DELAY);
|
|
302
305
|
this.retryTimerRunning = true;
|
|
@@ -361,4 +364,4 @@ export class Uploader {
|
|
|
361
364
|
}
|
|
362
365
|
}
|
|
363
366
|
}
|
|
364
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"uploader.js","sourceRoot":"","sources":["../../../../../projects/speechrecorderng/src/lib/net/uploader.ts"],"names":[],"mappings":"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","sourcesContent":["\r\n    import {HttpClient, HttpErrorResponse} from \"@angular/common/http\";\r\n    import { timeout } from 'rxjs/operators'\r\n\r\n    // state of an upload\r\n    export enum UploadStatus {IDLE = 1, UPLOADING = 2,  ABORT = 3, DONE = 0, ERR = -1}\r\n\r\n    // state of the uploader\r\n    // TRY_UPLOADING is uploading state after an error (for example if disconnected from server)\r\n    export enum UploaderStatus { DONE = 0, UPLOADING = 1, TRY_UPLOADING = 2,NEXT = 3, ERR = -1}\r\n\r\n    export class UploaderStatusChangeEvent {\r\n        private _sizeQueued:number;\r\n        private _sizeDone:number;\r\n        private _status: UploaderStatus;\r\n\r\n\r\n        constructor(sizeQueued: number, sizeDone: number, status: UploaderStatus) {\r\n            this._sizeQueued = sizeQueued;\r\n            this._sizeDone = sizeDone;\r\n            this._status = status;\r\n        }\r\n\r\n        get sizeQueued():number {\r\n            return this._sizeQueued;\r\n        }\r\n\r\n        get sizeDone():number {\r\n            return this._sizeDone;\r\n        }\r\n\r\n        sizeInQueue(){\r\n          return this._sizeQueued-this._sizeDone;\r\n        }\r\n\r\n        get status(): UploaderStatus {\r\n            return this._status;\r\n        }\r\n\r\n        uploadDone(): boolean {\r\n            return (this._sizeDone >= this._sizeQueued);\r\n        }\r\n\r\n        percentDone(): number {\r\n            if (this._sizeQueued == 0) {\r\n                // no data queued, we are \"done\"\r\n                return 100;\r\n            }\r\n            let percent = Math.floor(this._sizeDone * 100 / this._sizeQueued);\r\n            //console.log(\"Upload status: queued: \"+this._sizeQueued+\", done: \"+this._sizeDone+\", \"+percent+\"%\")\r\n            return percent;\r\n        }\r\n\r\n    }\r\n\r\n    export class Upload {\r\n\r\n      get serverPersistable(): ServerPersistable | null {\r\n        return this._serverPersistable;\r\n      }\r\n\r\n        private _data:Blob|FormData;\r\n        private _url:string;\r\n\r\n        status: UploadStatus;\r\n        onDone:((upload:Upload)=>void) | null=null;\r\n\r\n        constructor(blob:Blob|FormData, url:string,private _serverPersistable:ServerPersistable|null=null) {\r\n            this._data = blob;\r\n            this._url = url;\r\n            this.status = UploadStatus.IDLE;\r\n        }\r\n\r\n        get url():string {\r\n            return this._url;\r\n        }\r\n\r\n        get data():Blob|FormData {\r\n            return this._data;\r\n        }\r\n\r\n        done(){\r\n            this.status=UploadStatus.DONE;\r\n            //console.debug(\"Single upload done.\");\r\n            if(this._serverPersistable) {\r\n              this._serverPersistable.serverPersisted = true;\r\n              //console.debug(\"Single upload set server persisted: \"+this.serverPersistable);\r\n            }else{\r\n              //console.debug(\"Server persistable not set.\");\r\n            }\r\n            if(this.onDone){\r\n              this.onDone(this);\r\n            }\r\n        }\r\n\r\n\r\n\r\n      public toString = () : string => {\r\n        let s=`Upload: Status: ${this.status}, URL: ${this._url}`;\r\n        if(this._data instanceof Blob){\r\n            s=s+`, Size: ${this._data.size}`;\r\n        }else if(this._data instanceof  FormData){\r\n           // TODO (iterate through parts ??)\r\n        }\r\n        return s;\r\n      }\r\n    }\r\n\r\n    export class UploadHolder{\r\n\r\n      onUploadSet:((upload:Upload)=>void)|null=null;\r\n\r\n      get upload(): Upload | null {\r\n        return this._upload;\r\n      }\r\n\r\n      set upload(value: Upload | null) {\r\n        this._upload = value;\r\n        if(this._upload && this.onUploadSet){\r\n          this.onUploadSet(this._upload);\r\n        }\r\n      }\r\n      private _upload:Upload|null=null;\r\n    }\r\n\r\n    export class UploadSet{\r\n\r\n      private uploads: Array<Upload|UploadHolder>=new Array<Upload|UploadHolder>();\r\n      private _complete=false;\r\n\r\n      private _onDone:((uploadSet:UploadSet)=>void)|null=null;\r\n\r\n      add(upload:Upload|UploadHolder){\r\n        if(this._complete) {\r\n          throw new Error('Cannot add upload to upload set. Upload set already complete.')\r\n        }\r\n        if(upload instanceof UploadHolder){\r\n          upload.onUploadSet=(upl)=>{\r\n            upl.onDone=()=>{\r\n              this.checkUploadStates();\r\n            }\r\n            this.checkUploadStates();\r\n          }\r\n        }\r\n        this.uploads.push(upload);\r\n\r\n      }\r\n\r\n      complete(){\r\n        // Mark set as complete\r\n        this._complete=true;\r\n\r\n        // add listeners to each upload\r\n        for(let upl of this.uploads){\r\n          if(upl instanceof Upload) {\r\n            upl.onDone = (upl: Upload) => {\r\n              this.checkUploadStates();\r\n            }\r\n          }else if(upl instanceof UploadHolder){\r\n            if(upl.upload){\r\n              upl.upload.onDone = (upl: Upload) => {\r\n                this.checkUploadStates();\r\n              }\r\n            }\r\n          }\r\n        }\r\n\r\n        // check immediately if already done\r\n        this.checkUploadStates();\r\n      }\r\n\r\n      set onDone(value: ((uploadSet: UploadSet) => void) | null) {\r\n        this._onDone = value;\r\n        this.checkUploadStates();\r\n      }\r\n\r\n      private checkUploadStates(){\r\n        //console.debug(\"Check upload state...\")\r\n        if(this._complete){\r\n\r\n          if(this._onDone) {\r\n            for(let upl of this.uploads){\r\n              if(upl instanceof Upload) {\r\n                if (UploadStatus.DONE !== upl.status) {\r\n                  // At least this upload is not yet done\r\n                  // Do nothing\r\n                  //console.debug(\"Check upload state: Upload not done.\")\r\n                  return;\r\n                }\r\n              }else if(upl instanceof UploadHolder){\r\n                if(upl.upload){\r\n                  if (UploadStatus.DONE !== upl.upload.status) {\r\n                    // At least this upload is not yet done\r\n                    // Do nothing\r\n                    //console.debug(\"Check upload state: Upload (holder) not done.\")\r\n                    return;\r\n                  }\r\n                }else{\r\n                  // The actual upload is not yet set\r\n                  //console.debug(\"Check upload state: Upload (holder): upload not yet set.\")\r\n                  return;\r\n                }\r\n              }\r\n            }\r\n            // set is complete and all upload parts are done, call done callback\r\n            //console.debug(\"Check upload state: All done.\")\r\n            this._onDone(this);\r\n          }\r\n        }\r\n      }\r\n    }\r\n\r\n    export interface ServerPersistable{\r\n        serverPersisted:boolean;\r\n    }\r\n\r\n    export class Uploader {\r\n        POST_MIN_TIMEOUT = 120000; // 2min plus ...\r\n        POST_TIMEOUT_PER_KB = 1000;  // ... 1s per kB\r\n\r\n        RETRY_DELAY: number = 30000; // retry every 30s\r\n        DEBUG_DELAY: number = 0;\r\n        //DEBUG_DELAY:number=0;\r\n        private status: UploaderStatus = UploaderStatus.DONE;\r\n        private que: Array<Upload>;\r\n        listener: ((ue: UploaderStatusChangeEvent) => void) | null = null;\r\n        private _sizeQueued: number = 0;\r\n        private _sizeDone: number = 0;\r\n\r\n        private retryTimerRunning = false;\r\n        private retryTimerId: number | null = null;\r\n\r\n        private te:TextEncoder=new TextEncoder();\r\n\r\n        constructor(private http: HttpClient, private withCredentials: boolean = false) {\r\n            this.que = new Array<Upload>();\r\n        }\r\n\r\n        private dataSize(dt:Blob|FormData){\r\n            let si=0;\r\n            if(dt instanceof Blob){\r\n                si=dt.size;\r\n            }else if(dt instanceof  FormData){\r\n                dt.forEach((v,k)=>{\r\n                    if(v instanceof File){\r\n                        si+=v.size;\r\n                    }else if(typeof v ==='string'){\r\n                        // encode to UT-f8 to get upload size\r\n                        si+= this.te.encode().length;\r\n                    }\r\n                })\r\n            }\r\n            return si;\r\n        }\r\n\r\n        private  uploadDone(ul:Upload) {\r\n\r\n            ul.done();\r\n\r\n            // remove upload from queue\r\n            for (let i = 0; i < this.que.length; i++) {\r\n                if (this.que[i] === ul) {\r\n                    // found, remove\r\n                    this.que.splice(i, 1);\r\n                    let ulSize = this.dataSize(ul.data);\r\n                    this._sizeDone += ulSize;\r\n                    break;\r\n                }\r\n            }\r\n            // set to done for now...\r\n            this.status = UploaderStatus.NEXT;\r\n            // continue\r\n            this.process();\r\n        }\r\n\r\n        private startUpload(ul:Upload) {\r\n\r\n            ul.status = UploadStatus.UPLOADING;\r\n            if (UploaderStatus.ERR === this.status) {\r\n                this.status = UploaderStatus.TRY_UPLOADING;\r\n            } else {\r\n                this.status = UploaderStatus.UPLOADING;\r\n            }\r\n\r\n            let dSize=this.dataSize(ul.data);\r\n            let timeoutForDataSize=dSize*this.POST_TIMEOUT_PER_KB/1000;\r\n            let timeoVal:number=Math.round(this.POST_MIN_TIMEOUT+timeoutForDataSize)\r\n            // pipe(timeout()) is not the same as xhr.timeout\r\n\r\n          let uploadedUpload:Upload|null=null;\r\n            //console.debug(\"Post upload: \"+ul)\r\n          this.http.post(ul.url,ul.data,{withCredentials:this.withCredentials}).pipe(timeout(timeoVal)).subscribe(\r\n            data => {\r\n              uploadedUpload = ul;\r\n              //console.debug('Next method called for upload: '+uploadedUpload)\r\n            },(err: HttpErrorResponse) => {\r\n              if (err.error instanceof Error) {\r\n                // A client-side or network error occurred. Handle it accordingly.\r\n                console.error('Upload error occurred:', err.error.message);\r\n              } else {\r\n                // The backend returned an unsuccessful response code.\r\n                // The response body may contain clues as to what went wrong,\r\n                console.error(`Upload error: Server returned code ${err.status}`);\r\n              }\r\n              this.processError(ul)\r\n            },()=>{\r\n              //console.debug('Upload complete method called')\r\n              if(uploadedUpload) {\r\n                if (this.DEBUG_DELAY>0) {\r\n                  window.setTimeout(() => {\r\n                    this.uploadDone(ul);\r\n                  }, this.DEBUG_DELAY);\r\n                } else {\r\n                  this.uploadDone(uploadedUpload)\r\n                }\r\n              }else{\r\n                console.error('Upload post complete, but upload not set in next method!')\r\n              }\r\n            });\r\n        }\r\n\r\n        private processError(ul:Upload) {\r\n            //console.debug(\"Process upload error...\")\r\n            ul.status=UploadStatus.ERR\r\n            this.status = UploaderStatus.ERR;\r\n\r\n            let ue = new UploaderStatusChangeEvent(this._sizeQueued, this._sizeDone, this.status);\r\n            if (this.listener) {\r\n                this.listener(ue);\r\n            }\r\n\r\n            // set retry timer\r\n            this.retryTimerId=window.setTimeout(() => {\r\n                this.retryTimerRunning=false;\r\n                //console.debug(\"Upload retry timer exprired. Continue processing...\")\r\n                this.process();\r\n            }, this.RETRY_DELAY);\r\n\r\n            this.retryTimerRunning=true;\r\n            //console.debug(\"Started upload retry timer \"+this.RETRY_DELAY+\"ms ...\")\r\n        }\r\n\r\n      private process() {\r\n        // clear retry timer if set\r\n        if(this.retryTimerRunning){\r\n            if(this.retryTimerId) {\r\n                window.clearTimeout(this.retryTimerId)\r\n            }\r\n          this.retryTimerRunning=false\r\n          //console.debug(\"Cleared retry timer.\")\r\n        }\r\n\r\n        let pul: Upload | null = null;\r\n\r\n        //console.debug(\"Uploader status: \"+this.status)\r\n\r\n        let s = this.que.length;\r\n        //console.debug(s+\" uploads are in the queue.\")\r\n\r\n        if (s>0 && UploaderStatus.UPLOADING != this.status && UploaderStatus.TRY_UPLOADING != this.status) {\r\n\r\n          for (let i = 0; i < s; i++) {\r\n            let ul = this.que[i];\r\n            //console.log(\"Upload \"+ul+\" status:\"+ul.status)\r\n            if (ul.status === UploadStatus.IDLE) {\r\n              //console.log(\"Upload \"+ul+\" startUpload\")\r\n              this.startUpload(ul);\r\n              pul = ul;\r\n              break;\r\n            }\r\n          }\r\n          if (!pul) {\r\n            //console.log(\"Check ERR uploads...\")\r\n            // now failed uploads\r\n            //console.debug(\"No regular upload found. Looking for error state uploads.\")\r\n            for (let i = 0; i < s; i++) {\r\n              let ul = this.que[i];\r\n              //console.log(\"Upload \"+ul+\" status:\"+ul.status)\r\n              if (ul.status === UploadStatus.ERR) {\r\n                //console.log(\"Upload (ERR) \"+ul+\" startUpload\")\r\n                //console.debug(\"Start error state upload \"+ul)\r\n                this.startUpload(ul);\r\n                pul = ul;\r\n                break;\r\n              }\r\n            }\r\n          }\r\n        }\r\n        if(s==0){\r\n          //console.debug(\"Upload done.\")\r\n          this.status=UploaderStatus.DONE\r\n        }\r\n        let ue = new UploaderStatusChangeEvent(this._sizeQueued, this._sizeDone, this.status);\r\n        if (this.listener) {\r\n          this.listener(ue);\r\n        }\r\n      }\r\n\r\n        queueUpload(ul: Upload) {\r\n            if (ul) {\r\n                let ulSize = this.dataSize(ul.data);\r\n                this.que.push(ul);\r\n                this._sizeQueued += ulSize;\r\n                this.process();\r\n            }\r\n        }\r\n    }\r\n\r\n\r\n"]}
|
|
367
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"uploader.js","sourceRoot":"","sources":["../../../../../projects/speechrecorderng/src/lib/net/uploader.ts"],"names":[],"mappings":"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","sourcesContent":["\r\n    import {HttpClient, HttpErrorResponse} from \"@angular/common/http\";\r\n    import { timeout } from 'rxjs/operators'\r\n\r\n    // state of an upload\r\n    export enum UploadStatus {IDLE = 1, UPLOADING = 2,  ABORT = 3, DONE = 0, ERR = -1}\r\n\r\n    // state of the uploader\r\n    // TRY_UPLOADING is uploading state after an error (for example if disconnected from server)\r\n    export enum UploaderStatus { DONE = 0, UPLOADING = 1, TRY_UPLOADING = 2,NEXT = 3, ERR = -1}\r\n\r\n    export class UploaderStatusChangeEvent {\r\n        private readonly _sizeQueued:number;\r\n        private readonly _sizeDone:number;\r\n        private readonly _status: UploaderStatus;\r\n\r\n\r\n        constructor(sizeQueued: number, sizeDone: number, status: UploaderStatus) {\r\n            this._sizeQueued = sizeQueued;\r\n            this._sizeDone = sizeDone;\r\n            this._status = status;\r\n        }\r\n\r\n        get sizeQueued():number {\r\n            return this._sizeQueued;\r\n        }\r\n\r\n        get sizeDone():number {\r\n            return this._sizeDone;\r\n        }\r\n\r\n        sizeInQueue(){\r\n          return this._sizeQueued-this._sizeDone;\r\n        }\r\n\r\n        get status(): UploaderStatus {\r\n            return this._status;\r\n        }\r\n\r\n        uploadDone(): boolean {\r\n            return (this._sizeDone >= this._sizeQueued);\r\n        }\r\n\r\n        percentDone(): number {\r\n            if (this._sizeQueued == 0) {\r\n                // no data queued, we are \"done\"\r\n                return 100;\r\n            }\r\n            let percent = Math.floor(this._sizeDone * 100 / this._sizeQueued);\r\n            //console.log(\"Upload status: queued: \"+this._sizeQueued+\", done: \"+this._sizeDone+\", \"+percent+\"%\")\r\n            return percent;\r\n        }\r\n\r\n    }\r\n\r\n    export class Upload {\r\n\r\n      get serverPersistable(): ServerPersistable | null {\r\n        return this._serverPersistable;\r\n      }\r\n\r\n        private readonly _data:Blob|FormData;\r\n        private readonly _url:string;\r\n\r\n        status: UploadStatus;\r\n        onDone:((upload:Upload)=>void) | null=null;\r\n\r\n        constructor(blob:Blob|FormData, url:string,private _serverPersistable:ServerPersistable|null=null) {\r\n            this._data = blob;\r\n            this._url = url;\r\n            this.status = UploadStatus.IDLE;\r\n        }\r\n\r\n        get url():string {\r\n            return this._url;\r\n        }\r\n\r\n        get data():Blob|FormData {\r\n            return this._data;\r\n        }\r\n\r\n        done(){\r\n            this.status=UploadStatus.DONE;\r\n            //console.debug(\"Single upload done.\");\r\n            if(this._serverPersistable) {\r\n              this._serverPersistable.serverPersisted = true;\r\n              //console.debug(\"Single upload set server persisted: \"+this.serverPersistable);\r\n            }else{\r\n              //console.debug(\"Server persistable not set.\");\r\n            }\r\n            if(this.onDone){\r\n              this.onDone(this);\r\n            }\r\n        }\r\n\r\n\r\n\r\n      public toString = () : string => {\r\n        let s=`Upload: Status: ${this.status}, URL: ${this._url}`;\r\n        if(this._data instanceof Blob){\r\n            s=s+`, Size: ${this._data.size}`;\r\n        }else if(this._data instanceof  FormData){\r\n           // TODO (iterate through parts ??)\r\n        }\r\n        return s;\r\n      }\r\n    }\r\n\r\n    export class UploadHolder{\r\n\r\n      onUploadSet:((upload:Upload)=>void)|null=null;\r\n\r\n      get upload(): Upload | null {\r\n        return this._upload;\r\n      }\r\n\r\n      set upload(value: Upload | null) {\r\n        this._upload = value;\r\n        if(this._upload && this.onUploadSet){\r\n          this.onUploadSet(this._upload);\r\n        }\r\n      }\r\n      private _upload:Upload|null=null;\r\n    }\r\n\r\n    export class UploadSet{\r\n\r\n      private uploads: Array<Upload|UploadHolder>=new Array<Upload|UploadHolder>();\r\n      private _complete=false;\r\n\r\n      private _onDone:((uploadSet:UploadSet)=>void)|null=null;\r\n\r\n      add(upload:Upload|UploadHolder){\r\n        if(this._complete) {\r\n          throw new Error('Cannot add upload to upload set. Upload set already complete.')\r\n        }\r\n        if(upload instanceof UploadHolder){\r\n          upload.onUploadSet=(upl)=>{\r\n            upl.onDone=()=>{\r\n              this.checkUploadStates();\r\n            }\r\n            this.checkUploadStates();\r\n          }\r\n        }\r\n        this.uploads.push(upload);\r\n\r\n      }\r\n\r\n      complete(){\r\n        // Mark set as complete\r\n        this._complete=true;\r\n\r\n        // add listeners to each upload\r\n        for(let upl of this.uploads){\r\n          if(upl instanceof Upload) {\r\n            upl.onDone = (upl: Upload) => {\r\n              this.checkUploadStates();\r\n            }\r\n          }else if(upl instanceof UploadHolder){\r\n            if(upl.upload){\r\n              upl.upload.onDone = (upl: Upload) => {\r\n                this.checkUploadStates();\r\n              }\r\n            }\r\n          }\r\n        }\r\n\r\n        // check immediately if already done\r\n        this.checkUploadStates();\r\n      }\r\n\r\n      set onDone(value: ((uploadSet: UploadSet) => void) | null) {\r\n        this._onDone = value;\r\n        this.checkUploadStates();\r\n      }\r\n\r\n      private checkUploadStates(){\r\n        //console.debug(\"Check upload state...\")\r\n        if(this._complete){\r\n\r\n          if(this._onDone) {\r\n            for(let upl of this.uploads){\r\n              if(upl instanceof Upload) {\r\n                if (UploadStatus.DONE !== upl.status) {\r\n                  // At least this upload is not yet done\r\n                  // Do nothing\r\n                  //console.debug(\"Check upload state: Upload not done.\")\r\n                  return;\r\n                }\r\n              }else if(upl instanceof UploadHolder){\r\n                if(upl.upload){\r\n                  if (UploadStatus.DONE !== upl.upload.status) {\r\n                    // At least this upload is not yet done\r\n                    // Do nothing\r\n                    //console.debug(\"Check upload state: Upload (holder) not done.\")\r\n                    return;\r\n                  }\r\n                }else{\r\n                  // The actual upload is not yet set\r\n                  //console.debug(\"Check upload state: Upload (holder): upload not yet set.\")\r\n                  return;\r\n                }\r\n              }\r\n            }\r\n            // set is complete and all upload parts are done, call done callback\r\n            //console.debug(\"Check upload state: All done.\")\r\n            this._onDone(this);\r\n          }\r\n        }\r\n      }\r\n    }\r\n\r\n    export interface ServerPersistable{\r\n        serverPersisted:boolean;\r\n    }\r\n\r\n    export class Uploader {\r\n        POST_MIN_TIMEOUT = 120000; // 2min plus ...\r\n        POST_TIMEOUT_PER_KB = 1000;  // ... 1s per kB\r\n\r\n        RETRY_DELAY: number = 30000; // retry every 30s\r\n        DEBUG_DELAY: number = 0;\r\n        //DEBUG_DELAY:number=0;\r\n        private status: UploaderStatus = UploaderStatus.DONE;\r\n        private readonly que: Array<Upload>;\r\n        listener: ((ue: UploaderStatusChangeEvent) => void) | null = null;\r\n        private _sizeQueued: number = 0;\r\n        private _sizeDone: number = 0;\r\n\r\n        private retryTimerRunning = false;\r\n        private retryTimerId: number | null = null;\r\n\r\n        private te:TextEncoder=new TextEncoder();\r\n\r\n        constructor(private http: HttpClient, private withCredentials: boolean = false) {\r\n            this.que = new Array<Upload>();\r\n        }\r\n\r\n        private dataSize(dt:Blob|FormData){\r\n            let si=0;\r\n            if(dt instanceof Blob){\r\n                si=dt.size;\r\n            }else if(dt instanceof  FormData){\r\n                dt.forEach((v,k)=>{\r\n                    if(v instanceof File){\r\n                        si+=v.size;\r\n                    }else if(typeof v ==='string'){\r\n                        // encode to UTF-8 to get upload size\r\n                        si+= this.te.encode().length;\r\n                    }\r\n                })\r\n            }\r\n            return si;\r\n        }\r\n\r\n        private  uploadDone(ul:Upload) {\r\n\r\n            ul.done();\r\n\r\n            // remove upload from queue\r\n            for (let i = 0; i < this.que.length; i++) {\r\n                if (this.que[i] === ul) {\r\n                    // found, remove\r\n                    this.que.splice(i, 1);\r\n                    let ulSize = this.dataSize(ul.data);\r\n                    this._sizeDone += ulSize;\r\n                    break;\r\n                }\r\n            }\r\n            // set to done for now...\r\n            this.status = UploaderStatus.NEXT;\r\n            // continue\r\n            this.process();\r\n        }\r\n\r\n        private startUpload(ul:Upload) {\r\n\r\n            ul.status = UploadStatus.UPLOADING;\r\n            if (UploaderStatus.ERR === this.status) {\r\n                this.status = UploaderStatus.TRY_UPLOADING;\r\n            } else {\r\n                this.status = UploaderStatus.UPLOADING;\r\n            }\r\n\r\n            let dSize=this.dataSize(ul.data);\r\n            let timeoutForDataSize=dSize*this.POST_TIMEOUT_PER_KB/1000;\r\n            let timeoVal:number=Math.round(this.POST_MIN_TIMEOUT+timeoutForDataSize)\r\n            // pipe(timeout()) is not the same as xhr.timeout\r\n\r\n          let uploadedUpload:Upload|null=null;\r\n            //console.debug(\"Post upload: \"+ul)\r\n            this.http.post(ul.url,ul.data,{withCredentials:this.withCredentials}).pipe(timeout(timeoVal)).subscribe(\r\n                {\r\n                    next: (data)=>\r\n                    {\r\n                        uploadedUpload = ul;\r\n                        //console.debug('Next method called for upload: '+uploadedUpload)\r\n                    }\r\n                    , error:(err: HttpErrorResponse) => {\r\n                        if (err.error instanceof Error) {\r\n                            // A client-side or network error occurred. Handle it accordingly.\r\n                            console.error('Upload error occurred:', err.error.message);\r\n                        } else {\r\n                            // The backend returned an unsuccessful response code.\r\n                            // The response body may contain clues as to what went wrong,\r\n                            console.error(`Upload error: Server returned code ${err.status}`);\r\n                        }\r\n                        this.processError(ul)\r\n                    }, complete: () => {\r\n                        //console.debug('Upload complete method called')\r\n                        if (uploadedUpload) {\r\n                            if (this.DEBUG_DELAY > 0) {\r\n                                window.setTimeout(() => {\r\n                                    this.uploadDone(ul);\r\n                                }, this.DEBUG_DELAY);\r\n                            } else {\r\n                                this.uploadDone(uploadedUpload)\r\n                            }\r\n                        } else {\r\n                            console.error('Upload post complete, but upload not set in next method!')\r\n                        }\r\n                    }\r\n                });\r\n        }\r\n\r\n        private processError(ul:Upload) {\r\n            //console.debug(\"Process upload error...\")\r\n            ul.status=UploadStatus.ERR\r\n            this.status = UploaderStatus.ERR;\r\n\r\n            let ue = new UploaderStatusChangeEvent(this._sizeQueued, this._sizeDone, this.status);\r\n            if (this.listener) {\r\n                this.listener(ue);\r\n            }\r\n\r\n            // set retry timer\r\n            this.retryTimerId=window.setTimeout(() => {\r\n                this.retryTimerRunning=false;\r\n                //console.debug(\"Upload retry timer expired. Continue processing...\")\r\n                this.process();\r\n            }, this.RETRY_DELAY);\r\n\r\n            this.retryTimerRunning=true;\r\n            //console.debug(\"Started upload retry timer \"+this.RETRY_DELAY+\"ms ...\")\r\n        }\r\n\r\n      private process() {\r\n        // clear retry timer if set\r\n        if(this.retryTimerRunning){\r\n            if(this.retryTimerId) {\r\n                window.clearTimeout(this.retryTimerId)\r\n            }\r\n          this.retryTimerRunning=false\r\n          //console.debug(\"Cleared retry timer.\")\r\n        }\r\n\r\n        let pul: Upload | null = null;\r\n\r\n        //console.debug(\"Uploader status: \"+this.status)\r\n\r\n        let s = this.que.length;\r\n        //console.debug(s+\" uploads are in the queue.\")\r\n\r\n        if (s>0 && UploaderStatus.UPLOADING != this.status && UploaderStatus.TRY_UPLOADING != this.status) {\r\n\r\n          for (let i = 0; i < s; i++) {\r\n            let ul = this.que[i];\r\n            //console.log(\"Upload \"+ul+\" status:\"+ul.status)\r\n            if (ul.status === UploadStatus.IDLE) {\r\n              //console.log(\"Upload \"+ul+\" startUpload\")\r\n              this.startUpload(ul);\r\n              pul = ul;\r\n              break;\r\n            }\r\n          }\r\n          if (!pul) {\r\n            //console.log(\"Check ERR uploads...\")\r\n            // now failed uploads\r\n            //console.debug(\"No regular upload found. Looking for error state uploads.\")\r\n            for (let i = 0; i < s; i++) {\r\n              let ul = this.que[i];\r\n              //console.log(\"Upload \"+ul+\" status:\"+ul.status)\r\n              if (ul.status === UploadStatus.ERR) {\r\n                //console.log(\"Upload (ERR) \"+ul+\" startUpload\")\r\n                //console.debug(\"Start error state upload \"+ul)\r\n                this.startUpload(ul);\r\n                pul = ul;\r\n                break;\r\n              }\r\n            }\r\n          }\r\n        }\r\n        if(s==0){\r\n          //console.debug(\"Upload done.\")\r\n          this.status=UploaderStatus.DONE\r\n        }\r\n        let ue = new UploaderStatusChangeEvent(this._sizeQueued, this._sizeDone, this.status);\r\n        if (this.listener) {\r\n          this.listener(ue);\r\n        }\r\n      }\r\n\r\n        queueUpload(ul: Upload) {\r\n            if (ul) {\r\n                let ulSize = this.dataSize(ul.data);\r\n                this.que.push(ul);\r\n                this._sizeQueued += ulSize;\r\n                this.process();\r\n            }\r\n        }\r\n    }\r\n\r\n\r\n"]}
|
|
@@ -9,12 +9,12 @@ import * as i1 from "@angular/common/http";
|
|
|
9
9
|
import * as i2 from "@angular/common";
|
|
10
10
|
import * as i3 from "../../spr.config";
|
|
11
11
|
export class ProjectService {
|
|
12
|
-
constructor(http,
|
|
12
|
+
constructor(http, platformLocation, config) {
|
|
13
13
|
this.http = http;
|
|
14
|
-
this.
|
|
14
|
+
this.platformLocation = platformLocation;
|
|
15
15
|
this.config = config;
|
|
16
16
|
this.withCredentials = false;
|
|
17
|
-
console.log("Base Href: " +
|
|
17
|
+
console.log("Base Href: " + platformLocation.getBaseHrefFromDOM());
|
|
18
18
|
let apiEndPoint = '';
|
|
19
19
|
if (config && config.apiEndPoint) {
|
|
20
20
|
apiEndPoint = config.apiEndPoint;
|
|
@@ -53,12 +53,12 @@ export class ProjectService {
|
|
|
53
53
|
}
|
|
54
54
|
}
|
|
55
55
|
ProjectService.PROJECT_API_CTX = 'project';
|
|
56
|
-
ProjectService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.
|
|
57
|
-
ProjectService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "14.
|
|
58
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.
|
|
56
|
+
ProjectService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: ProjectService, deps: [{ token: i1.HttpClient }, { token: i2.PlatformLocation }, { token: SPEECHRECORDER_CONFIG }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
57
|
+
ProjectService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: ProjectService });
|
|
58
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: ProjectService, decorators: [{
|
|
59
59
|
type: Injectable
|
|
60
60
|
}], ctorParameters: function () { return [{ type: i1.HttpClient }, { type: i2.PlatformLocation }, { type: i3.SpeechRecorderConfig, decorators: [{
|
|
61
61
|
type: Inject,
|
|
62
62
|
args: [SPEECHRECORDER_CONFIG]
|
|
63
63
|
}] }]; } });
|
|
64
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
64
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvamVjdC5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc3BlZWNocmVjb3JkZXJuZy9zcmMvbGliL3NwZWVjaHJlY29yZGVyL3Byb2plY3QvcHJvamVjdC5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBQ0gsT0FBTyxFQUFDLE1BQU0sRUFBRSxVQUFVLEVBQUMsTUFBTSxlQUFlLENBQUM7QUFFakQsT0FBTyxFQUFDLE9BQU8sRUFBRSxxQkFBcUIsRUFBdUIsTUFBTSxrQkFBa0IsQ0FBQztBQUV0RixPQUFPLEVBQUMsSUFBSSxFQUFDLE1BQU0sbUJBQW1CLENBQUM7Ozs7O0FBT3ZDLE1BQU0sT0FBTyxjQUFjO0lBS3pCLFlBQW9CLElBQWUsRUFBUyxnQkFBaUMsRUFBd0MsTUFBNEI7UUFBN0gsU0FBSSxHQUFKLElBQUksQ0FBVztRQUFTLHFCQUFnQixHQUFoQixnQkFBZ0IsQ0FBaUI7UUFBd0MsV0FBTSxHQUFOLE1BQU0sQ0FBc0I7UUFGaEksb0JBQWUsR0FBUyxLQUFLLENBQUM7UUFJN0MsT0FBTyxDQUFDLEdBQUcsQ0FBQyxhQUFhLEdBQUMsZ0JBQWdCLENBQUMsa0JBQWtCLEVBQUUsQ0FBQyxDQUFDO1FBRWpFLElBQUksV0FBVyxHQUFHLEVBQUUsQ0FBQTtRQUVwQixJQUFHLE1BQU0sSUFBSSxNQUFNLENBQUMsV0FBVyxFQUFFO1lBQy9CLFdBQVcsR0FBQyxNQUFNLENBQUMsV0FBVyxDQUFDO1NBQ2hDO1FBQ0QsSUFBRyxXQUFXLEtBQUssRUFBRSxFQUFDO1lBQ3BCLFdBQVcsR0FBQyxXQUFXLEdBQUMsR0FBRyxDQUFBO1NBQzVCO1FBQ0QsSUFBRyxNQUFNLElBQUUsSUFBSSxJQUFJLE1BQU0sQ0FBQyxlQUFlLElBQUUsSUFBSSxFQUFDO1lBQzlDLElBQUksQ0FBQyxlQUFlLEdBQUMsTUFBTSxDQUFDLGVBQWUsQ0FBQztTQUM3QztRQUVELElBQUksQ0FBQyxhQUFhLEdBQUcsV0FBVyxHQUFHLGNBQWMsQ0FBQyxlQUFlLENBQUM7SUFFcEUsQ0FBQztJQUVPLHFDQUFxQyxDQUFDLEdBQVU7UUFDdEQsSUFBSSxNQUFNLEdBQUMsR0FBRyxDQUFDO1FBQ2YsSUFBSSxJQUFJLENBQUMsTUFBTSxJQUFJLElBQUksQ0FBQyxNQUFNLENBQUMsT0FBTyxLQUFLLE9BQU8sQ0FBQyxLQUFLLEVBQUU7WUFDeEQsMkJBQTJCO1lBQzNCLDJFQUEyRTtZQUMzRSxNQUFNLEdBQUcsTUFBTSxHQUFHLG9CQUFvQixHQUFDLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQztTQUN4RDtRQUNELE9BQU8sTUFBTSxDQUFBO0lBQ2YsQ0FBQztJQUVELFVBQVUsQ0FBQyxFQUFTO1FBQ2xCLElBQUksUUFBUSxHQUFDLGtCQUFrQixDQUFDLEVBQUUsQ0FBQyxDQUFDO1FBQ3BDLE9BQU8sSUFBSSxDQUFDLHFDQUFxQyxDQUFDLElBQUksQ0FBQyxhQUFhLEdBQUcsR0FBRyxHQUFHLFFBQVEsQ0FBQyxDQUFDO0lBQ3pGLENBQUM7SUFFRCxrQkFBa0IsQ0FBQyxTQUFpQixFQUFDLGVBQXNCO1FBQ3pELElBQUksUUFBUSxHQUFDLGtCQUFrQixDQUFDLFNBQVMsQ0FBQyxDQUFDO1FBQzNDLDhGQUE4RjtRQUM5Riw0RkFBNEY7UUFDNUYsSUFBSSxhQUFhLEdBQUMsU0FBUyxDQUFDLGVBQWUsQ0FBQyxDQUFDO1FBQzdDLE9BQU8sSUFBSSxDQUFDLGFBQWEsR0FBRyxHQUFHLEdBQUcsUUFBUSxHQUFFLEdBQUcsR0FBQyxhQUFhLENBQUM7SUFDaEUsQ0FBQztJQUVELGlCQUFpQixDQUFDLEVBQVM7UUFDekIsSUFBSSxNQUFNLEdBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxFQUFFLENBQUMsQ0FBQztRQUM5QixPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFVLE1BQU0sRUFBQyxFQUFDLGVBQWUsRUFBRSxJQUFJLENBQUMsZUFBZSxFQUFDLENBQUMsQ0FBQTtJQUMvRSxDQUFDOztBQWxEcUIsOEJBQWUsR0FBQyxTQUFTLENBQUE7NEdBRHJDLGNBQWMsNEVBSzZELHFCQUFxQjtnSEFMaEcsY0FBYzs0RkFBZCxjQUFjO2tCQUQxQixVQUFVOzswQkFNc0UsTUFBTTsyQkFBQyxxQkFBcUIiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcclxuICogQ3JlYXRlZCBieSBrbGF1c2ogb24gMTcuMDYuMjAxNy5cclxuICovXHJcbmltcG9ydCB7SW5qZWN0LCBJbmplY3RhYmxlfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHtIdHRwQ2xpZW50fSBmcm9tIFwiQGFuZ3VsYXIvY29tbW9uL2h0dHBcIjtcclxuaW1wb3J0IHtBcGlUeXBlLCBTUEVFQ0hSRUNPUkRFUl9DT05GSUcsIFNwZWVjaFJlY29yZGVyQ29uZmlnfSBmcm9tIFwiLi4vLi4vc3ByLmNvbmZpZ1wiO1xyXG5pbXBvcnQge1Byb2plY3R9IGZyb20gXCIuL3Byb2plY3RcIjtcclxuaW1wb3J0IHtVVUlEfSBmcm9tIFwiLi4vLi4vdXRpbHMvdXRpbHNcIjtcclxuaW1wb3J0IHtPYnNlcnZhYmxlfSBmcm9tIFwicnhqc1wiO1xyXG5pbXBvcnQge1BsYXRmb3JtTG9jYXRpb259IGZyb20gXCJAYW5ndWxhci9jb21tb25cIjtcclxuXHJcblxyXG5cclxuQEluamVjdGFibGUoKVxyXG5leHBvcnQgY2xhc3MgUHJvamVjdFNlcnZpY2Uge1xyXG4gIHB1YmxpYyBzdGF0aWMgcmVhZG9ubHkgUFJPSkVDVF9BUElfQ1RYPSdwcm9qZWN0J1xyXG4gIHByaXZhdGUgcmVhZG9ubHkgcHJvamVjdEN0eFVybDpzdHJpbmc7XHJcbiAgcHJpdmF0ZSByZWFkb25seSB3aXRoQ3JlZGVudGlhbHM6Ym9vbGVhbj1mYWxzZTtcclxuXHJcbiAgY29uc3RydWN0b3IocHJpdmF0ZSBodHRwOkh0dHBDbGllbnQscHJpdmF0ZSBwbGF0Zm9ybUxvY2F0aW9uOlBsYXRmb3JtTG9jYXRpb24sQEluamVjdChTUEVFQ0hSRUNPUkRFUl9DT05GSUcpIHByaXZhdGUgY29uZmlnPzpTcGVlY2hSZWNvcmRlckNvbmZpZykge1xyXG5cclxuICAgIGNvbnNvbGUubG9nKFwiQmFzZSBIcmVmOiBcIitwbGF0Zm9ybUxvY2F0aW9uLmdldEJhc2VIcmVmRnJvbURPTSgpKTtcclxuXHJcbiAgICBsZXQgYXBpRW5kUG9pbnQgPSAnJ1xyXG5cclxuICAgIGlmKGNvbmZpZyAmJiBjb25maWcuYXBpRW5kUG9pbnQpIHtcclxuICAgICAgYXBpRW5kUG9pbnQ9Y29uZmlnLmFwaUVuZFBvaW50O1xyXG4gICAgfVxyXG4gICAgaWYoYXBpRW5kUG9pbnQgIT09ICcnKXtcclxuICAgICAgYXBpRW5kUG9pbnQ9YXBpRW5kUG9pbnQrJy8nXHJcbiAgICB9XHJcbiAgICBpZihjb25maWchPW51bGwgJiYgY29uZmlnLndpdGhDcmVkZW50aWFscyE9bnVsbCl7XHJcbiAgICAgIHRoaXMud2l0aENyZWRlbnRpYWxzPWNvbmZpZy53aXRoQ3JlZGVudGlhbHM7XHJcbiAgICB9XHJcblxyXG4gICAgdGhpcy5wcm9qZWN0Q3R4VXJsID0gYXBpRW5kUG9pbnQgKyBQcm9qZWN0U2VydmljZS5QUk9KRUNUX0FQSV9DVFg7XHJcblxyXG4gIH1cclxuXHJcbiAgcHJpdmF0ZSBhcHBlbmRSZXF1ZXN0VVVJREZvckRldmVsb3BtZW50U2VydmVyKHVybDpzdHJpbmcpOnN0cmluZ3tcclxuICAgIGxldCByZXNVcmw9dXJsO1xyXG4gICAgaWYgKHRoaXMuY29uZmlnICYmIHRoaXMuY29uZmlnLmFwaVR5cGUgPT09IEFwaVR5cGUuRklMRVMpIHtcclxuICAgICAgLy8gZm9yIGRldmVsb3BtZW50IGFuZCBkZW1vXHJcbiAgICAgIC8vIGFwcGVuZCBVVUlEIHRvIG1ha2UgcmVxdWVzdCBVUkwgdW5pcXVlIHRvIGF2b2lkIGxvY2FsaG9zdCBzZXJ2ZXIgY2FjaGluZ1xyXG4gICAgICByZXNVcmwgPSByZXNVcmwgKyAnLmpzb24/cmVxdWVzdFVVSUQ9JytVVUlELmdlbmVyYXRlKCk7XHJcbiAgICB9XHJcbiAgICByZXR1cm4gcmVzVXJsXHJcbiAgfVxyXG5cclxuICBwcm9qZWN0VXJsKGlkOnN0cmluZyk6c3RyaW5ne1xyXG4gICAgbGV0IGVuY1ByaklkPWVuY29kZVVSSUNvbXBvbmVudChpZCk7XHJcbiAgICByZXR1cm4gdGhpcy5hcHBlbmRSZXF1ZXN0VVVJREZvckRldmVsb3BtZW50U2VydmVyKHRoaXMucHJvamVjdEN0eFVybCArICcvJyArIGVuY1ByaklkKTtcclxuICB9XHJcblxyXG4gIHByb2plY3RSZXNvdXJjZVVybChwcm9qZWN0SWQ6IHN0cmluZyxyZWxSZXNvdXJjZVBhdGg6c3RyaW5nKTpzdHJpbmd7XHJcbiAgICBsZXQgZW5jUHJqSWQ9ZW5jb2RlVVJJQ29tcG9uZW50KHByb2plY3RJZCk7XHJcbiAgICAvLyBVc2UgZW5jb2RlVVJJIGhlcmUgc2luY2UgdGhlIHJlbGF0aWNlIHJlc291cmNlIHBhdGggZnJvbSByZWNvcmRpbmcgc2NyaXB0cyBpcyBhbHJlYWR5IGEgVVJJXHJcbiAgICAvLyBlbmNvZGVVUklDb21wb25lbnQgZnVuY3Rpb24gd291bGQgZW5jb2RlIHRoZSBzbGFzaGVzIGFuZCBmZXRjaGluZyB0aGUgcmVzb3VyY2Ugd291bGQgZmFpbFxyXG4gICAgbGV0IGVuY1JlbFJlc1BhdGg9ZW5jb2RlVVJJKHJlbFJlc291cmNlUGF0aCk7XHJcbiAgICByZXR1cm4gdGhpcy5wcm9qZWN0Q3R4VXJsICsgJy8nICsgZW5jUHJqSWQgKycvJytlbmNSZWxSZXNQYXRoO1xyXG4gIH1cclxuXHJcbiAgcHJvamVjdE9ic2VydmFibGUoaWQ6c3RyaW5nKTpPYnNlcnZhYmxlPFByb2plY3Q+e1xyXG4gICAgbGV0IHByalVybD10aGlzLnByb2plY3RVcmwoaWQpO1xyXG4gICAgIHJldHVybiB0aGlzLmh0dHAuZ2V0PFByb2plY3Q+KHByalVybCx7d2l0aENyZWRlbnRpYWxzOiB0aGlzLndpdGhDcmVkZW50aWFsc30pXHJcbiAgIH1cclxuXHJcbn1cclxuXHJcblxyXG4iXX0=
|
|
@@ -69,62 +69,64 @@ let BasicRecordingService = class BasicRecordingService {
|
|
|
69
69
|
ausps.set('requestUUID', UUID.generate());
|
|
70
70
|
}
|
|
71
71
|
let obs = new Observable(observer => {
|
|
72
|
-
this.audioRequestByURL(baseAudioUrl, ausps).subscribe(
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
72
|
+
this.audioRequestByURL(baseAudioUrl, ausps).subscribe({
|
|
73
|
+
next: resp => {
|
|
74
|
+
// Do not use Promise version, which does not work with Safari 13 (13.0.5)
|
|
75
|
+
if (resp.body) {
|
|
76
|
+
//console.debug("chunkAudioRequest: observer.closed: "+observer.closed);
|
|
77
|
+
//console.debug("Audio file bytes: "+resp.body.byteLength);
|
|
78
|
+
// Check original audio format
|
|
79
|
+
let wr = new WavReader(resp.body);
|
|
80
|
+
const pcmFmt = wr.readFormat();
|
|
81
|
+
const orgFl = wr.frameLength();
|
|
82
|
+
// if(pcmFmt){
|
|
83
|
+
// console.debug("Original WAVE format of download chunk: "+pcmFmt);
|
|
84
|
+
// }else{
|
|
85
|
+
// console.error("Could not read WAVE format of original download chunk!");
|
|
86
|
+
// }
|
|
87
|
+
// if(orgFl){
|
|
88
|
+
// console.debug("Original frame length of download chunk: "+orgFl);
|
|
89
|
+
// }else{
|
|
90
|
+
// console.error("Could not read WAVE format of original download chunk!");
|
|
91
|
+
// }
|
|
92
|
+
if (pcmFmt && orgFl) {
|
|
93
|
+
aCtx.decodeAudioData(resp.body, ab => {
|
|
94
|
+
//console.debug("Decoded audio chunk frames: "+ab.length);
|
|
95
|
+
let chDl = new ChunkDownload(pcmFmt, orgFl, ab);
|
|
96
|
+
observer.next(chDl);
|
|
97
|
+
observer.complete();
|
|
98
|
+
}, error => {
|
|
99
|
+
//if(error instanceof HttpErrorResponse) {
|
|
100
|
+
// if (error.status == 404) {
|
|
101
|
+
// // Interpret not as an error, the file ist not recorded yet
|
|
102
|
+
// observer.next(null);
|
|
103
|
+
// observer.complete()
|
|
104
|
+
// } else {
|
|
105
|
+
// // all other states are errors
|
|
106
|
+
console.error("Recordings service chunkAudioRequest error decoding audio data: " + error.name + ": " + error.message);
|
|
107
|
+
observer.error(error);
|
|
108
|
+
// }
|
|
109
|
+
// }
|
|
110
|
+
});
|
|
111
|
+
}
|
|
112
|
+
else {
|
|
113
|
+
const errMsg = 'Could not parse audio header for format and/or frame length of download.';
|
|
114
|
+
console.error(errMsg);
|
|
115
|
+
observer.error(errMsg);
|
|
116
|
+
}
|
|
110
117
|
}
|
|
111
118
|
else {
|
|
112
|
-
const errMsg = '
|
|
119
|
+
const errMsg = 'Fetching audio file: response has no body';
|
|
113
120
|
console.error(errMsg);
|
|
114
121
|
observer.error(errMsg);
|
|
115
122
|
}
|
|
123
|
+
}, error: (error) => {
|
|
124
|
+
// all other states are errors
|
|
125
|
+
//const errMsg='Fetching audio file HTTP error: '+error;
|
|
126
|
+
//console.error(errMsg);
|
|
127
|
+
observer.error(error);
|
|
128
|
+
//observer.complete();
|
|
116
129
|
}
|
|
117
|
-
else {
|
|
118
|
-
const errMsg = 'Fetching audio file: response has no body';
|
|
119
|
-
console.error(errMsg);
|
|
120
|
-
observer.error(errMsg);
|
|
121
|
-
}
|
|
122
|
-
}, (error) => {
|
|
123
|
-
// all other states are errors
|
|
124
|
-
//const errMsg='Fetching audio file HTTP error: '+error;
|
|
125
|
-
//console.error(errMsg);
|
|
126
|
-
observer.error(error);
|
|
127
|
-
//observer.complete();
|
|
128
130
|
});
|
|
129
131
|
});
|
|
130
132
|
return obs;
|
|
@@ -210,4 +212,4 @@ BasicRecordingService = __decorate([
|
|
|
210
212
|
__param(1, Inject(SPEECHRECORDER_CONFIG))
|
|
211
213
|
], BasicRecordingService);
|
|
212
214
|
export { BasicRecordingService };
|
|
213
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"basic_recording.service.js","sourceRoot":"","sources":["../../../../../../projects/speechrecorderng/src/lib/speechrecorder/recordings/basic_recording.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {HttpClient, HttpHeaders, HttpResponse} from \"@angular/common/http\";\r\nimport {Inject} from \"@angular/core\";\r\nimport {ApiType, SPEECHRECORDER_CONFIG, SpeechRecorderConfig} from \"../../spr.config\";\r\nimport {Observable} from \"rxjs\";\r\nimport {NetAudioBuffer} from \"../../audio/net_audio_buffer\";\r\nimport {UUID} from \"../../utils/utils\";\r\nimport {WavReader} from \"../../audio/impl/wavreader\";\r\nimport {PCMAudioFormat} from \"../../audio/format\";\r\n\r\nexport class ChunkDownload{\r\n  get orgPCMAudioFormat(): PCMAudioFormat {\r\n    return this._orgPCMAudioFormat;\r\n  }\r\n\r\n  get orgFrameLength(): number {\r\n    return this._orgFrameLength;\r\n  }\r\n\r\n  get decodedAudioBuffer(): AudioBuffer {\r\n    return this._decodedAudioBuffer;\r\n  }\r\n  constructor(private _orgPCMAudioFormat:PCMAudioFormat,private _orgFrameLength:number,private _decodedAudioBuffer:AudioBuffer){}\r\n}\r\n\r\nexport class BasicRecordingService{\r\n  // iPad 9th generation, iOS 15.7.1, sometimes:\r\n  // Failed to load resource: WebKit hat einen internen Fehler festgestellt\r\n  // Firefox on Windows crashes\r\n  public static readonly DEFAULT_CHUNKED_DOWNLOAD_SECONDS:number=10;\r\n  public static readonly DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES:number=2880000*5; // Default 5 minutes one channel at 48kHz\r\n  protected _maxAutoNetMemStoreSamples:number=BasicRecordingService.DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES;\r\n    protected apiEndPoint: string;\r\n    protected withCredentials: boolean = false;\r\n    constructor(protected http: HttpClient, @Inject(SPEECHRECORDER_CONFIG) protected config?: SpeechRecorderConfig) {\r\n        this.apiEndPoint = ''\r\n\r\n        if (config && config.apiEndPoint) {\r\n            this.apiEndPoint = config.apiEndPoint;\r\n        }\r\n        if (this.apiEndPoint !== '') {\r\n            this.apiEndPoint = this.apiEndPoint + '/'\r\n        }\r\n        if (config != null && config.withCredentials != null) {\r\n            this.withCredentials = config.withCredentials;\r\n        }\r\n    }\r\n\r\n    protected audioRequestByURL(audioBaseUrl:string,audioURLSearchParams:URLSearchParams): Observable<HttpResponse<ArrayBuffer>> {\r\n        let audioUrl=audioBaseUrl;\r\n        if (this.config && this.config.apiType === ApiType.FILES) {\r\n            // for development and demo\r\n            audioUrl=audioUrl+'.wav';\r\n            // append UUID to make request URL unique to avoid localhost server caching\r\n            audioURLSearchParams.set('requestUUID',UUID.generate());\r\n            //audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();\r\n        }\r\n\r\n        audioUrl=audioUrl+'?'+audioURLSearchParams.toString();\r\n\r\n        let headers = new HttpHeaders();\r\n        headers = headers.set('Accept', 'audio/wav');\r\n        return this.http.get(audioUrl, {\r\n            headers: headers,\r\n            observe: 'response',\r\n            responseType: 'arraybuffer',\r\n            withCredentials: this.withCredentials\r\n        });\r\n\r\n    }\r\n\r\n    public chunkAudioRequest(aCtx:AudioContext,baseAudioUrl:string,startFrame:number=0,frameLength:number): Observable<ChunkDownload|null> {\r\n\r\n        let ausps=new URLSearchParams();\r\n        ausps.set('startFrame',startFrame.toString());\r\n        ausps.set('frameLength',frameLength.toString());\r\n        if (this.config && this.config.apiType === ApiType.FILES) {\r\n            // for development and demo\r\n            // append UUID to make request URL unique to avoid localhost server caching\r\n            //audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();\r\n            ausps.set('requestUUID',UUID.generate());\r\n        }\r\n\r\n        let obs=new Observable<ChunkDownload|null>(observer=> {\r\n            this.audioRequestByURL(baseAudioUrl,ausps).subscribe(resp => {\r\n                    // Do not use Promise version, which does not work with Safari 13 (13.0.5)\r\n                    if (resp.body) {\r\n                        //console.debug(\"chunkAudioRequest: observer.closed: \"+observer.closed);\r\n                        //console.debug(\"Audio file bytes: \"+resp.body.byteLength);\r\n\r\n                        // Check original audio format\r\n                        let wr=new WavReader(resp.body);\r\n                        const pcmFmt=wr.readFormat();\r\n                        const orgFl=wr.frameLength();\r\n                        // if(pcmFmt){\r\n                        //   console.debug(\"Original WAVE format of download chunk: \"+pcmFmt);\r\n                        // }else{\r\n                        //   console.error(\"Could not read WAVE format of original download chunk!\");\r\n                        // }\r\n                        // if(orgFl){\r\n                        //   console.debug(\"Original frame length of download chunk: \"+orgFl);\r\n                        // }else{\r\n                        //   console.error(\"Could not read WAVE format of original download chunk!\");\r\n                        // }\r\n\r\n                        if(pcmFmt && orgFl) {\r\n                            aCtx.decodeAudioData(resp.body, ab => {\r\n                                    //console.debug(\"Decoded audio chunk frames: \"+ab.length);\r\n                                    let chDl = new ChunkDownload(pcmFmt, orgFl, ab);\r\n                                    observer.next(chDl);\r\n                                    observer.complete();\r\n                                }\r\n                                , error => {\r\n                                    //if(error instanceof HttpErrorResponse) {\r\n                                    // if (error.status == 404) {\r\n                                    //   // Interpret not as an error, the file ist not recorded yet\r\n                                    //   observer.next(null);\r\n                                    //   observer.complete()\r\n                                    // } else {\r\n                                    //   // all other states are errors\r\n                                    console.error(\"Recordings service chunkAudioRequest error decoding audio data: \"+error.name+\": \"+error.message);\r\n                                    observer.error(error);\r\n                                    // }\r\n                                    // }\r\n                                });\r\n                        }else{\r\n                            const errMsg='Could not parse audio header for format and/or frame length of download.';\r\n                            console.error(errMsg);\r\n                            observer.error(errMsg);\r\n                        }\r\n                    } else {\r\n                        const errMsg='Fetching audio file: response has no body';\r\n                        console.error(errMsg);\r\n                        observer.error(errMsg);\r\n                    }\r\n                },\r\n                (error) => {\r\n                    // all other states are errors\r\n                    //const errMsg='Fetching audio file HTTP error: '+error;\r\n                    //console.error(errMsg);\r\n                    observer.error(error);\r\n                    //observer.complete();\r\n\r\n                });\r\n        });\r\n        return obs;\r\n    }\r\n\r\n\r\n    protected chunkAudioRequestToNetAudioBuffer(aCtx: AudioContext, baseAudioUrl: string, startFrame: number = 0, orgSampleRate: number, seconds:number,frames: number | null): Observable<NetAudioBuffer | null> {\r\n        //let audioUrl=baseAudioUrl+'?startFrame='+startFrame+'&frameLength='+frameLength;\r\n        //let audioUrl=new URL(baseAudioUrl);\r\n        // if(orgSampleRate!=null && frameLength%orgSampleRate>0){\r\n        //   const errMsg='frameLength must be equal or multiple of original samplerate.';\r\n        //   console.error(errMsg+' ('+frameLength+'%'+orgSampleRate+'=='+(frameLength%orgSampleRate)+')');\r\n        //   throw Error(errMsg)\r\n        // }\r\n        let frameLength:number=orgSampleRate*Math.round(seconds);\r\n        let ausps=new URLSearchParams();\r\n        ausps.set('startFrame',startFrame.toString());\r\n        ausps.set('frameLength',frameLength.toString());\r\n        if (this.config && this.config.apiType === ApiType.FILES) {\r\n            // for development and demo\r\n            // append UUID to make request URL unique to avoid localhost server caching\r\n            //audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();\r\n            ausps.set('requestUUID',UUID.generate());\r\n        }\r\n        let obs=new Observable<NetAudioBuffer|null>(subscriber=> {\r\n            this.audioRequestByURL(baseAudioUrl,ausps).subscribe({next:(resp) => {\r\n                    // Do not use Promise version, which does not work with Safari 13 (13.0.5)\r\n                    if (resp.body) {\r\n                        //console.debug(\"chunkAudioRequestTonetAb: subscriber.closed: \"+subscriber.closed);\r\n                        //console.debug(\"chunkAudioRequestTonetAb: Audio file bytes: \"+resp.body.byteLength);\r\n                        aCtx.decodeAudioData(resp.body, ab => {\r\n                                //console.debug(\"chunkAudioRequestTonetAb: Decoded audio chunk frames for netAb: \"+ab.length);\r\n                                //console.debug(\"chunkAudioRequestTonetAb: Create netAb ab from chunk ab...\");\r\n                                if(frames===null){\r\n                                    subscriber.error(new Error('Could not get frame length from recording file object'));\r\n                                }else {\r\n                                    let fl = frames;\r\n                                    if (ab.sampleRate !== orgSampleRate) {\r\n                                        if (orgSampleRate && frames) {\r\n                                            fl = Math.round(ab.sampleRate * frames / orgSampleRate);\r\n                                            //console.debug(\"Platform sr: \"+ab.sampleRate+\", file sr: \"+orgSampleRate+\", decoded/org frame length: \"+fl+\"/\"+frames+\", ab.length: \"+ab.length);\r\n                                        }\r\n                                    }\r\n\r\n                                    let nab = NetAudioBuffer.fromChunkAudioBuffer(aCtx, this,baseAudioUrl, ab, fl,frameLength);\r\n                                    //let rp=new ReadyProvider();\r\n                                    //nab.readyProvider=rp;\r\n                                    //rp.ready();\r\n                                    nab.ready();\r\n                                    if (nab.frameLen < frameLength) {\r\n                                        //console.debug(\"chunkAudioRequestTonetAb: Built netAb ab from chunk ab: First chunk shorter tha frameLength (\"+netAbAudioBuffer.frameLen+\"<\"+frameLength+\"), assuming end of data, sealing netAb ab.\");\r\n                                        nab.seal();\r\n                                    }\r\n                                    subscriber.next(nab);\r\n\r\n                                    subscriber.complete();\r\n                                }\r\n                            }\r\n                            , error => {\r\n                                console.error('chunkAudioRequestToNetAb: error: '+error);\r\n                                //if(error instanceof HttpErrorResponse) {\r\n                                subscriber.error(error);\r\n                                //}\r\n                            })\r\n                    } else {\r\n                        console.error('chunkAudioRequestToNetAb: Fetching audio file: response has no body');\r\n                        subscriber.error('chunkAudioRequestToNetAb: Fetching audio file: response has no body');\r\n                    }\r\n                },\r\n                error:(error) => {\r\n                    console.error('chunkAudioRequestToNetAb: error: '+error);\r\n                    subscriber.error(error);\r\n                    //subscriber.complete();\r\n                }\r\n            });\r\n        });\r\n        return obs;\r\n    }\r\n\r\n}\r\n"]}
|
|
215
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"basic_recording.service.js","sourceRoot":"","sources":["../../../../../../projects/speechrecorderng/src/lib/speechrecorder/recordings/basic_recording.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {HttpClient, HttpHeaders, HttpResponse} from \"@angular/common/http\";\r\nimport {Inject} from \"@angular/core\";\r\nimport {ApiType, SPEECHRECORDER_CONFIG, SpeechRecorderConfig} from \"../../spr.config\";\r\nimport {Observable} from \"rxjs\";\r\nimport {NetAudioBuffer} from \"../../audio/net_audio_buffer\";\r\nimport {UUID} from \"../../utils/utils\";\r\nimport {WavReader} from \"../../audio/impl/wavreader\";\r\nimport {PCMAudioFormat} from \"../../audio/format\";\r\n\r\nexport class ChunkDownload{\r\n  get orgPCMAudioFormat(): PCMAudioFormat {\r\n    return this._orgPCMAudioFormat;\r\n  }\r\n\r\n  get orgFrameLength(): number {\r\n    return this._orgFrameLength;\r\n  }\r\n\r\n  get decodedAudioBuffer(): AudioBuffer {\r\n    return this._decodedAudioBuffer;\r\n  }\r\n  constructor(private _orgPCMAudioFormat:PCMAudioFormat,private _orgFrameLength:number,private _decodedAudioBuffer:AudioBuffer){}\r\n}\r\n\r\nexport class BasicRecordingService{\r\n  // iPad 9th generation, iOS 15.7.1, sometimes:\r\n  // Failed to load resource: WebKit hat einen internen Fehler festgestellt\r\n  // Firefox on Windows crashes\r\n  public static readonly DEFAULT_CHUNKED_DOWNLOAD_SECONDS:number=10;\r\n  public static readonly DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES:number=2880000*5; // Default 5 minutes one channel at 48kHz\r\n  protected _maxAutoNetMemStoreSamples:number=BasicRecordingService.DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES;\r\n    protected apiEndPoint: string;\r\n    protected withCredentials: boolean = false;\r\n    constructor(protected http: HttpClient, @Inject(SPEECHRECORDER_CONFIG) protected config?: SpeechRecorderConfig) {\r\n        this.apiEndPoint = ''\r\n\r\n        if (config && config.apiEndPoint) {\r\n            this.apiEndPoint = config.apiEndPoint;\r\n        }\r\n        if (this.apiEndPoint !== '') {\r\n            this.apiEndPoint = this.apiEndPoint + '/'\r\n        }\r\n        if (config != null && config.withCredentials != null) {\r\n            this.withCredentials = config.withCredentials;\r\n        }\r\n    }\r\n\r\n    protected audioRequestByURL(audioBaseUrl:string,audioURLSearchParams:URLSearchParams): Observable<HttpResponse<ArrayBuffer>> {\r\n        let audioUrl=audioBaseUrl;\r\n        if (this.config && this.config.apiType === ApiType.FILES) {\r\n            // for development and demo\r\n            audioUrl=audioUrl+'.wav';\r\n            // append UUID to make request URL unique to avoid localhost server caching\r\n            audioURLSearchParams.set('requestUUID',UUID.generate());\r\n            //audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();\r\n        }\r\n\r\n        audioUrl=audioUrl+'?'+audioURLSearchParams.toString();\r\n\r\n        let headers = new HttpHeaders();\r\n        headers = headers.set('Accept', 'audio/wav');\r\n        return this.http.get(audioUrl, {\r\n            headers: headers,\r\n            observe: 'response',\r\n            responseType: 'arraybuffer',\r\n            withCredentials: this.withCredentials\r\n        });\r\n\r\n    }\r\n\r\n    public chunkAudioRequest(aCtx:AudioContext,baseAudioUrl:string,startFrame:number=0,frameLength:number): Observable<ChunkDownload|null> {\r\n\r\n        let ausps=new URLSearchParams();\r\n        ausps.set('startFrame',startFrame.toString());\r\n        ausps.set('frameLength',frameLength.toString());\r\n        if (this.config && this.config.apiType === ApiType.FILES) {\r\n            // for development and demo\r\n            // append UUID to make request URL unique to avoid localhost server caching\r\n            //audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();\r\n            ausps.set('requestUUID',UUID.generate());\r\n        }\r\n\r\n        let obs=new Observable<ChunkDownload|null>(observer=> {\r\n            this.audioRequestByURL(baseAudioUrl,ausps).subscribe(\r\n                {\r\n                    next: resp => {\r\n                        // Do not use Promise version, which does not work with Safari 13 (13.0.5)\r\n                        if (resp.body) {\r\n                            //console.debug(\"chunkAudioRequest: observer.closed: \"+observer.closed);\r\n                            //console.debug(\"Audio file bytes: \"+resp.body.byteLength);\r\n\r\n                            // Check original audio format\r\n                            let wr = new WavReader(resp.body);\r\n                            const pcmFmt = wr.readFormat();\r\n                            const orgFl = wr.frameLength();\r\n                            // if(pcmFmt){\r\n                            //   console.debug(\"Original WAVE format of download chunk: \"+pcmFmt);\r\n                            // }else{\r\n                            //   console.error(\"Could not read WAVE format of original download chunk!\");\r\n                            // }\r\n                            // if(orgFl){\r\n                            //   console.debug(\"Original frame length of download chunk: \"+orgFl);\r\n                            // }else{\r\n                            //   console.error(\"Could not read WAVE format of original download chunk!\");\r\n                            // }\r\n\r\n                            if (pcmFmt && orgFl) {\r\n                                aCtx.decodeAudioData(resp.body, ab => {\r\n                                        //console.debug(\"Decoded audio chunk frames: \"+ab.length);\r\n                                        let chDl = new ChunkDownload(pcmFmt, orgFl, ab);\r\n                                        observer.next(chDl);\r\n                                        observer.complete();\r\n                                    }\r\n                                    , error => {\r\n                                        //if(error instanceof HttpErrorResponse) {\r\n                                        // if (error.status == 404) {\r\n                                        //   // Interpret not as an error, the file ist not recorded yet\r\n                                        //   observer.next(null);\r\n                                        //   observer.complete()\r\n                                        // } else {\r\n                                        //   // all other states are errors\r\n                                        console.error(\"Recordings service chunkAudioRequest error decoding audio data: \" + error.name + \": \" + error.message);\r\n                                        observer.error(error);\r\n                                        // }\r\n                                        // }\r\n                                    });\r\n                            } else {\r\n                                const errMsg = 'Could not parse audio header for format and/or frame length of download.';\r\n                                console.error(errMsg);\r\n                                observer.error(errMsg);\r\n                            }\r\n                        } else {\r\n                            const errMsg = 'Fetching audio file: response has no body';\r\n                            console.error(errMsg);\r\n                            observer.error(errMsg);\r\n                        }\r\n                    }, error:\r\n                        (error) => {\r\n                            // all other states are errors\r\n                            //const errMsg='Fetching audio file HTTP error: '+error;\r\n                            //console.error(errMsg);\r\n                            observer.error(error);\r\n                            //observer.complete();\r\n\r\n                        }\r\n                });\r\n        });\r\n        return obs;\r\n    }\r\n\r\n\r\n    protected chunkAudioRequestToNetAudioBuffer(aCtx: AudioContext, baseAudioUrl: string, startFrame: number = 0, orgSampleRate: number, seconds:number,frames: number | null): Observable<NetAudioBuffer | null> {\r\n        //let audioUrl=baseAudioUrl+'?startFrame='+startFrame+'&frameLength='+frameLength;\r\n        //let audioUrl=new URL(baseAudioUrl);\r\n        // if(orgSampleRate!=null && frameLength%orgSampleRate>0){\r\n        //   const errMsg='frameLength must be equal or multiple of original samplerate.';\r\n        //   console.error(errMsg+' ('+frameLength+'%'+orgSampleRate+'=='+(frameLength%orgSampleRate)+')');\r\n        //   throw Error(errMsg)\r\n        // }\r\n        let frameLength:number=orgSampleRate*Math.round(seconds);\r\n        let ausps=new URLSearchParams();\r\n        ausps.set('startFrame',startFrame.toString());\r\n        ausps.set('frameLength',frameLength.toString());\r\n        if (this.config && this.config.apiType === ApiType.FILES) {\r\n            // for development and demo\r\n            // append UUID to make request URL unique to avoid localhost server caching\r\n            //audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();\r\n            ausps.set('requestUUID',UUID.generate());\r\n        }\r\n        let obs=new Observable<NetAudioBuffer|null>(subscriber=> {\r\n            this.audioRequestByURL(baseAudioUrl,ausps).subscribe({next:(resp) => {\r\n                    // Do not use Promise version, which does not work with Safari 13 (13.0.5)\r\n                    if (resp.body) {\r\n                        //console.debug(\"chunkAudioRequestTonetAb: subscriber.closed: \"+subscriber.closed);\r\n                        //console.debug(\"chunkAudioRequestTonetAb: Audio file bytes: \"+resp.body.byteLength);\r\n                        aCtx.decodeAudioData(resp.body, ab => {\r\n                                //console.debug(\"chunkAudioRequestTonetAb: Decoded audio chunk frames for netAb: \"+ab.length);\r\n                                //console.debug(\"chunkAudioRequestTonetAb: Create netAb ab from chunk ab...\");\r\n                                if(frames===null){\r\n                                    subscriber.error(new Error('Could not get frame length from recording file object'));\r\n                                }else {\r\n                                    let fl = frames;\r\n                                    if (ab.sampleRate !== orgSampleRate) {\r\n                                        if (orgSampleRate && frames) {\r\n                                            fl = Math.round(ab.sampleRate * frames / orgSampleRate);\r\n                                            //console.debug(\"Platform sr: \"+ab.sampleRate+\", file sr: \"+orgSampleRate+\", decoded/org frame length: \"+fl+\"/\"+frames+\", ab.length: \"+ab.length);\r\n                                        }\r\n                                    }\r\n\r\n                                    let nab = NetAudioBuffer.fromChunkAudioBuffer(aCtx, this,baseAudioUrl, ab, fl,frameLength);\r\n                                    //let rp=new ReadyProvider();\r\n                                    //nab.readyProvider=rp;\r\n                                    //rp.ready();\r\n                                    nab.ready();\r\n                                    if (nab.frameLen < frameLength) {\r\n                                        //console.debug(\"chunkAudioRequestTonetAb: Built netAb ab from chunk ab: First chunk shorter tha frameLength (\"+netAbAudioBuffer.frameLen+\"<\"+frameLength+\"), assuming end of data, sealing netAb ab.\");\r\n                                        nab.seal();\r\n                                    }\r\n                                    subscriber.next(nab);\r\n\r\n                                    subscriber.complete();\r\n                                }\r\n                            }\r\n                            , error => {\r\n                                console.error('chunkAudioRequestToNetAb: error: '+error);\r\n                                //if(error instanceof HttpErrorResponse) {\r\n                                subscriber.error(error);\r\n                                //}\r\n                            })\r\n                    } else {\r\n                        console.error('chunkAudioRequestToNetAb: Fetching audio file: response has no body');\r\n                        subscriber.error('chunkAudioRequestToNetAb: Fetching audio file: response has no body');\r\n                    }\r\n                },\r\n                error:(error) => {\r\n                    console.error('chunkAudioRequestToNetAb: error: '+error);\r\n                    subscriber.error(error);\r\n                    //subscriber.complete();\r\n                }\r\n            });\r\n        });\r\n        return obs;\r\n    }\r\n\r\n}\r\n"]}
|