itslanguage
Advanced tools
Comparing version 3.0.0-beta-3 to 3.0.0-beta-4
@@ -279,2 +279,2 @@ 'use strict'; | ||
exports.default = ChoiceRecognitionController; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/administrative-sdk/choice-recognition/choice-recognition-controller.js"],"names":["ChoiceRecognitionController","connection","_connection","choiceRecognitionInitChallenge","challenge","call","_recognitionId","id","then","console","log","recognitionId","choiceRecognitionInitAudio","recorder","dataavailableCb","specs","getAudioSpecs","audioFormat","audioParameters","addEventListener","startStreamingChoiceRecognition","trim","Promise","reject","Error","_session","isRecording","self","trimAudioStart","trimAudioEnd","resolve","notify","_cb","data","recognition","userId","Date","created","updated","addAccessToken","audioUrl","recognised","_ecb","message","chunk","encoded","_arrayBufferToBase64","catch","error","res","recognitionInitCb","trimStart","trimEnd","p","hasUserMediaApproval","resolve_","recordedCb","kwargs","analysis","removeEventListener","logRPCError","getChoiceRecognition","challengeId","url","_settings","apiUrl","_secureAjaxGet","datum","getChoiceRecognitions","map"],"mappings":";;;;AAAA;;;;AACA;;;;AACA;;;;AACA;;;;AACA;;;;;;;;AACA;;;;IAIqBA,2B;AACnB;;;AAGA,uCAAYC,UAAZ,EAAwB;AAAA;;AACtB;;;;AAIA,SAAKC,WAAL,GAAmBD,UAAnB;AACD;;AAED;;;;;;;;wCAMAE,8B,2CAA+BC,S,EAAW;AAAA;;AACxC,WAAO,KAAKF,WAAL,CAAiBG,IAAjB,CAAsB,uBAAtB,EACL,CAAC,KAAKH,WAAL,CAAiBI,cAAlB,EAAkCF,UAAUG,EAA5C,CADK,EAEJC,IAFI;AAGH;AACA,6BAAiB;AACfC,cAAQC,GAAR,CAAY,8CAA8C,MAAKR,WAAL,CAAiBI,cAA3E;AACA,aAAOK,aAAP;AACD,KAPE,CAAP;AAQD,G;;AAED;;;;;;;;;wCAOAC,0B,uCAA2BC,Q,EAAUC,e,EAAiB;AAAA;;AACpD;AACA;AACA;AACA,QAAMC,QAAQF,SAASG,aAAT,EAAd;AACA,WAAO,KAAKd,WAAL,CAAiBG,IAAjB,CAAsB,mBAAtB,EACL,CAAC,KAAKH,WAAL,CAAiBI,cAAlB,EAAkCS,MAAME,WAAxC,CADK,EACiDF,MAAMG,eADvD,EACwEV,IADxE;AAEL;AACA,6BAAiB;AACfC,cAAQC,GAAR,CAAY,mEAAmE,OAAKR,WAAL,CAAiBI,cAAhG;AACA;AACAO,eAASM,gBAAT,CAA0B,eAA1B,EAA2CL,eAA3C;AACA,aAAOH,aAAP;AACD,KARI,CAAP;AASD,G;;AAED;;;;;;;;;;;;;;;;;wCAeAS,+B,4CAAgChB,S,EAAWS,Q,EAAUQ,I,EAAM;AACzD,QAAI,EAAEjB,8CAAF,CAAJ,EAA6C;AAC3C,aAAOkB,QAAQC,MAAR,CAAe,IAAIC,KAAJ,CAAU,2DAAV,CAAf,CAAP;AACD;;AAED,QAAI,OAAOpB,UAAUG,EAAjB,KAAwB,QAA5B,EAAsC;AACpC,aAAOe,QAAQC,MAAR,CAAe,IAAIC,KAAJ,CAAU,iDAAV,CAAf,CAAP;AACD;;AAED;AACA,QAAI,CAAC,KAAKtB,WAAL,CAAiBuB,QAAtB,EAAgC;AAC9B,aAAOH,QAAQC,MAAR,CAAe,IAAIC,KAAJ,CAAU,oCAAV,CAAf,CAAP;AACD;;AAED,QAAIX,SAASa,WAAT,EAAJ,EAA4B;AAC1B,aAAOJ,QAAQC,MAAR,CAAe,IAAIC,KAAJ,CAAU,uCAAV,CAAf,CAAP;AACD;;AAED,QAAI,KAAKtB,WAAL,CAAiBI,cAAjB,KAAoC,IAAxC,EAA8C;AAC5C,aAAOgB,QAAQC,MAAR,CAAe,IAAIC,KAAJ,CAAU,gCAAgC,KAAKtB,WAAL,CAAiBI,cAAjD,GAC9B,qBADoB,CAAf,CAAP;AAED;;AAED,QAAMqB,OAAO,IAAb;AACA,QAAIC,iBAAiB,IAArB;AACA,QAAMC,eAAe,GAArB;AACA,QAAIR,SAAS,KAAb,EAAoB;AAClBO,uBAAiB,GAAjB;AACD;AACD,WAAO,IAAI,eAAKN,OAAT,CAAiB,UAACQ,OAAD,EAAUP,MAAV,EAAkBQ,MAAlB,EAA6B;AACnD,eAASC,GAAT,CAAaC,IAAb,EAAmB;AACjB,YAAMC,cAAc,gCAClB9B,UAAUG,EADQ,EACJ0B,KAAKE,MADD,EACSF,KAAK1B,EADd,EAElB,IAAI6B,IAAJ,CAASH,KAAKI,OAAd,CAFkB,EAEM,IAAID,IAAJ,CAASH,KAAKK,OAAd,CAFN,EAGlBX,KAAKzB,WAAL,CAAiBqC,cAAjB,CAAgCN,KAAKO,QAArC,CAHkB,EAG8BP,KAAKQ,UAHnC,CAApB;AAIAX,gBAAQ,EAACnB,eAAegB,KAAKzB,WAAL,CAAiBI,cAAjC,EAAiD4B,wBAAjD,EAAR;AACD;;AAED,eAASQ,IAAT,CAAcT,IAAd,EAAoB;AAClB;AACA,YAAMC,cAAc,gCAClB9B,UAAUG,EADQ,EACJ0B,KAAKE,MADD,EACSF,KAAK1B,EADd,EAElB,IAAI6B,IAAJ,CAASH,KAAKI,OAAd,CAFkB,EAEM,IAAID,IAAJ,CAASH,KAAKK,OAAd,CAFN,EAGlBX,KAAKzB,WAAL,CAAiBqC,cAAjB,CAAgCN,KAAKO,QAArC,CAHkB,EAG8B,IAH9B,CAApB;AAIAjB,eACE;AACEW,kCADF;AAEES,mBAASV,KAAKU;AAFhB,SADF;AAMD;;AAEDhB,WAAKzB,WAAL,CAAiBI,cAAjB,GAAkC,IAAlC;;AAEA;AACA;AACA,eAASQ,eAAT,CAAyB8B,KAAzB,EAAgC;AAC9B,YAAMC,UAAU,sBAAYC,oBAAZ,CAAiCF,KAAjC,CAAhB;AACAnC,gBAAQC,GAAR,CAAY,yDACViB,KAAKzB,WAAL,CAAiBI,cADnB;AAEAqB,aAAKzB,WAAL,CAAiBG,IAAjB,CAAsB,cAAtB,EACE,CAACsB,KAAKzB,WAAL,CAAiBI,cAAlB,EAAkCuC,OAAlC,EAA2C,QAA3C,CADF,EAEGE,KAFH,CAES,eAAO;AACZtC,kBAAQuC,KAAR,CAAc,qBAAd,EAAqCC,IAAID,KAAzC;AACAN,eAAKO,GAAL;AACD,SALH;AAMD;;AAED,eAASC,iBAAT,CAA2BvC,aAA3B,EAA0C;AACxCgB,aAAKzB,WAAL,CAAiBI,cAAjB,GAAkCK,aAAlC;AACAF,gBAAQC,GAAR,CAAY,6CAA6CiB,KAAKzB,WAAL,CAAiBI,cAA1E;AACD;AACDqB,WAAKzB,WAAL,CAAiBG,IAAjB,CAAsB,yBAAtB,EAAiD,EAAjD,EACE;AACE8C,mBAAWvB,cADb;AAEEwB,iBAASvB;AAFX,OADF,EAKGrB,IALH,CAKQ0C,iBALR,EAMG1C,IANH,CAMQ;AAAA,eACJmB,KAAKxB,8BAAL,CAAoCC,SAApC,EACGI,IADH,CACQ,YAAM;AACV,cAAM6C,IAAI,IAAI/B,OAAJ,CAAY,oBAAY;AAChC,gBAAIT,SAASyC,oBAAT,EAAJ,EAAqC;AACnCC;AACD,aAFD,MAEO;AACL1C,uBAASM,gBAAT,CAA0B,OAA1B,EAAmCoC,QAAnC;AACD;AACF,WANS,CAAV;AAOAF,YAAE7C,IAAF,CAAO,YAAM;AACXmB,iBAAKf,0BAAL,CAAgCC,QAAhC,EAA0CC,eAA1C,EACGiC,KADH,CACSxB,MADT;AAED,WAHD;AAID,SAbH,EAcGf,IAdH,CAcQ;AAAA,iBAAMuB,OAAO,gBAAP,CAAN;AAAA,SAdR,CADI;AAAA,OANR,EAuBGgB,KAvBH,CAuBSxB,MAvBT;;AAyBA;AACA,eAASiC,UAAT,GAAsB;AACpB;AACA7B,aAAKzB,WAAL,CAAiBG,IAAjB,CAAsB,kBAAtB,EACE,CAACsB,KAAKzB,WAAL,CAAiBI,cAAlB,CADF,EACqCE,IADrC;AAEE;AACA,uBAAO;AACL;AACAwB,cAAIiB,GAAJ;AACD,SANH;AAOE;AACA,uBAAO;AACLxC,kBAAQuC,KAAR,CAAc,qBAAd,EAAqCC,IAAID,KAAzC;AACA,cAAIC,IAAID,KAAJ,KAAc,mCAAlB,EAAuD;AACrDC,gBAAIQ,MAAJ,CAAWvB,WAAX,CAAuBS,OAAvB,GAAiC,oBAAjC;AACD,WAFD,MAEO;AACLM,gBAAIQ,MAAJ,CAAWvB,WAAX,CAAuBS,OAAvB,GAAiC,iBAAjC;AACD;AACDD,eAAKO,IAAIQ,MAAJ,CAAWC,QAAhB;AACD,SAhBH;;AAkBA7C,iBAAS8C,mBAAT,CAA6B,UAA7B,EAAyCH,UAAzC;AACA3C,iBAAS8C,mBAAT,CAA6B,eAA7B,EAA8C7C,eAA9C;AACD;AACDD,eAASM,gBAAT,CAA0B,UAA1B,EAAsCqC,UAAtC;AACD,KA7FM,EA8FJhD,IA9FI,CA8FC,eAAO;AACXmB,WAAKzB,WAAL,CAAiBI,cAAjB,GAAkC,IAAlC;AACA,aAAOgB,QAAQQ,OAAR,CAAgBmB,GAAhB,CAAP;AACD,KAjGI,EAkGJF,KAlGI,CAkGE,iBAAS;AACdpB,WAAKzB,WAAL,CAAiBI,cAAjB,GAAkC,IAAlC;AACA,qCAAWsD,WAAX,CAAuBZ,KAAvB;AACA,aAAO1B,QAAQC,MAAR,CAAeyB,KAAf,CAAP;AACD,KAtGI,CAAP;AAuGD,G;;AAED;;;;;;;;;;;;;wCAWAa,oB,iCAAqBC,W,EAAanD,a,EAAe;AAC/C,QAAI,OAAOmD,WAAP,KAAuB,QAA3B,EAAqC;AACnC,aAAOxC,QAAQC,MAAR,CAAe,IAAIC,KAAJ,CAAU,oDAAV,CAAf,CAAP;AACD;;AAED,QAAI,OAAOb,aAAP,KAAyB,QAA7B,EAAuC;AACrC,aAAOW,QAAQC,MAAR,CAAe,IAAIC,KAAJ,CAAU,sDAAV,CAAf,CAAP;AACD;AACD,QAAMuC,MAAM,KAAK7D,WAAL,CAAiB8D,SAAjB,CAA2BC,MAA3B,GAAoC,qBAApC,GACVH,WADU,GACI,gBADJ,GACuBnD,aADnC;;AAGA,WAAO,KAAKT,WAAL,CAAiBgE,cAAjB,CAAgCH,GAAhC,EACJvD,IADI,CACC;AAAA,aAAS,gCAAsBsD,WAAtB,EAAmCK,MAAMhC,MAAzC,EACXgC,MAAM5D,EADK,EACD,IAAI6B,IAAJ,CAAS+B,MAAM9B,OAAf,CADC,EACwB,IAAID,IAAJ,CAAS+B,MAAM7B,OAAf,CADxB,EAEX6B,MAAM3B,QAFK,EAEK2B,MAAM1B,UAFX,CAAT;AAAA,KADD,CAAP;AAID,G;;AAED;;;;;;;;;;;wCASA2B,qB,kCAAsBN,W,EAAa;AACjC,QAAI,OAAOA,WAAP,KAAuB,QAA3B,EAAqC;AACnC,aAAOxC,QAAQC,MAAR,CAAe,IAAIC,KAAJ,CAAU,oDAAV,CAAf,CAAP;AACD;AACD,QAAMuC,MAAM,KAAK7D,WAAL,CAAiB8D,SAAjB,CAA2BC,MAA3B,GAAoC,qBAApC,GACVH,WADU,GACI,eADhB;AAEA,WAAO,KAAK5D,WAAL,CAAiBgE,cAAjB,CAAgCH,GAAhC,EACJvD,IADI,CACC;AAAA,aAAQyB,KAAKoC,GAAL,CAAS,iBAAS;AAC9B,YAAMnC,cAAc,gCAAsB4B,WAAtB,EAAmCK,MAAMhC,MAAzC,EAClBgC,MAAM5D,EADY,EACR,IAAI6B,IAAJ,CAAS+B,MAAM9B,OAAf,CADQ,EACiB,IAAID,IAAJ,CAAS+B,MAAM7B,OAAf,CADjB,EAElB6B,MAAM3B,QAFY,EAEF2B,MAAM1B,UAFJ,CAApB;AAGA,eAAOP,WAAP;AACD,OALa,CAAR;AAAA,KADD,CAAP;AAOD,G;;;;;kBA3PkBlC,2B","file":"choice-recognition-controller.js","sourcesContent":["import Base64Utils from '../utils/base64-utils';\nimport ChoiceChallenge from '../choice-challenge/choice-challenge';\nimport ChoiceRecognition from './choice-recognition';\nimport Connection from '../connection/connection-controller';\nimport when from 'when';\n/**\n * Controller class for the ChoiceRecognition model.\n * @private\n */\nexport default class ChoiceRecognitionController {\n  /**\n   * @param {Connection} connection - Object to use for making a connection to the REST API and Websocket server.\n   */\n  constructor(connection) {\n    /**\n     * Object to use for making a connection to the REST API and Websocket server.\n     * @type {Connection}\n     */\n    this._connection = connection;\n  }\n\n  /**\n   * Initialise the choice recognition challenge through RPCs.\n   *\n   * @param {ChoiceChallenge} challenge - ChoiceChallenge.\n   * @private\n   */\n  choiceRecognitionInitChallenge(challenge) {\n    return this._connection.call('choice.init_challenge',\n      [this._connection._recognitionId, challenge.id])\n      .then(\n        // RPC success callback\n        recognitionId => {\n          console.log('Challenge initialised for recognitionId: ' + this._connection._recognitionId);\n          return recognitionId;\n        });\n  }\n\n  /**\n   * Initialise the pronunciation analysis audio specs through RPCs.\n   *\n   * @param {AudioRecorder} recorder - AudioRecorder.\n   * @param {Function} dataavailableCb - Callback.\n   * @private\n   */\n  choiceRecognitionInitAudio(recorder, dataavailableCb) {\n    // Indicate to the socket server that we're about to start recording a\n    // challenge. This allows the socket server some time to fetch the metadata\n    // and reference audio to start the analysis when audio is actually submitted.\n    const specs = recorder.getAudioSpecs();\n    return this._connection.call('choice.init_audio',\n      [this._connection._recognitionId, specs.audioFormat], specs.audioParameters).then(\n      // RPC success callback\n      recognitionId => {\n        console.log('Accepted audio parameters for recognitionId after init_audio: ' + this._connection._recognitionId);\n        // Start listening for streaming data.\n        recorder.addEventListener('dataavailable', dataavailableCb);\n        return recognitionId;\n      });\n  }\n\n  /**\n   * Start a choice recognition from streaming audio.\n   *\n   * @param {ChoiceChallenge} challenge - The choice challenge to perform.\n   * @param {AudioRecorder} recorder - The audio recorder to extract audio from.\n   * @param {boolean} [trim=true] - Whether to trim the start and end of recorded audio.\n   * @returns {Promise.<ChoiceRecognition>} A {@link https://github.com/cujojs/when} Promise containing a {@link ChoiceRecognition}.\n   * @emits {string} 'ReadyToReceive' when the call is made to receive audio. The recorder can now send audio.\n   * @throws {Promise.<Error>} challenge parameter of type \"ChoiceChallenge\" is required.\n   * @throws {Promise.<Error>} challenge.id field of type \"string\" is required.\n   * @throws {Promise.<Error>} If the connection is not open.\n   * @throws {Promise.<Error>} If the recorder is already recording.\n   * @throws {Promise.<Error>} If a recognition session is already in progress.\n   * @throws {Promise.<Error>} If something went wrong during analysis.\n   */\n  startStreamingChoiceRecognition(challenge, recorder, trim) {\n    if (!(challenge instanceof ChoiceChallenge)) {\n      return Promise.reject(new Error('challenge parameter of type \"ChoiceChallenge\" is required'));\n    }\n\n    if (typeof challenge.id !== 'string') {\n      return Promise.reject(new Error('challenge.id field of type \"string\" is required'));\n    }\n\n    // Validate environment prerequisites.\n    if (!this._connection._session) {\n      return Promise.reject(new Error('WebSocket connection was not open.'));\n    }\n\n    if (recorder.isRecording()) {\n      return Promise.reject(new Error('Recorder should not yet be recording.'));\n    }\n\n    if (this._connection._recognitionId !== null) {\n      return Promise.reject(new Error('Session with recognitionId ' + this._connection._recognitionId +\n        ' still in progress.'));\n    }\n\n    const self = this;\n    let trimAudioStart = 0.15;\n    const trimAudioEnd = 0.0;\n    if (trim === false) {\n      trimAudioStart = 0.0;\n    }\n    return new when.Promise((resolve, reject, notify) => {\n      function _cb(data) {\n        const recognition = new ChoiceRecognition(\n          challenge.id, data.userId, data.id,\n          new Date(data.created), new Date(data.updated),\n          self._connection.addAccessToken(data.audioUrl), data.recognised);\n        resolve({recognitionId: self._connection._recognitionId, recognition});\n      }\n\n      function _ecb(data) {\n        // There was an unexpected error.\n        const recognition = new ChoiceRecognition(\n          challenge.id, data.userId, data.id,\n          new Date(data.created), new Date(data.updated),\n          self._connection.addAccessToken(data.audioUrl), null);\n        reject(\n          {\n            recognition,\n            message: data.message\n          }\n        );\n      }\n\n      self._connection._recognitionId = null;\n\n      // Start streaming the binary audio when the user instructs\n      // the audio recorder to start recording.\n      function dataavailableCb(chunk) {\n        const encoded = Base64Utils._arrayBufferToBase64(chunk);\n        console.log('Sending audio chunk to websocket for recognitionId: ' +\n          self._connection._recognitionId);\n        self._connection.call('choice.write',\n          [self._connection._recognitionId, encoded, 'base64'])\n          .catch(res => {\n            console.error('RPC error returned:', res.error);\n            _ecb(res);\n          });\n      }\n\n      function recognitionInitCb(recognitionId) {\n        self._connection._recognitionId = recognitionId;\n        console.log('Got recognitionId after initialisation: ' + self._connection._recognitionId);\n      }\n      self._connection.call('choice.init_recognition', [],\n        {\n          trimStart: trimAudioStart,\n          trimEnd: trimAudioEnd\n        })\n        .then(recognitionInitCb)\n        .then(() =>\n          self.choiceRecognitionInitChallenge(challenge)\n            .then(() => {\n              const p = new Promise(resolve_ => {\n                if (recorder.hasUserMediaApproval()) {\n                  resolve_();\n                } else {\n                  recorder.addEventListener('ready', resolve_);\n                }\n              });\n              p.then(() => {\n                self.choiceRecognitionInitAudio(recorder, dataavailableCb)\n                  .catch(reject);\n              });\n            })\n            .then(() => notify('ReadyToReceive'))\n        )\n        .catch(reject);\n\n      // Stop listening when the audio recorder stopped.\n      function recordedCb() {\n        // When done, submit any plain text (non-JSON) to start analysing.\n        self._connection.call('choice.recognise',\n          [self._connection._recognitionId]).then(\n          // RPC success callback\n          res => {\n            // Wait for analysis results to come back.\n            _cb(res);\n          },\n          // RPC error callback\n          res => {\n            console.error('RPC error returned:', res.error);\n            if (res.error === 'nl.itslanguage.recognition_failed') {\n              res.kwargs.recognition.message = 'Recognition failed';\n            } else {\n              res.kwargs.recognition.message = 'Unhandled error';\n            }\n            _ecb(res.kwargs.analysis);\n          });\n\n        recorder.removeEventListener('recorded', recordedCb);\n        recorder.removeEventListener('dataavailable', dataavailableCb);\n      }\n      recorder.addEventListener('recorded', recordedCb);\n    })\n      .then(res => {\n        self._connection._recognitionId = null;\n        return Promise.resolve(res);\n      })\n      .catch(error => {\n        self._connection._recognitionId = null;\n        Connection.logRPCError(error);\n        return Promise.reject(error);\n      });\n  }\n\n  /**\n   * Get a choice recognition in a choice challenge from the current active {@link Organisation} derived from\n   * the OAuth2 scope.\n   *\n   * @param {string} challengeId - Specify a choice challenge identifier.\n   * @param {string} recognitionId - Specify a choice recognition identifier.\n   * @returns {Promise.<ChoiceRecognition>} Promise containing a ChoiceRecognition.\n   * @throws {Promise.<Error>} challengeId parameter of type \"string\" is required.\n   * @throws {Promise.<Error>} recognitionId parameter of type \"string\" is required.\n   * @throws {Promise.<Error>} If no result could not be found.\n   */\n  getChoiceRecognition(challengeId, recognitionId) {\n    if (typeof challengeId !== 'string') {\n      return Promise.reject(new Error('challengeId parameter of type \"string\" is required'));\n    }\n\n    if (typeof recognitionId !== 'string') {\n      return Promise.reject(new Error('recognitionId parameter of type \"string\" is required'));\n    }\n    const url = this._connection._settings.apiUrl + '/challenges/choice/' +\n      challengeId + '/recognitions/' + recognitionId;\n\n    return this._connection._secureAjaxGet(url)\n      .then(datum => new ChoiceRecognition(challengeId, datum.userId,\n          datum.id, new Date(datum.created), new Date(datum.updated),\n          datum.audioUrl, datum.recognised));\n  }\n\n  /**\n   * Get and return all choice recognitions in a specific {@link ChoiceChallenge} from the current active\n   * {@link Organisation} derived from the OAuth2 scope.\n   *\n   * @param {string} challengeId - Specify a choice challenge to list speech recognitions for.\n   * @returns {Promise.<ChoiceRecognition[]>} Promise containing an array of ChoiceRecognitions.\n   * @throws {Promise.<Error>} challengeId parameter of type \"string\" is required.\n   * @throws {Promise.<Error>} If no result could not be found.\n   */\n  getChoiceRecognitions(challengeId) {\n    if (typeof challengeId !== 'string') {\n      return Promise.reject(new Error('challengeId parameter of type \"string\" is required'));\n    }\n    const url = this._connection._settings.apiUrl + '/challenges/choice/' +\n      challengeId + '/recognitions';\n    return this._connection._secureAjaxGet(url)\n      .then(data => data.map(datum => {\n        const recognition = new ChoiceRecognition(challengeId, datum.userId,\n          datum.id, new Date(datum.created), new Date(datum.updated),\n          datum.audioUrl, datum.recognised);\n        return recognition;\n      }));\n  }\n}\n"]} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/administrative-sdk/choice-recognition/choice-recognition-controller.js"],"names":["ChoiceRecognitionController","connection","_connection","choiceRecognitionInitChallenge","challenge","call","_recognitionId","id","then","console","log","recognitionId","choiceRecognitionInitAudio","recorder","dataavailableCb","specs","getAudioSpecs","audioFormat","audioParameters","addEventListener","startStreamingChoiceRecognition","trim","Promise","reject","Error","_session","isRecording","self","trimAudioStart","trimAudioEnd","resolve","notify","_cb","data","recognition","userId","Date","created","updated","addAccessToken","audioUrl","recognised","_ecb","message","chunk","encoded","_arrayBufferToBase64","catch","error","res","recognitionInitCb","trimStart","trimEnd","p","hasUserMediaApproval","resolve_","recordedCb","kwargs","analysis","removeEventListener","logRPCError","getChoiceRecognition","challengeId","url","_settings","apiUrl","_secureAjaxGet","datum","getChoiceRecognitions","map"],"mappings":"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","file":"choice-recognition-controller.js","sourcesContent":["import Base64Utils from '../utils/base64-utils';\nimport ChoiceChallenge from '../choice-challenge/choice-challenge';\nimport ChoiceRecognition from './choice-recognition';\nimport Connection from '../connection/connection-controller';\nimport when from 'when';\n/**\n * Controller class for the ChoiceRecognition model.\n * @private\n */\nexport default class ChoiceRecognitionController {\n  /**\n   * @param {Connection} connection - Object to use for making a connection to the REST API and Websocket server.\n   */\n  constructor(connection) {\n    /**\n     * Object to use for making a connection to the REST API and Websocket server.\n     * @type {Connection}\n     */\n    this._connection = connection;\n  }\n\n  /**\n   * Initialise the choice recognition challenge through RPCs.\n   *\n   * @param {ChoiceChallenge} challenge - ChoiceChallenge.\n   * @private\n   */\n  choiceRecognitionInitChallenge(challenge) {\n    return this._connection.call('choice.init_challenge',\n      [this._connection._recognitionId, challenge.id])\n      .then(\n        // RPC success callback\n        recognitionId => {\n          console.log('Challenge initialised for recognitionId: ' + this._connection._recognitionId);\n          return recognitionId;\n        });\n  }\n\n  /**\n   * Initialise the pronunciation analysis audio specs through RPCs.\n   *\n   * @param {AudioRecorder} recorder - AudioRecorder.\n   * @param {Function} dataavailableCb - Callback.\n   * @private\n   */\n  choiceRecognitionInitAudio(recorder, dataavailableCb) {\n    // Indicate to the socket server that we're about to start recording a\n    // challenge. This allows the socket server some time to fetch the metadata\n    // and reference audio to start the analysis when audio is actually submitted.\n    const specs = recorder.getAudioSpecs();\n    return this._connection.call('choice.init_audio',\n      [this._connection._recognitionId, specs.audioFormat], specs.audioParameters).then(\n      // RPC success callback\n      recognitionId => {\n        console.log('Accepted audio parameters for recognitionId after init_audio: ' + this._connection._recognitionId);\n        // Start listening for streaming data.\n        recorder.addEventListener('dataavailable', dataavailableCb);\n        return recognitionId;\n      });\n  }\n\n  /**\n   * Start a choice recognition from streaming audio.\n   *\n   * @param {ChoiceChallenge} challenge - The choice challenge to perform.\n   * @param {AudioRecorder} recorder - The audio recorder to extract audio from.\n   * @param {boolean} [trim=true] - Whether to trim the start and end of recorded audio.\n   * @returns {Promise.<ChoiceRecognition>} A {@link https://github.com/cujojs/when} Promise containing a {@link ChoiceRecognition}.\n   * @emits {string} 'ReadyToReceive' when the call is made to receive audio. The recorder can now send audio.\n   * @throws {Promise.<Error>} challenge parameter of type \"ChoiceChallenge\" is required.\n   * @throws {Promise.<Error>} challenge.id field of type \"string\" is required.\n   * @throws {Promise.<Error>} If the connection is not open.\n   * @throws {Promise.<Error>} If the recorder is already recording.\n   * @throws {Promise.<Error>} If a recognition session is already in progress.\n   * @throws {Promise.<Error>} If something went wrong during analysis.\n   */\n  startStreamingChoiceRecognition(challenge, recorder, trim) {\n    if (!(challenge instanceof ChoiceChallenge)) {\n      return Promise.reject(new Error('challenge parameter of type \"ChoiceChallenge\" is required'));\n    }\n\n    if (typeof challenge.id !== 'string') {\n      return Promise.reject(new Error('challenge.id field of type \"string\" is required'));\n    }\n\n    // Validate environment prerequisites.\n    if (!this._connection._session) {\n      return Promise.reject(new Error('WebSocket connection was not open.'));\n    }\n\n    if (recorder.isRecording()) {\n      return Promise.reject(new Error('Recorder should not yet be recording.'));\n    }\n\n    if (this._connection._recognitionId !== null) {\n      return Promise.reject(new Error('Session with recognitionId ' + this._connection._recognitionId +\n        ' still in progress.'));\n    }\n\n    const self = this;\n    let trimAudioStart = 0.15;\n    const trimAudioEnd = 0.0;\n    if (trim === false) {\n      trimAudioStart = 0.0;\n    }\n    return new when.Promise((resolve, reject, notify) => {\n      function _cb(data) {\n        const recognition = new ChoiceRecognition(\n          challenge.id, data.userId, data.id,\n          new Date(data.created), new Date(data.updated),\n          self._connection.addAccessToken(data.audioUrl), data.recognised);\n        resolve({recognitionId: self._connection._recognitionId, recognition});\n      }\n\n      function _ecb(data) {\n        // There was an unexpected error.\n        const recognition = new ChoiceRecognition(\n          challenge.id, data.userId, data.id,\n          new Date(data.created), new Date(data.updated),\n          self._connection.addAccessToken(data.audioUrl), null);\n        reject(\n          {\n            recognition,\n            message: data.message\n          }\n        );\n      }\n\n      self._connection._recognitionId = null;\n\n      // Start streaming the binary audio when the user instructs\n      // the audio recorder to start recording.\n      function dataavailableCb(chunk) {\n        const encoded = Base64Utils._arrayBufferToBase64(chunk);\n        console.log('Sending audio chunk to websocket for recognitionId: ' +\n          self._connection._recognitionId);\n        self._connection.call('choice.write',\n          [self._connection._recognitionId, encoded, 'base64'])\n          .catch(res => {\n            console.error('RPC error returned:', res.error);\n            _ecb(res);\n          });\n      }\n\n      function recognitionInitCb(recognitionId) {\n        self._connection._recognitionId = recognitionId;\n        console.log('Got recognitionId after initialisation: ' + self._connection._recognitionId);\n      }\n      self._connection.call('choice.init_recognition', [],\n        {\n          trimStart: trimAudioStart,\n          trimEnd: trimAudioEnd\n        })\n        .then(recognitionInitCb)\n        .then(() =>\n          self.choiceRecognitionInitChallenge(challenge)\n            .then(() => {\n              const p = new Promise(resolve_ => {\n                if (recorder.hasUserMediaApproval()) {\n                  resolve_();\n                } else {\n                  recorder.addEventListener('ready', resolve_);\n                }\n              });\n              p.then(() => {\n                self.choiceRecognitionInitAudio(recorder, dataavailableCb)\n                  .catch(reject);\n              });\n            })\n            .then(() => notify('ReadyToReceive'))\n        )\n        .catch(reject);\n\n      // Stop listening when the audio recorder stopped.\n      function recordedCb() {\n        // When done, submit any plain text (non-JSON) to start analysing.\n        self._connection.call('choice.recognise',\n          [self._connection._recognitionId]).then(\n          // RPC success callback\n          res => {\n            // Wait for analysis results to come back.\n            _cb(res);\n          },\n          // RPC error callback\n          res => {\n            console.error('RPC error returned:', res.error);\n            if (res.error === 'nl.itslanguage.recognition_failed') {\n              res.kwargs.recognition.message = 'Recognition failed';\n            } else {\n              res.kwargs.recognition.message = 'Unhandled error';\n            }\n            _ecb(res.kwargs.analysis);\n          });\n\n        recorder.removeEventListener('recorded', recordedCb);\n        recorder.removeEventListener('dataavailable', dataavailableCb);\n      }\n      recorder.addEventListener('recorded', recordedCb);\n    })\n      .then(res => {\n        self._connection._recognitionId = null;\n        return Promise.resolve(res);\n      })\n      .catch(error => {\n        self._connection._recognitionId = null;\n        Connection.logRPCError(error);\n        return Promise.reject(error);\n      });\n  }\n\n  /**\n   * Get a choice recognition in a choice challenge from the current active {@link Organisation} derived from\n   * the OAuth2 scope.\n   *\n   * @param {string} challengeId - Specify a choice challenge identifier.\n   * @param {string} recognitionId - Specify a choice recognition identifier.\n   * @returns {Promise.<ChoiceRecognition>} Promise containing a ChoiceRecognition.\n   * @throws {Promise.<Error>} challengeId parameter of type \"string\" is required.\n   * @throws {Promise.<Error>} recognitionId parameter of type \"string\" is required.\n   * @throws {Promise.<Error>} If no result could not be found.\n   */\n  getChoiceRecognition(challengeId, recognitionId) {\n    if (typeof challengeId !== 'string') {\n      return Promise.reject(new Error('challengeId parameter of type \"string\" is required'));\n    }\n\n    if (typeof recognitionId !== 'string') {\n      return Promise.reject(new Error('recognitionId parameter of type \"string\" is required'));\n    }\n    const url = this._connection._settings.apiUrl + '/challenges/choice/' +\n      challengeId + '/recognitions/' + recognitionId;\n\n    return this._connection._secureAjaxGet(url)\n      .then(datum => new ChoiceRecognition(challengeId, datum.userId,\n        datum.id, new Date(datum.created), new Date(datum.updated),\n        datum.audioUrl, datum.recognised));\n  }\n\n  /**\n   * Get and return all choice recognitions in a specific {@link ChoiceChallenge} from the current active\n   * {@link Organisation} derived from the OAuth2 scope.\n   *\n   * @param {string} challengeId - Specify a choice challenge to list speech recognitions for.\n   * @returns {Promise.<ChoiceRecognition[]>} Promise containing an array of ChoiceRecognitions.\n   * @throws {Promise.<Error>} challengeId parameter of type \"string\" is required.\n   * @throws {Promise.<Error>} If no result could not be found.\n   */\n  getChoiceRecognitions(challengeId) {\n    if (typeof challengeId !== 'string') {\n      return Promise.reject(new Error('challengeId parameter of type \"string\" is required'));\n    }\n    const url = this._connection._settings.apiUrl + '/challenges/choice/' +\n      challengeId + '/recognitions';\n    return this._connection._secureAjaxGet(url)\n      .then(data => data.map(datum => {\n        const recognition = new ChoiceRecognition(challengeId, datum.userId,\n          datum.id, new Date(datum.created), new Date(datum.updated),\n          datum.audioUrl, datum.recognised);\n        return recognition;\n      }));\n  }\n}\n"]} |
@@ -25,3 +25,2 @@ 'use strict'; | ||
var Connection = function () { | ||
/** | ||
@@ -351,2 +350,2 @@ * | ||
exports.default = Connection; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/administrative-sdk/connection/connection-controller.js"],"names":["Connection","options","_settings","Object","assign","apiUrl","oAuth2Token","wsUrl","wsToken","_sdkCompatibility","_analysisId","_recordingId","_recognitionId","_emitter","_connection","authorizationToken","addEventListener","name","handler","on","removeEventListener","off","fireEvent","args","emit","webSocketConnect","self","onOAuth2Challenge","session","method","Error","authUrl","connection","url","realm","authmethods","authid","details","ticket","onchallenge","e","console","log","onerror","onopen","_session","onclose","open","call","rpc","debug","webSocketDisconnect","close","_secureAjaxGet","_secureAjaxPost","formdata","_secureAjaxDelete","addAccessToken","secureUrl","match","encodeURIComponent","window","cancelStreaming","recorder","info","removeAllEventListeners","isRecording","stop","logRPCError","result","error","getOauth2Token","basicAuth","scope","body","URLSearchParams","append","principal","credentials","then","response","access_token","getUserAuth","organisationId","scopes","tenantId"],"mappings":"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","file":"connection-controller.js","sourcesContent":["/* eslint-disable\ncamelcase\n */\n\nimport {authorisedRequest, request, updateSettings} from '../../api/communication';\n\nimport autobahn from 'autobahn';\nimport ee from 'event-emitter';\n/**\n * Controller class for managing connection interaction.\n */\nexport default class Connection {\n\n  /**\n   *\n   * @param {Object} options - Options to configure the connection with.\n   * Valid options include:\n   * * apiUrl - The URL of the REST api.\n   * * wsUrl - The URL of the Websocket server.\n   * * oAuth2Token - An OAuth2 token string.\n   * * adminPrincipal - The username of the admin account.\n   * * adminPassword - The password of the admin account.\n   */\n  constructor(options = {}) {\n    /**\n     * @type {Object}\n     */\n    this._settings = Object.assign({\n      // ITSL connection parameters.\n      apiUrl: 'https://api.itslanguage.nl',\n      oAuth2Token: null,\n      wsUrl: null,\n      wsToken: null\n    }, options);\n    Connection._sdkCompatibility();\n    this._analysisId = null;\n    this._recordingId = null;\n    this._recognitionId = null;\n    this._emitter = ee({});\n    this._connection = null;\n\n    // Use the new connection file for future requests.\n    updateSettings(Object.assign({}, options, {authorizationToken: options.oAuth2Token}));\n  }\n\n  /**\n   * Add an event listener. Listens to events emitted from the websocket server connection.\n   *\n   * @param {string} name - Name of the event.\n   * @param {Function} handler - Handler function to add.\n   */\n  addEventListener(name, handler) {\n    this._emitter.on(name, handler);\n  }\n\n  /**\n   * Remove an event listener of the websocket connection.\n   *\n   * @param {string} name - Name of the event.\n   * @param {Function} handler - Handler function to remove.\n   */\n  removeEventListener(name, handler) {\n    this._emitter.off(name, handler);\n  }\n\n  /**\n   * Fire an event.\n   *\n   * @param {string} name - Name of the event.\n   * @param {[]} args - Arguments.\n   * @private\n   */\n  fireEvent(name, args = []) {\n    this._emitter.emit(name, ...args);\n  }\n\n  /**\n   * Create a connection to the websocket server.\n   *\n   */\n  webSocketConnect() {\n    const self = this;\n    /**\n     * This callback is fired during Ticket-based authentication.\n     *\n     * @param {Session} session - Session.\n     * @param {string} method - Authentication method.\n     */\n    function onOAuth2Challenge(session, method) {\n      if (method === 'ticket') {\n        return self._settings.oAuth2Token;\n      }\n      throw new Error(`don't know how to authenticate using '${method}'`);\n    }\n\n    const authUrl = this._settings.wsUrl;\n    let connection = null;\n    // Open a websocket connection for streaming audio\n    try {\n      // Set up WAMP connection to router\n      connection = new autobahn.Connection({\n        url: authUrl,\n        realm: 'default',\n        // the following attributes must be set for Ticket-based authentication\n        authmethods: ['ticket'],\n        authid: 'oauth2',\n        details: {\n          ticket: this._settings.oAuth2Token\n        },\n        onchallenge: onOAuth2Challenge\n      });\n    } catch (e) {\n      console.log('WebSocket creation error: ' + e);\n      return;\n    }\n    connection.onerror = function(e) {\n      console.log('WebSocket error: ' + e);\n      self.fireEvent('websocketError', [e]);\n    };\n    connection.onopen = function(session) {\n      console.log('WebSocket connection opened');\n      self._session = session;\n      self.fireEvent('websocketOpened');\n    };\n    connection.onclose = function() {\n      console.log('WebSocket disconnected');\n      self._session = null;\n      self.fireEvent('websocketClosed');\n    };\n    this._connection = connection;\n    this._connection.open();\n  }\n\n  /**\n   * Make an RPC to active current session.\n   *\n   * @param {string} rpc - The RPC to call. It will be prefixed with `'nl.itslanguage.'`.\n   * @param {...any} args - Any arguments to pass to the RPC.\n   * @return {Promise} The result of the call.\n   */\n  call(rpc, ...args) {\n    const url = 'nl.itslanguage.' + rpc;\n    console.debug('Calling RPC:', url);\n    return this._session.call(url, ...args);\n  }\n\n  webSocketDisconnect() {\n    this._connection.close(null, 'Requested formal disconnect');\n  }\n\n  /**\n   * Perform a HTTP GET to the API using authentication.\n   *\n   * @param {string} url - Url to retrieve.\n   * @returns {Promise} Promise containing a result.\n   * @throws {Promise.<Error>} If the server returned an error.\n   */\n  _secureAjaxGet(url) {\n    return authorisedRequest('GET', url);\n  }\n\n  /**\n   * Perform a HTTP POST to the API using authentication.\n   *\n   * @param {string} url - Url to submit to.\n   * @param {FormData} formdata - The form to POST.\n   * @returns {Promise} Promise containing a result.\n   * @throws {Promise.<Error>} If the server returned an error.\n   */\n  _secureAjaxPost(url, formdata) {\n    return authorisedRequest('POST', url, formdata);\n  }\n\n  /**\n   * Perform a HTTP DELETE to the API using authentication.\n   *\n   * @param {string} url - Url to submit to.\n   * @returns {Promise} Promise containing a result.\n   * @throws {Promise.<Error>} If the server returned an error.\n   */\n  _secureAjaxDelete(url) {\n    return authorisedRequest('DELETE', url);\n  }\n\n  /**\n   * Add an access token to the given URL.\n   *\n   * @param {string} url - The URL to add an access token to.\n   * @returns {string} An url with the access token appended.\n   */\n  addAccessToken(url) {\n    if (!this._settings.oAuth2Token) {\n      throw new Error('Please set oAuth2Token');\n    }\n    const secureUrl = url + (url.match(/\\?/) ? '&' : '?') + 'access_token=' +\n      encodeURIComponent(this._settings.oAuth2Token);\n    return secureUrl;\n  }\n\n  /**\n   * Logs browser compatibility for required and optional SDK capabilities.\n   *\n   * @throws {Error} In case of compatibility issues.\n   */\n  static _sdkCompatibility() {\n    // WebSocket\n    // http://caniuse.com/#feat=websockets\n    if (!('WebSocket' in window)) {\n      throw new Error('No WebSocket capabilities');\n    }\n  }\n\n  /**\n   * Cancel any current streaming audio recording.\n   *\n   * @param {AudioRecorder} recorder - The audio recorder currently recording.\n   */\n  cancelStreaming(recorder) {\n    const self = this;\n\n    if (this._recordingId === null && this._analysisId === null && this._recognitionId === null) {\n      console.info('No session in progress, nothing to cancel.');\n      return;\n    }\n\n    recorder.removeAllEventListeners();\n    if (recorder.isRecording()) {\n      recorder.stop();\n    }\n\n    // This session is over.\n    self._recordingId = null;\n    self._analysisId = null;\n    self._recognitionId = null;\n  }\n\n  /**\n   * Log an error caught from an RPC call.\n   *\n   * @param {Object} result - Error object.\n   */\n  static logRPCError(result) {\n    console.error('RPC error returned:', result.error);\n  }\n\n  /**\n   * Ask the server for an OAuth2 token.\n   *\n   * @param {BasicAuth} basicAuth - Basic Auth to obtain credentials from.\n   * @param {string} [scope] - The scope which should be availible for the requested token.\n   * @returns {Promise} Promise containing a access_token, token_type and scope.\n   * @throws {Promise.<Error>} If the server returned an error.\n   */\n  getOauth2Token(basicAuth, scope) {\n    const body = new URLSearchParams();\n    body.append('grant_type', 'password');\n    body.append('username', basicAuth.principal);\n    body.append('password', basicAuth.credentials);\n\n    if (scope) {\n      body.append('scope', scope);\n    }\n\n    return request('POST', '/tokens', body).then(response => {\n      this._settings.oAuth2Token = response.access_token;\n      updateSettings({authorizationToken: response.access_token});\n    });\n  }\n\n  /**\n   * Request authentication for a {@link User}. The basicAuth now contains the user's username and password.\n   *\n   * This method also generates the appropriate scope for the given params.\n   *\n   * @param {BasicAuth} basicAuth - Basic Auth to obtain credentials from.\n   * @param {string} organisationId - Id of the organisation this user is part of.\n   */\n  getUserAuth(basicAuth, organisationId) {\n    let scopes = 'tenant/' + basicAuth.tenantId;\n    if (organisationId) {\n      scopes += '/organisation/' + organisationId;\n      if (basicAuth.principal) {\n        scopes += '/user/' + basicAuth.principal;\n      }\n    }\n\n    return this.getOauth2Token(basicAuth, scopes);\n  }\n}\n"]} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/administrative-sdk/connection/connection-controller.js"],"names":["Connection","options","_settings","Object","assign","apiUrl","oAuth2Token","wsUrl","wsToken","_sdkCompatibility","_analysisId","_recordingId","_recognitionId","_emitter","_connection","authorizationToken","addEventListener","name","handler","on","removeEventListener","off","fireEvent","args","emit","webSocketConnect","self","onOAuth2Challenge","session","method","Error","authUrl","connection","url","realm","authmethods","authid","details","ticket","onchallenge","e","console","log","onerror","onopen","_session","onclose","open","call","rpc","debug","webSocketDisconnect","close","_secureAjaxGet","_secureAjaxPost","formdata","_secureAjaxDelete","addAccessToken","secureUrl","match","encodeURIComponent","window","cancelStreaming","recorder","info","removeAllEventListeners","isRecording","stop","logRPCError","result","error","getOauth2Token","basicAuth","scope","body","URLSearchParams","append","principal","credentials","then","response","access_token","getUserAuth","organisationId","scopes","tenantId"],"mappings":"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","file":"connection-controller.js","sourcesContent":["/* eslint-disable\ncamelcase\n */\n\nimport {authorisedRequest, request, updateSettings} from '../../api/communication';\n\nimport autobahn from 'autobahn';\nimport ee from 'event-emitter';\n/**\n * Controller class for managing connection interaction.\n */\nexport default class Connection {\n  /**\n   *\n   * @param {Object} options - Options to configure the connection with.\n   * Valid options include:\n   * * apiUrl - The URL of the REST api.\n   * * wsUrl - The URL of the Websocket server.\n   * * oAuth2Token - An OAuth2 token string.\n   * * adminPrincipal - The username of the admin account.\n   * * adminPassword - The password of the admin account.\n   */\n  constructor(options = {}) {\n    /**\n     * @type {Object}\n     */\n    this._settings = Object.assign({\n      // ITSL connection parameters.\n      apiUrl: 'https://api.itslanguage.nl',\n      oAuth2Token: null,\n      wsUrl: null,\n      wsToken: null\n    }, options);\n    Connection._sdkCompatibility();\n    this._analysisId = null;\n    this._recordingId = null;\n    this._recognitionId = null;\n    this._emitter = ee({});\n    this._connection = null;\n\n    // Use the new connection file for future requests.\n    updateSettings(Object.assign({}, options, {authorizationToken: options.oAuth2Token}));\n  }\n\n  /**\n   * Add an event listener. Listens to events emitted from the websocket server connection.\n   *\n   * @param {string} name - Name of the event.\n   * @param {Function} handler - Handler function to add.\n   */\n  addEventListener(name, handler) {\n    this._emitter.on(name, handler);\n  }\n\n  /**\n   * Remove an event listener of the websocket connection.\n   *\n   * @param {string} name - Name of the event.\n   * @param {Function} handler - Handler function to remove.\n   */\n  removeEventListener(name, handler) {\n    this._emitter.off(name, handler);\n  }\n\n  /**\n   * Fire an event.\n   *\n   * @param {string} name - Name of the event.\n   * @param {[]} args - Arguments.\n   * @private\n   */\n  fireEvent(name, args = []) {\n    this._emitter.emit(name, ...args);\n  }\n\n  /**\n   * Create a connection to the websocket server.\n   *\n   */\n  webSocketConnect() {\n    const self = this;\n    /**\n     * This callback is fired during Ticket-based authentication.\n     *\n     * @param {Session} session - Session.\n     * @param {string} method - Authentication method.\n     */\n    function onOAuth2Challenge(session, method) {\n      if (method === 'ticket') {\n        return self._settings.oAuth2Token;\n      }\n      throw new Error(`don't know how to authenticate using '${method}'`);\n    }\n\n    const authUrl = this._settings.wsUrl;\n    let connection = null;\n    // Open a websocket connection for streaming audio\n    try {\n      // Set up WAMP connection to router\n      connection = new autobahn.Connection({\n        url: authUrl,\n        realm: 'default',\n        // the following attributes must be set for Ticket-based authentication\n        authmethods: ['ticket'],\n        authid: 'oauth2',\n        details: {\n          ticket: this._settings.oAuth2Token\n        },\n        onchallenge: onOAuth2Challenge\n      });\n    } catch (e) {\n      console.log('WebSocket creation error: ' + e);\n      return;\n    }\n    connection.onerror = function(e) {\n      console.log('WebSocket error: ' + e);\n      self.fireEvent('websocketError', [e]);\n    };\n    connection.onopen = function(session) {\n      console.log('WebSocket connection opened');\n      self._session = session;\n      self.fireEvent('websocketOpened');\n    };\n    connection.onclose = function() {\n      console.log('WebSocket disconnected');\n      self._session = null;\n      self.fireEvent('websocketClosed');\n    };\n    this._connection = connection;\n    this._connection.open();\n  }\n\n  /**\n   * Make an RPC to active current session.\n   *\n   * @param {string} rpc - The RPC to call. It will be prefixed with `'nl.itslanguage.'`.\n   * @param {...any} args - Any arguments to pass to the RPC.\n   * @return {Promise} The result of the call.\n   */\n  call(rpc, ...args) {\n    const url = 'nl.itslanguage.' + rpc;\n    console.debug('Calling RPC:', url);\n    return this._session.call(url, ...args);\n  }\n\n  webSocketDisconnect() {\n    this._connection.close(null, 'Requested formal disconnect');\n  }\n\n  /**\n   * Perform a HTTP GET to the API using authentication.\n   *\n   * @param {string} url - Url to retrieve.\n   * @returns {Promise} Promise containing a result.\n   * @throws {Promise.<Error>} If the server returned an error.\n   */\n  _secureAjaxGet(url) {\n    return authorisedRequest('GET', url);\n  }\n\n  /**\n   * Perform a HTTP POST to the API using authentication.\n   *\n   * @param {string} url - Url to submit to.\n   * @param {FormData} formdata - The form to POST.\n   * @returns {Promise} Promise containing a result.\n   * @throws {Promise.<Error>} If the server returned an error.\n   */\n  _secureAjaxPost(url, formdata) {\n    return authorisedRequest('POST', url, formdata);\n  }\n\n  /**\n   * Perform a HTTP DELETE to the API using authentication.\n   *\n   * @param {string} url - Url to submit to.\n   * @returns {Promise} Promise containing a result.\n   * @throws {Promise.<Error>} If the server returned an error.\n   */\n  _secureAjaxDelete(url) {\n    return authorisedRequest('DELETE', url);\n  }\n\n  /**\n   * Add an access token to the given URL.\n   *\n   * @param {string} url - The URL to add an access token to.\n   * @returns {string} An url with the access token appended.\n   */\n  addAccessToken(url) {\n    if (!this._settings.oAuth2Token) {\n      throw new Error('Please set oAuth2Token');\n    }\n    const secureUrl = url + (url.match(/\\?/) ? '&' : '?') + 'access_token=' +\n      encodeURIComponent(this._settings.oAuth2Token);\n    return secureUrl;\n  }\n\n  /**\n   * Logs browser compatibility for required and optional SDK capabilities.\n   *\n   * @throws {Error} In case of compatibility issues.\n   */\n  static _sdkCompatibility() {\n    // WebSocket\n    // http://caniuse.com/#feat=websockets\n    if (!('WebSocket' in window)) {\n      throw new Error('No WebSocket capabilities');\n    }\n  }\n\n  /**\n   * Cancel any current streaming audio recording.\n   *\n   * @param {AudioRecorder} recorder - The audio recorder currently recording.\n   */\n  cancelStreaming(recorder) {\n    const self = this;\n\n    if (this._recordingId === null && this._analysisId === null && this._recognitionId === null) {\n      console.info('No session in progress, nothing to cancel.');\n      return;\n    }\n\n    recorder.removeAllEventListeners();\n    if (recorder.isRecording()) {\n      recorder.stop();\n    }\n\n    // This session is over.\n    self._recordingId = null;\n    self._analysisId = null;\n    self._recognitionId = null;\n  }\n\n  /**\n   * Log an error caught from an RPC call.\n   *\n   * @param {Object} result - Error object.\n   */\n  static logRPCError(result) {\n    console.error('RPC error returned:', result.error);\n  }\n\n  /**\n   * Ask the server for an OAuth2 token.\n   *\n   * @param {BasicAuth} basicAuth - Basic Auth to obtain credentials from.\n   * @param {string} [scope] - The scope which should be availible for the requested token.\n   * @returns {Promise} Promise containing a access_token, token_type and scope.\n   * @throws {Promise.<Error>} If the server returned an error.\n   */\n  getOauth2Token(basicAuth, scope) {\n    const body = new URLSearchParams();\n    body.append('grant_type', 'password');\n    body.append('username', basicAuth.principal);\n    body.append('password', basicAuth.credentials);\n\n    if (scope) {\n      body.append('scope', scope);\n    }\n\n    return request('POST', '/tokens', body).then(response => {\n      this._settings.oAuth2Token = response.access_token;\n      updateSettings({authorizationToken: response.access_token});\n    });\n  }\n\n  /**\n   * Request authentication for a {@link User}. The basicAuth now contains the user's username and password.\n   *\n   * This method also generates the appropriate scope for the given params.\n   *\n   * @param {BasicAuth} basicAuth - Basic Auth to obtain credentials from.\n   * @param {string} organisationId - Id of the organisation this user is part of.\n   */\n  getUserAuth(basicAuth, organisationId) {\n    let scopes = 'tenant/' + basicAuth.tenantId;\n    if (organisationId) {\n      scopes += '/organisation/' + organisationId;\n      if (basicAuth.principal) {\n        scopes += '/user/' + basicAuth.principal;\n      }\n    }\n\n    return this.getOauth2Token(basicAuth, scopes);\n  }\n}\n"]} |
@@ -107,2 +107,2 @@ 'use strict'; | ||
exports.default = GroupController; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9hZG1pbmlzdHJhdGl2ZS1zZGsvZ3JvdXAvZ3JvdXAtY29udHJvbGxlci5qcyJdLCJuYW1lcyI6WyJHcm91cENvbnRyb2xsZXIiLCJjb25uZWN0aW9uIiwiX2Nvbm5lY3Rpb24iLCJjcmVhdGVHcm91cCIsImdyb3VwIiwiUHJvbWlzZSIsInJlamVjdCIsIkVycm9yIiwidXJsIiwiX3NldHRpbmdzIiwiYXBpVXJsIiwiZmQiLCJKU09OIiwic3RyaW5naWZ5IiwiX3NlY3VyZUFqYXhQb3N0IiwidGhlbiIsInJlc3VsdCIsImRhdGEiLCJpZCIsIm5hbWUiLCJjcmVhdGVkIiwiRGF0ZSIsInVwZGF0ZWQiLCJnZXRHcm91cCIsImdyb3VwSWQiLCJfc2VjdXJlQWpheEdldCIsIm9yZ2FuaXNhdGlvbiIsImdldEdyb3VwcyIsImdyb3VwcyIsImZvckVhY2giLCJkYXR1bSIsInB1c2giXSwibWFwcGluZ3MiOiI7Ozs7QUFBQTs7Ozs7Ozs7QUFFQTs7OztJQUlxQkEsZTtBQUNuQjs7O0FBR0EsMkJBQVlDLFVBQVosRUFBd0I7QUFBQTs7QUFDdEI7Ozs7QUFJQSxTQUFLQyxXQUFMLEdBQW1CRCxVQUFuQjtBQUNEOztBQUVEOzs7Ozs7Ozs7OzRCQVFBRSxXLHdCQUFZQyxLLEVBQU87QUFDakIsUUFBSSxFQUFFQSxnQ0FBRixDQUFKLEVBQStCO0FBQzdCLGFBQU9DLFFBQVFDLE1BQVIsQ0FBZSxJQUFJQyxLQUFKLENBQVUsNkNBQVYsQ0FBZixDQUFQO0FBQ0Q7QUFDRCxRQUFNQyxNQUFNLEtBQUtOLFdBQUwsQ0FBaUJPLFNBQWpCLENBQTJCQyxNQUEzQixHQUFvQyxTQUFoRDtBQUNBLFFBQU1DLEtBQUtDLEtBQUtDLFNBQUwsQ0FBZVQsS0FBZixDQUFYOztBQUVBLFdBQU8sS0FBS0YsV0FBTCxDQUFpQlksZUFBakIsQ0FBaUNOLEdBQWpDLEVBQXNDRyxFQUF0QyxFQUNKSSxJQURJLENBQ0MsZ0JBQVE7QUFDWixVQUFNQyxTQUFTLG9CQUFVQyxLQUFLQyxFQUFmLEVBQW1CRCxLQUFLRSxJQUF4QixDQUFmO0FBQ0FILGFBQU9JLE9BQVAsR0FBaUIsSUFBSUMsSUFBSixDQUFTSixLQUFLRyxPQUFkLENBQWpCO0FBQ0FKLGFBQU9NLE9BQVAsR0FBaUIsSUFBSUQsSUFBSixDQUFTSixLQUFLSyxPQUFkLENBQWpCO0FBQ0EsYUFBT04sTUFBUDtBQUNELEtBTkksQ0FBUDtBQU9ELEc7O0FBRUQ7Ozs7Ozs7Ozs7NEJBUUFPLFEscUJBQVNDLE8sRUFBUztBQUNoQixRQUFJLE9BQU9BLE9BQVAsS0FBbUIsUUFBdkIsRUFBaUM7QUFDL0IsYUFBT25CLFFBQVFDLE1BQVIsQ0FBZSxJQUFJQyxLQUFKLENBQVUsZ0RBQVYsQ0FBZixDQUFQO0FBQ0Q7QUFDRCxRQUFNQyxNQUFNLEtBQUtOLFdBQUwsQ0FBaUJPLFNBQWpCLENBQTJCQyxNQUEzQixHQUFvQyxVQUFwQyxHQUFpRGMsT0FBN0Q7O0FBRUEsV0FBTyxLQUFLdEIsV0FBTCxDQUFpQnVCLGNBQWpCLENBQWdDakIsR0FBaEMsRUFDSk8sSUFESSxDQUNDLGdCQUFRO0FBQ1osVUFBTVcsZUFBZSxvQkFBVVQsS0FBS0MsRUFBZixFQUFtQkQsS0FBS0UsSUFBeEIsQ0FBckI7QUFDQU8sbUJBQWFOLE9BQWIsR0FBdUIsSUFBSUMsSUFBSixDQUFTSixLQUFLRyxPQUFkLENBQXZCO0FBQ0FNLG1CQUFhSixPQUFiLEdBQXVCLElBQUlELElBQUosQ0FBU0osS0FBS0ssT0FBZCxDQUF2QjtBQUNBLGFBQU9JLFlBQVA7QUFDRCxLQU5JLENBQVA7QUFPRCxHOztBQUVEOzs7Ozs7Ozs0QkFNQUMsUyx3QkFBWTtBQUNWLFFBQU1uQixNQUFNLEtBQUtOLFdBQUwsQ0FBaUJPLFNBQWpCLENBQTJCQyxNQUEzQixHQUFvQyxTQUFoRDs7QUFFQSxXQUFPLEtBQUtSLFdBQUwsQ0FBaUJ1QixjQUFqQixDQUFnQ2pCLEdBQWhDLEVBQ0pPLElBREksQ0FDQyxnQkFBUTtBQUNaLFVBQU1hLFNBQVMsRUFBZjtBQUNBWCxXQUFLWSxPQUFMLENBQWEsaUJBQVM7QUFDcEIsWUFBTXpCLFFBQVEsb0JBQVUwQixNQUFNWixFQUFoQixFQUFvQlksTUFBTVgsSUFBMUIsQ0FBZDtBQUNBZixjQUFNZ0IsT0FBTixHQUFnQixJQUFJQyxJQUFKLENBQVNTLE1BQU1WLE9BQWYsQ0FBaEI7QUFDQWhCLGNBQU1rQixPQUFOLEdBQWdCLElBQUlELElBQUosQ0FBU1MsTUFBTVIsT0FBZixDQUFoQjtBQUNBTSxlQUFPRyxJQUFQLENBQVkzQixLQUFaO0FBQ0QsT0FMRDtBQU1BLGFBQU93QixNQUFQO0FBQ0QsS0FWSSxDQUFQO0FBV0QsRzs7Ozs7a0JBL0VrQjVCLGUiLCJmaWxlIjoiZ3JvdXAtY29udHJvbGxlci5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBHcm91cCBmcm9tICcuL2dyb3VwJztcblxuLyoqXG4gKiBDb250cm9sbGVyIGNsYXNzIGZvciB0aGUgR3JvdXAgbW9kZWwuXG4gKiBAcHJpdmF0ZVxuICovXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBHcm91cENvbnRyb2xsZXIge1xuICAvKipcbiAgICogQHBhcmFtIHtDb25uZWN0aW9ufSBjb25uZWN0aW9uIC0gT2JqZWN0IHRvIHVzZSBmb3IgbWFraW5nIGEgY29ubmVjdGlvbiB0byB0aGUgUkVTVCBBUEkgYW5kIFdlYnNvY2tldCBzZXJ2ZXIuXG4gICAqL1xuICBjb25zdHJ1Y3Rvcihjb25uZWN0aW9uKSB7XG4gICAgLyoqXG4gICAgICogT2JqZWN0IHRvIHVzZSBmb3IgbWFraW5nIGEgY29ubmVjdGlvbiB0byB0aGUgUkVTVCBBUEkgYW5kIFdlYnNvY2tldCBzZXJ2ZXIuXG4gICAgICogQHR5cGUge0Nvbm5lY3Rpb259XG4gICAgICovXG4gICAgdGhpcy5fY29ubmVjdGlvbiA9IGNvbm5lY3Rpb247XG4gIH1cblxuICAvKipcbiAgICogQ3JlYXRlIGEgZ3JvdXAuIFRoZSBncm91cCB3aWxsIGJlIHBhcnQgb2YgdGhlIGN1cnJlbnQgYWN0aXZlIHtAbGluayBPcmdhbmlzYXRpb259IGRlcml2ZWQgZnJvbSB0aGUgT0F1dGgyIHNjb3BlLlxuICAgKlxuICAgKiBAcGFyYW0ge09yZ2FuaXNhdGlvbn0gZ3JvdXAgLSBPYmplY3QgdG8gY3JlYXRlLlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZS48R3JvdXA+fSBQcm9taXNlIGNvbnRhaW5pbmcgdGhlIG5ld2x5IGNyZWF0ZWQgR3JvdXAuXG4gICAqIEB0aHJvd3Mge1Byb21pc2UuPEVycm9yPn0gb3JnYW5pc2F0aW9uIHBhcmFtZXRlciBvZiB0eXBlIFwiR3JvdXBcIiBpcyByZXF1aXJlZC5cbiAgICogQHRocm93cyB7UHJvbWlzZS48RXJyb3I+fSBJZiB0aGUgc2VydmVyIHJldHVybmVkIGFuIGVycm9yLlxuICAgKi9cbiAgY3JlYXRlR3JvdXAoZ3JvdXApIHtcbiAgICBpZiAoIShncm91cCBpbnN0YW5jZW9mIEdyb3VwKSkge1xuICAgICAgcmV0dXJuIFByb21pc2UucmVqZWN0KG5ldyBFcnJvcignZ3JvdXAgcGFyYW1ldGVyIG9mIHR5cGUgXCJHcm91cFwiIGlzIHJlcXVpcmVkJykpO1xuICAgIH1cbiAgICBjb25zdCB1cmwgPSB0aGlzLl9jb25uZWN0aW9uLl9zZXR0aW5ncy5hcGlVcmwgKyAnL2dyb3Vwcyc7XG4gICAgY29uc3QgZmQgPSBKU09OLnN0cmluZ2lmeShncm91cCk7XG5cbiAgICByZXR1cm4gdGhpcy5fY29ubmVjdGlvbi5fc2VjdXJlQWpheFBvc3QodXJsLCBmZClcbiAgICAgIC50aGVuKGRhdGEgPT4ge1xuICAgICAgICBjb25zdCByZXN1bHQgPSBuZXcgR3JvdXAoZGF0YS5pZCwgZGF0YS5uYW1lKTtcbiAgICAgICAgcmVzdWx0LmNyZWF0ZWQgPSBuZXcgRGF0ZShkYXRhLmNyZWF0ZWQpO1xuICAgICAgICByZXN1bHQudXBkYXRlZCA9IG5ldyBEYXRlKGRhdGEudXBkYXRlZCk7XG4gICAgICAgIHJldHVybiByZXN1bHQ7XG4gICAgICB9KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZXQgYSBncm91cCB3aGljaCBpcyBwYXJ0IG9mIHRoZSBjdXJyZW50IGFjdGl2ZSB7QGxpbmsgT3JnYW5pc2F0aW9ufSBkZXJpdmVkIGZyb20gdGhlIE9BdXRoMiBzY29wZS5cbiAgICpcbiAgICogQHBhcmFtIHtzdHJpbmd9IGdyb3VwSWQgLSBTcGVjaWZ5IGEgZ3JvdXAgaWRlbnRpZmllci5cbiAgICogQHJldHVybnMge1Byb21pc2UuPEdyb3VwPn0gUHJvbWlzZSBjb250YWluaW5nIGFuIEdyb3VwLlxuICAgKiBAdGhyb3dzIHtQcm9taXNlLjxFcnJvcj59IGdyb3VwSWQgcGFyYW1ldGVyIG9mIHR5cGUgXCJzdHJpbmdcIiBpcyByZXF1aXJlZC5cbiAgICogQHRocm93cyB7UHJvbWlzZS48RXJyb3I+fSBJZiBubyByZXN1bHQgY291bGQgbm90IGJlIGZvdW5kLlxuICAgKi9cbiAgZ2V0R3JvdXAoZ3JvdXBJZCkge1xuICAgIGlmICh0eXBlb2YgZ3JvdXBJZCAhPT0gJ3N0cmluZycpIHtcbiAgICAgIHJldHVybiBQcm9taXNlLnJlamVjdChuZXcgRXJyb3IoJ2dyb3VwSWQgcGFyYW1ldGVyIG9mIHR5cGUgXCJzdHJpbmdcIiBpcyByZXF1aXJlZCcpKTtcbiAgICB9XG4gICAgY29uc3QgdXJsID0gdGhpcy5fY29ubmVjdGlvbi5fc2V0dGluZ3MuYXBpVXJsICsgJy9ncm91cHMvJyArIGdyb3VwSWQ7XG5cbiAgICByZXR1cm4gdGhpcy5fY29ubmVjdGlvbi5fc2VjdXJlQWpheEdldCh1cmwpXG4gICAgICAudGhlbihkYXRhID0+IHtcbiAgICAgICAgY29uc3Qgb3JnYW5pc2F0aW9uID0gbmV3IEdyb3VwKGRhdGEuaWQsIGRhdGEubmFtZSk7XG4gICAgICAgIG9yZ2FuaXNhdGlvbi5jcmVhdGVkID0gbmV3IERhdGUoZGF0YS5jcmVhdGVkKTtcbiAgICAgICAgb3JnYW5pc2F0aW9uLnVwZGF0ZWQgPSBuZXcgRGF0ZShkYXRhLnVwZGF0ZWQpO1xuICAgICAgICByZXR1cm4gb3JnYW5pc2F0aW9uO1xuICAgICAgfSk7XG4gIH1cblxuICAvKipcbiAgICogR2V0IGFuZCByZXR1cm4gYWxsIGdyb3VwcyBpbiB0aGUgY3VycmVudCBhY3RpdmUge0BsaW5rIE9yZ2FuaXNhdGlvbn0gZGVyaXZlZCBmcm9tIHRoZSBPQXV0aDIgc2NvcGUuXG4gICAqXG4gICAqIEByZXR1cm5zIHtQcm9taXNlLjxHcm91cFtdPn0gUHJvbWlzZSBjb250YWluaW5nIGFuIGFycmF5IG9mIEdyb3Vwcy5cbiAgICogQHRocm93cyB7UHJvbWlzZS48RXJyb3I+fSBJZiBubyByZXN1bHQgY291bGQgbm90IGJlIGZvdW5kLlxuICAgKi9cbiAgZ2V0R3JvdXBzKCkge1xuICAgIGNvbnN0IHVybCA9IHRoaXMuX2Nvbm5lY3Rpb24uX3NldHRpbmdzLmFwaVVybCArICcvZ3JvdXBzJztcblxuICAgIHJldHVybiB0aGlzLl9jb25uZWN0aW9uLl9zZWN1cmVBamF4R2V0KHVybClcbiAgICAgIC50aGVuKGRhdGEgPT4ge1xuICAgICAgICBjb25zdCBncm91cHMgPSBbXTtcbiAgICAgICAgZGF0YS5mb3JFYWNoKGRhdHVtID0+IHtcbiAgICAgICAgICBjb25zdCBncm91cCA9IG5ldyBHcm91cChkYXR1bS5pZCwgZGF0dW0ubmFtZSk7XG4gICAgICAgICAgZ3JvdXAuY3JlYXRlZCA9IG5ldyBEYXRlKGRhdHVtLmNyZWF0ZWQpO1xuICAgICAgICAgIGdyb3VwLnVwZGF0ZWQgPSBuZXcgRGF0ZShkYXR1bS51cGRhdGVkKTtcbiAgICAgICAgICBncm91cHMucHVzaChncm91cCk7XG4gICAgICAgIH0pO1xuICAgICAgICByZXR1cm4gZ3JvdXBzO1xuICAgICAgfSk7XG4gIH1cblxufVxuIl19 | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9hZG1pbmlzdHJhdGl2ZS1zZGsvZ3JvdXAvZ3JvdXAtY29udHJvbGxlci5qcyJdLCJuYW1lcyI6WyJHcm91cENvbnRyb2xsZXIiLCJjb25uZWN0aW9uIiwiX2Nvbm5lY3Rpb24iLCJjcmVhdGVHcm91cCIsImdyb3VwIiwiUHJvbWlzZSIsInJlamVjdCIsIkVycm9yIiwidXJsIiwiX3NldHRpbmdzIiwiYXBpVXJsIiwiZmQiLCJKU09OIiwic3RyaW5naWZ5IiwiX3NlY3VyZUFqYXhQb3N0IiwidGhlbiIsInJlc3VsdCIsImRhdGEiLCJpZCIsIm5hbWUiLCJjcmVhdGVkIiwiRGF0ZSIsInVwZGF0ZWQiLCJnZXRHcm91cCIsImdyb3VwSWQiLCJfc2VjdXJlQWpheEdldCIsIm9yZ2FuaXNhdGlvbiIsImdldEdyb3VwcyIsImdyb3VwcyIsImZvckVhY2giLCJkYXR1bSIsInB1c2giXSwibWFwcGluZ3MiOiI7Ozs7QUFBQTs7Ozs7Ozs7QUFFQTs7OztJQUlxQkEsZTtBQUNuQjs7O0FBR0EsMkJBQVlDLFVBQVosRUFBd0I7QUFBQTs7QUFDdEI7Ozs7QUFJQSxTQUFLQyxXQUFMLEdBQW1CRCxVQUFuQjtBQUNEOztBQUVEOzs7Ozs7Ozs7OzRCQVFBRSxXLHdCQUFZQyxLLEVBQU87QUFDakIsUUFBSSxFQUFFQSxnQ0FBRixDQUFKLEVBQStCO0FBQzdCLGFBQU9DLFFBQVFDLE1BQVIsQ0FBZSxJQUFJQyxLQUFKLENBQVUsNkNBQVYsQ0FBZixDQUFQO0FBQ0Q7QUFDRCxRQUFNQyxNQUFNLEtBQUtOLFdBQUwsQ0FBaUJPLFNBQWpCLENBQTJCQyxNQUEzQixHQUFvQyxTQUFoRDtBQUNBLFFBQU1DLEtBQUtDLEtBQUtDLFNBQUwsQ0FBZVQsS0FBZixDQUFYOztBQUVBLFdBQU8sS0FBS0YsV0FBTCxDQUFpQlksZUFBakIsQ0FBaUNOLEdBQWpDLEVBQXNDRyxFQUF0QyxFQUNKSSxJQURJLENBQ0MsZ0JBQVE7QUFDWixVQUFNQyxTQUFTLG9CQUFVQyxLQUFLQyxFQUFmLEVBQW1CRCxLQUFLRSxJQUF4QixDQUFmO0FBQ0FILGFBQU9JLE9BQVAsR0FBaUIsSUFBSUMsSUFBSixDQUFTSixLQUFLRyxPQUFkLENBQWpCO0FBQ0FKLGFBQU9NLE9BQVAsR0FBaUIsSUFBSUQsSUFBSixDQUFTSixLQUFLSyxPQUFkLENBQWpCO0FBQ0EsYUFBT04sTUFBUDtBQUNELEtBTkksQ0FBUDtBQU9ELEc7O0FBRUQ7Ozs7Ozs7Ozs7NEJBUUFPLFEscUJBQVNDLE8sRUFBUztBQUNoQixRQUFJLE9BQU9BLE9BQVAsS0FBbUIsUUFBdkIsRUFBaUM7QUFDL0IsYUFBT25CLFFBQVFDLE1BQVIsQ0FBZSxJQUFJQyxLQUFKLENBQVUsZ0RBQVYsQ0FBZixDQUFQO0FBQ0Q7QUFDRCxRQUFNQyxNQUFNLEtBQUtOLFdBQUwsQ0FBaUJPLFNBQWpCLENBQTJCQyxNQUEzQixHQUFvQyxVQUFwQyxHQUFpRGMsT0FBN0Q7O0FBRUEsV0FBTyxLQUFLdEIsV0FBTCxDQUFpQnVCLGNBQWpCLENBQWdDakIsR0FBaEMsRUFDSk8sSUFESSxDQUNDLGdCQUFRO0FBQ1osVUFBTVcsZUFBZSxvQkFBVVQsS0FBS0MsRUFBZixFQUFtQkQsS0FBS0UsSUFBeEIsQ0FBckI7QUFDQU8sbUJBQWFOLE9BQWIsR0FBdUIsSUFBSUMsSUFBSixDQUFTSixLQUFLRyxPQUFkLENBQXZCO0FBQ0FNLG1CQUFhSixPQUFiLEdBQXVCLElBQUlELElBQUosQ0FBU0osS0FBS0ssT0FBZCxDQUF2QjtBQUNBLGFBQU9JLFlBQVA7QUFDRCxLQU5JLENBQVA7QUFPRCxHOztBQUVEOzs7Ozs7Ozs0QkFNQUMsUyx3QkFBWTtBQUNWLFFBQU1uQixNQUFNLEtBQUtOLFdBQUwsQ0FBaUJPLFNBQWpCLENBQTJCQyxNQUEzQixHQUFvQyxTQUFoRDs7QUFFQSxXQUFPLEtBQUtSLFdBQUwsQ0FBaUJ1QixjQUFqQixDQUFnQ2pCLEdBQWhDLEVBQ0pPLElBREksQ0FDQyxnQkFBUTtBQUNaLFVBQU1hLFNBQVMsRUFBZjtBQUNBWCxXQUFLWSxPQUFMLENBQWEsaUJBQVM7QUFDcEIsWUFBTXpCLFFBQVEsb0JBQVUwQixNQUFNWixFQUFoQixFQUFvQlksTUFBTVgsSUFBMUIsQ0FBZDtBQUNBZixjQUFNZ0IsT0FBTixHQUFnQixJQUFJQyxJQUFKLENBQVNTLE1BQU1WLE9BQWYsQ0FBaEI7QUFDQWhCLGNBQU1rQixPQUFOLEdBQWdCLElBQUlELElBQUosQ0FBU1MsTUFBTVIsT0FBZixDQUFoQjtBQUNBTSxlQUFPRyxJQUFQLENBQVkzQixLQUFaO0FBQ0QsT0FMRDtBQU1BLGFBQU93QixNQUFQO0FBQ0QsS0FWSSxDQUFQO0FBV0QsRzs7Ozs7a0JBL0VrQjVCLGUiLCJmaWxlIjoiZ3JvdXAtY29udHJvbGxlci5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBHcm91cCBmcm9tICcuL2dyb3VwJztcblxuLyoqXG4gKiBDb250cm9sbGVyIGNsYXNzIGZvciB0aGUgR3JvdXAgbW9kZWwuXG4gKiBAcHJpdmF0ZVxuICovXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBHcm91cENvbnRyb2xsZXIge1xuICAvKipcbiAgICogQHBhcmFtIHtDb25uZWN0aW9ufSBjb25uZWN0aW9uIC0gT2JqZWN0IHRvIHVzZSBmb3IgbWFraW5nIGEgY29ubmVjdGlvbiB0byB0aGUgUkVTVCBBUEkgYW5kIFdlYnNvY2tldCBzZXJ2ZXIuXG4gICAqL1xuICBjb25zdHJ1Y3Rvcihjb25uZWN0aW9uKSB7XG4gICAgLyoqXG4gICAgICogT2JqZWN0IHRvIHVzZSBmb3IgbWFraW5nIGEgY29ubmVjdGlvbiB0byB0aGUgUkVTVCBBUEkgYW5kIFdlYnNvY2tldCBzZXJ2ZXIuXG4gICAgICogQHR5cGUge0Nvbm5lY3Rpb259XG4gICAgICovXG4gICAgdGhpcy5fY29ubmVjdGlvbiA9IGNvbm5lY3Rpb247XG4gIH1cblxuICAvKipcbiAgICogQ3JlYXRlIGEgZ3JvdXAuIFRoZSBncm91cCB3aWxsIGJlIHBhcnQgb2YgdGhlIGN1cnJlbnQgYWN0aXZlIHtAbGluayBPcmdhbmlzYXRpb259IGRlcml2ZWQgZnJvbSB0aGUgT0F1dGgyIHNjb3BlLlxuICAgKlxuICAgKiBAcGFyYW0ge09yZ2FuaXNhdGlvbn0gZ3JvdXAgLSBPYmplY3QgdG8gY3JlYXRlLlxuICAgKiBAcmV0dXJucyB7UHJvbWlzZS48R3JvdXA+fSBQcm9taXNlIGNvbnRhaW5pbmcgdGhlIG5ld2x5IGNyZWF0ZWQgR3JvdXAuXG4gICAqIEB0aHJvd3Mge1Byb21pc2UuPEVycm9yPn0gb3JnYW5pc2F0aW9uIHBhcmFtZXRlciBvZiB0eXBlIFwiR3JvdXBcIiBpcyByZXF1aXJlZC5cbiAgICogQHRocm93cyB7UHJvbWlzZS48RXJyb3I+fSBJZiB0aGUgc2VydmVyIHJldHVybmVkIGFuIGVycm9yLlxuICAgKi9cbiAgY3JlYXRlR3JvdXAoZ3JvdXApIHtcbiAgICBpZiAoIShncm91cCBpbnN0YW5jZW9mIEdyb3VwKSkge1xuICAgICAgcmV0dXJuIFByb21pc2UucmVqZWN0KG5ldyBFcnJvcignZ3JvdXAgcGFyYW1ldGVyIG9mIHR5cGUgXCJHcm91cFwiIGlzIHJlcXVpcmVkJykpO1xuICAgIH1cbiAgICBjb25zdCB1cmwgPSB0aGlzLl9jb25uZWN0aW9uLl9zZXR0aW5ncy5hcGlVcmwgKyAnL2dyb3Vwcyc7XG4gICAgY29uc3QgZmQgPSBKU09OLnN0cmluZ2lmeShncm91cCk7XG5cbiAgICByZXR1cm4gdGhpcy5fY29ubmVjdGlvbi5fc2VjdXJlQWpheFBvc3QodXJsLCBmZClcbiAgICAgIC50aGVuKGRhdGEgPT4ge1xuICAgICAgICBjb25zdCByZXN1bHQgPSBuZXcgR3JvdXAoZGF0YS5pZCwgZGF0YS5uYW1lKTtcbiAgICAgICAgcmVzdWx0LmNyZWF0ZWQgPSBuZXcgRGF0ZShkYXRhLmNyZWF0ZWQpO1xuICAgICAgICByZXN1bHQudXBkYXRlZCA9IG5ldyBEYXRlKGRhdGEudXBkYXRlZCk7XG4gICAgICAgIHJldHVybiByZXN1bHQ7XG4gICAgICB9KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZXQgYSBncm91cCB3aGljaCBpcyBwYXJ0IG9mIHRoZSBjdXJyZW50IGFjdGl2ZSB7QGxpbmsgT3JnYW5pc2F0aW9ufSBkZXJpdmVkIGZyb20gdGhlIE9BdXRoMiBzY29wZS5cbiAgICpcbiAgICogQHBhcmFtIHtzdHJpbmd9IGdyb3VwSWQgLSBTcGVjaWZ5IGEgZ3JvdXAgaWRlbnRpZmllci5cbiAgICogQHJldHVybnMge1Byb21pc2UuPEdyb3VwPn0gUHJvbWlzZSBjb250YWluaW5nIGFuIEdyb3VwLlxuICAgKiBAdGhyb3dzIHtQcm9taXNlLjxFcnJvcj59IGdyb3VwSWQgcGFyYW1ldGVyIG9mIHR5cGUgXCJzdHJpbmdcIiBpcyByZXF1aXJlZC5cbiAgICogQHRocm93cyB7UHJvbWlzZS48RXJyb3I+fSBJZiBubyByZXN1bHQgY291bGQgbm90IGJlIGZvdW5kLlxuICAgKi9cbiAgZ2V0R3JvdXAoZ3JvdXBJZCkge1xuICAgIGlmICh0eXBlb2YgZ3JvdXBJZCAhPT0gJ3N0cmluZycpIHtcbiAgICAgIHJldHVybiBQcm9taXNlLnJlamVjdChuZXcgRXJyb3IoJ2dyb3VwSWQgcGFyYW1ldGVyIG9mIHR5cGUgXCJzdHJpbmdcIiBpcyByZXF1aXJlZCcpKTtcbiAgICB9XG4gICAgY29uc3QgdXJsID0gdGhpcy5fY29ubmVjdGlvbi5fc2V0dGluZ3MuYXBpVXJsICsgJy9ncm91cHMvJyArIGdyb3VwSWQ7XG5cbiAgICByZXR1cm4gdGhpcy5fY29ubmVjdGlvbi5fc2VjdXJlQWpheEdldCh1cmwpXG4gICAgICAudGhlbihkYXRhID0+IHtcbiAgICAgICAgY29uc3Qgb3JnYW5pc2F0aW9uID0gbmV3IEdyb3VwKGRhdGEuaWQsIGRhdGEubmFtZSk7XG4gICAgICAgIG9yZ2FuaXNhdGlvbi5jcmVhdGVkID0gbmV3IERhdGUoZGF0YS5jcmVhdGVkKTtcbiAgICAgICAgb3JnYW5pc2F0aW9uLnVwZGF0ZWQgPSBuZXcgRGF0ZShkYXRhLnVwZGF0ZWQpO1xuICAgICAgICByZXR1cm4gb3JnYW5pc2F0aW9uO1xuICAgICAgfSk7XG4gIH1cblxuICAvKipcbiAgICogR2V0IGFuZCByZXR1cm4gYWxsIGdyb3VwcyBpbiB0aGUgY3VycmVudCBhY3RpdmUge0BsaW5rIE9yZ2FuaXNhdGlvbn0gZGVyaXZlZCBmcm9tIHRoZSBPQXV0aDIgc2NvcGUuXG4gICAqXG4gICAqIEByZXR1cm5zIHtQcm9taXNlLjxHcm91cFtdPn0gUHJvbWlzZSBjb250YWluaW5nIGFuIGFycmF5IG9mIEdyb3Vwcy5cbiAgICogQHRocm93cyB7UHJvbWlzZS48RXJyb3I+fSBJZiBubyByZXN1bHQgY291bGQgbm90IGJlIGZvdW5kLlxuICAgKi9cbiAgZ2V0R3JvdXBzKCkge1xuICAgIGNvbnN0IHVybCA9IHRoaXMuX2Nvbm5lY3Rpb24uX3NldHRpbmdzLmFwaVVybCArICcvZ3JvdXBzJztcblxuICAgIHJldHVybiB0aGlzLl9jb25uZWN0aW9uLl9zZWN1cmVBamF4R2V0KHVybClcbiAgICAgIC50aGVuKGRhdGEgPT4ge1xuICAgICAgICBjb25zdCBncm91cHMgPSBbXTtcbiAgICAgICAgZGF0YS5mb3JFYWNoKGRhdHVtID0+IHtcbiAgICAgICAgICBjb25zdCBncm91cCA9IG5ldyBHcm91cChkYXR1bS5pZCwgZGF0dW0ubmFtZSk7XG4gICAgICAgICAgZ3JvdXAuY3JlYXRlZCA9IG5ldyBEYXRlKGRhdHVtLmNyZWF0ZWQpO1xuICAgICAgICAgIGdyb3VwLnVwZGF0ZWQgPSBuZXcgRGF0ZShkYXR1bS51cGRhdGVkKTtcbiAgICAgICAgICBncm91cHMucHVzaChncm91cCk7XG4gICAgICAgIH0pO1xuICAgICAgICByZXR1cm4gZ3JvdXBzO1xuICAgICAgfSk7XG4gIH1cbn1cbiJdfQ== |
@@ -251,2 +251,2 @@ 'use strict'; | ||
exports.default = SpeechRecordingController; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/administrative-sdk/speech-recording/speech-recording-controller.js"],"names":["SpeechRecordingController","connection","_connection","speechRecordingInitChallenge","challenge","call","_recordingId","id","then","console","log","recordingId","speechRecordingInitAudio","recorder","dataavailableCb","specs","getAudioSpecs","audioFormat","audioParameters","addEventListener","startStreamingSpeechRecording","Promise","reject","Error","_session","isRecording","self","resolve","notify","_cb","data","recording","userId","Date","created","updated","addAccessToken","audioUrl","recordedCb","activeRecordingId","audioBlob","forcedStop","res","logRPCError","removeEventListener","startStreaming","chunk","encoded","_arrayBufferToBase64","startRecording","p","hasUserMediaApproval","resolve_","catch","error","getSpeechRecording","challengeId","url","_settings","apiUrl","_secureAjaxGet","getSpeechRecordings","map","datum"],"mappings":"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","file":"speech-recording-controller.js","sourcesContent":["import Base64Utils from '../utils/base64-utils';\nimport Connection from '../connection/connection-controller';\nimport SpeechRecording from './speech-recording';\nimport when from 'when';\n\n/**\n * Controller class for the SpeechRecording model.\n * @private\n */\nexport default class SpeechRecordingController {\n  /**\n   * @param {Connection} connection - Object to use for making a connection to the REST API and Websocket server.\n   */\n  constructor(connection) {\n    this._connection = connection;\n  }\n\n  /**\n   * Initialise the speech recording challenge through RPCs.\n   *\n   * @param {SpeechChallenge} challenge - SpeechChallenge.\n   * @private\n   */\n  speechRecordingInitChallenge(challenge) {\n    return this._connection.call('recording.init_challenge',\n      [this._connection._recordingId, challenge.id]).then(\n      // RPC success callback\n      recordingId => {\n        console.log('Challenge initialised for recordingId: ' + this._connection._recordingId);\n        return recordingId;\n      });\n  }\n\n  /**\n   * Initialise the speech recording audio specs through RPCs.\n   *\n   * @param {AudioRecorder} recorder - AudioRecorder.\n   * @param {Function} dataavailableCb - Callback.\n   * @private\n   */\n  speechRecordingInitAudio(recorder, dataavailableCb) {\n    // Indicate to the socket server that we're about to start recording a\n    // challenge. This allows the socket server some time to fetch the metadata\n    // and reference audio to start the recording when audio is actually submitted.\n    const specs = recorder.getAudioSpecs();\n    return this._connection.call('recording.init_audio',\n      [this._connection._recordingId, specs.audioFormat], specs.audioParameters)\n      .then(recordingId => {\n        console.log('Accepted audio parameters for recordingId after init_audio: ' + this._connection._recordingId);\n        // Start listening for streaming data.\n        recorder.addEventListener('dataavailable', dataavailableCb);\n        return recordingId;\n      });\n  }\n\n  /**\n   * Start a speech recording from streaming audio.\n   *\n   * @param {SpeechChallenge} challenge - The speech challenge to perform.\n   * @param {AudioRecorder} recorder - The audio recorder to extract audio from.\n   * @returns {Promise.<SpeechRecording>} A {@link https://github.com/cujojs/when} Promise containing a {@link SpeechRecording}.\n   * @emits {string} 'ReadyToReceive' when the call is made to receive audio. The recorder can now send audio.\n   * @throws {Promise.<Error>} If challenge is not an object or not defined.\n   * @throws {Promise.<Error>} If challenge has no id.\n   * @throws {Promise.<Error>} If the connection is not open.\n   * @throws {Promise.<Error>} If the recorder is already recording.\n   * @throws {Promise.<Error>} If a session is already in progress.\n   * @throws {Promise.<Error>} If something went wrong during recording.\n   */\n  startStreamingSpeechRecording(challenge, recorder) {\n    // Validate required domain model.\n    // Validate environment prerequisites.\n    if (typeof challenge !== 'object' || !challenge) {\n      return Promise.reject(new Error('\"challenge\" parameter is required or invalid'));\n    }\n    if (!challenge.id) {\n      return Promise.reject(new Error('challenge.id field is required'));\n    }\n    if (!this._connection._session) {\n      return Promise.reject(new Error('WebSocket connection was not open.'));\n    }\n    if (recorder.isRecording()) {\n      return Promise.reject(new Error('Recorder should not yet be recording.'));\n    }\n    if (this._connection._recordingId !== null) {\n      return Promise.reject(new Error('Session with recordingId ' + this._connection._recordingId +\n        ' still in progress.'));\n    }\n    const self = this;\n    return new when.Promise((resolve, reject, notify) => {\n      self._connection._recordingId = null;\n\n      function _cb(data) {\n        const recording = new SpeechRecording(\n          challenge.id, data.userId, data.id, new Date(data.created), new Date(data.updated),\n          self._connection.addAccessToken(data.audioUrl));\n        resolve({recordingId: self._connection._recordingId, recording});\n      }\n\n      function recordedCb(activeRecordingId, audioBlob, forcedStop) {\n        self._connection.call('recording.close',\n          [self._connection._recordingId]).then(\n          // RPC success callback\n          res => {\n            // Pass along details to the success callback\n            _cb(res, forcedStop);\n          },\n          // RPC error callback\n          res => {\n            Connection.logRPCError(res);\n            reject(res);\n          });\n        recorder.removeEventListener('recorded', recordedCb);\n        recorder.removeEventListener('dataavailable', startStreaming);\n      }\n\n      // Start streaming the binary audio when the user instructs\n      // the audio recorder to start recording.\n      function startStreaming(chunk) {\n        const encoded = Base64Utils._arrayBufferToBase64(chunk);\n        console.log('Sending audio chunk to websocket for recordingId: ' +\n          self._connection._recordingId);\n        self._connection.call('recording.write',\n          [self._connection._recordingId, encoded, 'base64']).then(\n          // RPC success callback\n          res => {\n            // Wrote data.\n            console.log('Wrote data');\n            return res;\n          },\n          // RPC error callback\n          res => {\n            Connection.logRPCError(res);\n            reject(res);\n          }\n        );\n      }\n\n      function startRecording(recordingId) {\n        self._connection._recordingId = recordingId;\n        console.log('Got recordingId after initialisation: ' + self._connection._recordingId);\n      }\n\n      recorder.addEventListener('recorded', recordedCb);\n      self._connection.call('recording.init_recording', [])\n        .then(startRecording)\n        .then(() =>\n          self.speechRecordingInitChallenge(challenge)\n            .then(() => {\n              const p = new Promise(resolve_ => {\n                if (recorder.hasUserMediaApproval()) {\n                  resolve_();\n                } else {\n                  recorder.addEventListener('ready', resolve_);\n                }\n              });\n              p.then(() => {\n                self.speechRecordingInitAudio(recorder, startStreaming)\n                  .catch(reject);\n              });\n            })\n            .then(() => notify('ReadyToReceive'))\n        )\n        .catch(reject);\n    })\n      .then(res => {\n        self._connection._recordingId = null;\n        return Promise.resolve(res);\n      })\n      .catch(error => {\n        self._connection._recordingId = null;\n        Connection.logRPCError(error);\n        return Promise.reject(error);\n      });\n  }\n\n  /**\n   * Get a speech recording in a speech challenge from the current active {@link Organisation} derived from the OAuth2\n   * scope.\n   *\n   * @param {string} challengeId - Specify a speech challenge identifier.\n   * @param {string} recordingId - Specify a speech recording identifier.\n   * @returns {Promise.<SpeechRecording>} Promise containing a SpeechRecording.\n   * @throws {Promise.<Error>} {@link SpeechChallenge#id} field is required.\n   * @throws {Promise.<Error>} {@link SpeechRecording#id} field is required.\n   * @throws {Promise.<Error>} If no result could not be found.\n   */\n  getSpeechRecording(challengeId, recordingId) {\n    if (!challengeId) {\n      return Promise.reject(new Error('challengeId field is required'));\n    }\n    if (!recordingId) {\n      return Promise.reject(new Error('recordingId field is required'));\n    }\n    const url = this._connection._settings.apiUrl + '/challenges/speech/' + challengeId + '/recordings/' + recordingId;\n    return this._connection._secureAjaxGet(url)\n      .then(data => new SpeechRecording(challengeId, data.userId, data.id, new Date(data.created),\n          new Date(data.updated), this._connection.addAccessToken(data.audioUrl)));\n  }\n\n  /**\n   * Get and return all speech recordings in a specific speech challenge from the current active {@link Organisation}\n   * derived from the OAuth2 scope.\n   *\n   * @param {string} challengeId - Specify a speech challenge identifier to list speech recordings for.\n   * @returns {Promise.<SpeechRecording[]>} Promise containing an array of SpeechRecording.\n   * @throws {Promise.<Error>} {@link SpeechChallenge#id} is required.\n   * @throws {Promise.<Error>} If no result could not be found.\n   */\n  getSpeechRecordings(challengeId) {\n    if (!challengeId) {\n      return Promise.reject(new Error('challengeId field is required'));\n    }\n    const url = this._connection._settings.apiUrl + '/challenges/speech/' + challengeId + '/recordings';\n    return this._connection._secureAjaxGet(url)\n      .then(data => data.map(datum => new SpeechRecording(challengeId, datum.userId, datum.id, new Date(datum.created),\n          new Date(datum.updated), this._connection.addAccessToken(datum.audioUrl))));\n  }\n}\n"]} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/administrative-sdk/speech-recording/speech-recording-controller.js"],"names":["SpeechRecordingController","connection","_connection","speechRecordingInitChallenge","challenge","call","_recordingId","id","then","console","log","recordingId","speechRecordingInitAudio","recorder","dataavailableCb","specs","getAudioSpecs","audioFormat","audioParameters","addEventListener","startStreamingSpeechRecording","Promise","reject","Error","_session","isRecording","self","resolve","notify","_cb","data","recording","userId","Date","created","updated","addAccessToken","audioUrl","recordedCb","activeRecordingId","audioBlob","forcedStop","res","logRPCError","removeEventListener","startStreaming","chunk","encoded","_arrayBufferToBase64","startRecording","p","hasUserMediaApproval","resolve_","catch","error","getSpeechRecording","challengeId","url","_settings","apiUrl","_secureAjaxGet","getSpeechRecordings","map","datum"],"mappings":"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","file":"speech-recording-controller.js","sourcesContent":["import Base64Utils from '../utils/base64-utils';\nimport Connection from '../connection/connection-controller';\nimport SpeechRecording from './speech-recording';\nimport when from 'when';\n\n/**\n * Controller class for the SpeechRecording model.\n * @private\n */\nexport default class SpeechRecordingController {\n  /**\n   * @param {Connection} connection - Object to use for making a connection to the REST API and Websocket server.\n   */\n  constructor(connection) {\n    this._connection = connection;\n  }\n\n  /**\n   * Initialise the speech recording challenge through RPCs.\n   *\n   * @param {SpeechChallenge} challenge - SpeechChallenge.\n   * @private\n   */\n  speechRecordingInitChallenge(challenge) {\n    return this._connection.call('recording.init_challenge',\n      [this._connection._recordingId, challenge.id]).then(\n      // RPC success callback\n      recordingId => {\n        console.log('Challenge initialised for recordingId: ' + this._connection._recordingId);\n        return recordingId;\n      });\n  }\n\n  /**\n   * Initialise the speech recording audio specs through RPCs.\n   *\n   * @param {AudioRecorder} recorder - AudioRecorder.\n   * @param {Function} dataavailableCb - Callback.\n   * @private\n   */\n  speechRecordingInitAudio(recorder, dataavailableCb) {\n    // Indicate to the socket server that we're about to start recording a\n    // challenge. This allows the socket server some time to fetch the metadata\n    // and reference audio to start the recording when audio is actually submitted.\n    const specs = recorder.getAudioSpecs();\n    return this._connection.call('recording.init_audio',\n      [this._connection._recordingId, specs.audioFormat], specs.audioParameters)\n      .then(recordingId => {\n        console.log('Accepted audio parameters for recordingId after init_audio: ' + this._connection._recordingId);\n        // Start listening for streaming data.\n        recorder.addEventListener('dataavailable', dataavailableCb);\n        return recordingId;\n      });\n  }\n\n  /**\n   * Start a speech recording from streaming audio.\n   *\n   * @param {SpeechChallenge} challenge - The speech challenge to perform.\n   * @param {AudioRecorder} recorder - The audio recorder to extract audio from.\n   * @returns {Promise.<SpeechRecording>} A {@link https://github.com/cujojs/when} Promise containing a {@link SpeechRecording}.\n   * @emits {string} 'ReadyToReceive' when the call is made to receive audio. The recorder can now send audio.\n   * @throws {Promise.<Error>} If challenge is not an object or not defined.\n   * @throws {Promise.<Error>} If challenge has no id.\n   * @throws {Promise.<Error>} If the connection is not open.\n   * @throws {Promise.<Error>} If the recorder is already recording.\n   * @throws {Promise.<Error>} If a session is already in progress.\n   * @throws {Promise.<Error>} If something went wrong during recording.\n   */\n  startStreamingSpeechRecording(challenge, recorder) {\n    // Validate required domain model.\n    // Validate environment prerequisites.\n    if (typeof challenge !== 'object' || !challenge) {\n      return Promise.reject(new Error('\"challenge\" parameter is required or invalid'));\n    }\n    if (!challenge.id) {\n      return Promise.reject(new Error('challenge.id field is required'));\n    }\n    if (!this._connection._session) {\n      return Promise.reject(new Error('WebSocket connection was not open.'));\n    }\n    if (recorder.isRecording()) {\n      return Promise.reject(new Error('Recorder should not yet be recording.'));\n    }\n    if (this._connection._recordingId !== null) {\n      return Promise.reject(new Error('Session with recordingId ' + this._connection._recordingId +\n        ' still in progress.'));\n    }\n    const self = this;\n    return new when.Promise((resolve, reject, notify) => {\n      self._connection._recordingId = null;\n\n      function _cb(data) {\n        const recording = new SpeechRecording(\n          challenge.id, data.userId, data.id, new Date(data.created), new Date(data.updated),\n          self._connection.addAccessToken(data.audioUrl));\n        resolve({recordingId: self._connection._recordingId, recording});\n      }\n\n      function recordedCb(activeRecordingId, audioBlob, forcedStop) {\n        self._connection.call('recording.close',\n          [self._connection._recordingId]).then(\n          // RPC success callback\n          res => {\n            // Pass along details to the success callback\n            _cb(res, forcedStop);\n          },\n          // RPC error callback\n          res => {\n            Connection.logRPCError(res);\n            reject(res);\n          });\n        recorder.removeEventListener('recorded', recordedCb);\n        recorder.removeEventListener('dataavailable', startStreaming);\n      }\n\n      // Start streaming the binary audio when the user instructs\n      // the audio recorder to start recording.\n      function startStreaming(chunk) {\n        const encoded = Base64Utils._arrayBufferToBase64(chunk);\n        console.log('Sending audio chunk to websocket for recordingId: ' +\n          self._connection._recordingId);\n        self._connection.call('recording.write',\n          [self._connection._recordingId, encoded, 'base64']).then(\n          // RPC success callback\n          res => {\n            // Wrote data.\n            console.log('Wrote data');\n            return res;\n          },\n          // RPC error callback\n          res => {\n            Connection.logRPCError(res);\n            reject(res);\n          }\n        );\n      }\n\n      function startRecording(recordingId) {\n        self._connection._recordingId = recordingId;\n        console.log('Got recordingId after initialisation: ' + self._connection._recordingId);\n      }\n\n      recorder.addEventListener('recorded', recordedCb);\n      self._connection.call('recording.init_recording', [])\n        .then(startRecording)\n        .then(() =>\n          self.speechRecordingInitChallenge(challenge)\n            .then(() => {\n              const p = new Promise(resolve_ => {\n                if (recorder.hasUserMediaApproval()) {\n                  resolve_();\n                } else {\n                  recorder.addEventListener('ready', resolve_);\n                }\n              });\n              p.then(() => {\n                self.speechRecordingInitAudio(recorder, startStreaming)\n                  .catch(reject);\n              });\n            })\n            .then(() => notify('ReadyToReceive'))\n        )\n        .catch(reject);\n    })\n      .then(res => {\n        self._connection._recordingId = null;\n        return Promise.resolve(res);\n      })\n      .catch(error => {\n        self._connection._recordingId = null;\n        Connection.logRPCError(error);\n        return Promise.reject(error);\n      });\n  }\n\n  /**\n   * Get a speech recording in a speech challenge from the current active {@link Organisation} derived from the OAuth2\n   * scope.\n   *\n   * @param {string} challengeId - Specify a speech challenge identifier.\n   * @param {string} recordingId - Specify a speech recording identifier.\n   * @returns {Promise.<SpeechRecording>} Promise containing a SpeechRecording.\n   * @throws {Promise.<Error>} {@link SpeechChallenge#id} field is required.\n   * @throws {Promise.<Error>} {@link SpeechRecording#id} field is required.\n   * @throws {Promise.<Error>} If no result could not be found.\n   */\n  getSpeechRecording(challengeId, recordingId) {\n    if (!challengeId) {\n      return Promise.reject(new Error('challengeId field is required'));\n    }\n    if (!recordingId) {\n      return Promise.reject(new Error('recordingId field is required'));\n    }\n    const url = this._connection._settings.apiUrl + '/challenges/speech/' + challengeId + '/recordings/' + recordingId;\n    return this._connection._secureAjaxGet(url)\n      .then(data => new SpeechRecording(challengeId, data.userId, data.id, new Date(data.created),\n        new Date(data.updated), this._connection.addAccessToken(data.audioUrl)));\n  }\n\n  /**\n   * Get and return all speech recordings in a specific speech challenge from the current active {@link Organisation}\n   * derived from the OAuth2 scope.\n   *\n   * @param {string} challengeId - Specify a speech challenge identifier to list speech recordings for.\n   * @returns {Promise.<SpeechRecording[]>} Promise containing an array of SpeechRecording.\n   * @throws {Promise.<Error>} {@link SpeechChallenge#id} is required.\n   * @throws {Promise.<Error>} If no result could not be found.\n   */\n  getSpeechRecordings(challengeId) {\n    if (!challengeId) {\n      return Promise.reject(new Error('challengeId field is required'));\n    }\n    const url = this._connection._settings.apiUrl + '/challenges/speech/' + challengeId + '/recordings';\n    return this._connection._secureAjaxGet(url)\n      .then(data => data.map(datum => new SpeechRecording(challengeId, datum.userId, datum.id, new Date(datum.created),\n        new Date(datum.updated), this._connection.addAccessToken(datum.audioUrl))));\n  }\n}\n"]} |
@@ -35,2 +35,2 @@ 'use strict'; | ||
exports.default = Base64Utils; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9hZG1pbmlzdHJhdGl2ZS1zZGsvdXRpbHMvYmFzZTY0LXV0aWxzLmpzIl0sIm5hbWVzIjpbIkJhc2U2NFV0aWxzIiwiX2FycmF5QnVmZmVyVG9CYXNlNjQiLCJidWZmZXIiLCJiaW5hcnkiLCJieXRlcyIsIlVpbnQ4QXJyYXkiLCJsZW4iLCJieXRlTGVuZ3RoIiwiaSIsIlN0cmluZyIsImZyb21DaGFyQ29kZSIsIndpbmRvdyIsImJ0b2EiXSwibWFwcGluZ3MiOiI7Ozs7OztBQUFBOzs7SUFHcUJBLFc7Ozs7O0FBRW5COzs7Ozs7Y0FNT0Msb0IsaUNBQXFCQyxNLEVBQVE7QUFDbEMsUUFBSUMsU0FBUyxFQUFiO0FBQ0EsUUFBTUMsUUFBUSxJQUFJQyxVQUFKLENBQWVILE1BQWYsQ0FBZDtBQUNBLFFBQU1JLE1BQU1GLE1BQU1HLFVBQWxCO0FBQ0EsU0FBSyxJQUFJQyxJQUFJLENBQWIsRUFBZ0JBLElBQUlGLEdBQXBCLEVBQXlCRSxHQUF6QixFQUE4QjtBQUM1QkwsZ0JBQVVNLE9BQU9DLFlBQVAsQ0FBb0JOLE1BQU1JLENBQU4sQ0FBcEIsQ0FBVjtBQUNEO0FBQ0QsV0FBT0csT0FBT0MsSUFBUCxDQUFZVCxNQUFaLENBQVA7QUFDRCxHOzs7OztrQkFoQmtCSCxXIiwiZmlsZSI6ImJhc2U2NC11dGlscy5qcyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQHByaXZhdGVcbiAqL1xuZXhwb3J0IGRlZmF1bHQgY2xhc3MgQmFzZTY0VXRpbHMge1xuXG4gIC8qKlxuICAgKiBDb252ZXJ0IGFuIGFycmF5IGJ1ZmZlciB0byBhIGJhc2U2NCBlbmNvZGVkIGJpbmFyeSBzdHJpbmcuXG4gICAqXG4gICAqIEBwYXJhbSB7YnVmZmVyfSBidWZmZXIgLSBCdWZmZXIgb2YgZGF0YS5cbiAgICogQHJldHVybnMge3N0cmluZ30gQmFzZTY0IGVuY29kZWQgYmluYXJ5IHN0cmluZy5cbiAgICovXG4gIHN0YXRpYyBfYXJyYXlCdWZmZXJUb0Jhc2U2NChidWZmZXIpIHtcbiAgICBsZXQgYmluYXJ5ID0gJyc7XG4gICAgY29uc3QgYnl0ZXMgPSBuZXcgVWludDhBcnJheShidWZmZXIpO1xuICAgIGNvbnN0IGxlbiA9IGJ5dGVzLmJ5dGVMZW5ndGg7XG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCBsZW47IGkrKykge1xuICAgICAgYmluYXJ5ICs9IFN0cmluZy5mcm9tQ2hhckNvZGUoYnl0ZXNbaV0pO1xuICAgIH1cbiAgICByZXR1cm4gd2luZG93LmJ0b2EoYmluYXJ5KTtcbiAgfVxufVxuIl19 | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9hZG1pbmlzdHJhdGl2ZS1zZGsvdXRpbHMvYmFzZTY0LXV0aWxzLmpzIl0sIm5hbWVzIjpbIkJhc2U2NFV0aWxzIiwiX2FycmF5QnVmZmVyVG9CYXNlNjQiLCJidWZmZXIiLCJiaW5hcnkiLCJieXRlcyIsIlVpbnQ4QXJyYXkiLCJsZW4iLCJieXRlTGVuZ3RoIiwiaSIsIlN0cmluZyIsImZyb21DaGFyQ29kZSIsIndpbmRvdyIsImJ0b2EiXSwibWFwcGluZ3MiOiI7Ozs7OztBQUFBOzs7SUFHcUJBLFc7Ozs7O0FBQ25COzs7Ozs7Y0FNT0Msb0IsaUNBQXFCQyxNLEVBQVE7QUFDbEMsUUFBSUMsU0FBUyxFQUFiO0FBQ0EsUUFBTUMsUUFBUSxJQUFJQyxVQUFKLENBQWVILE1BQWYsQ0FBZDtBQUNBLFFBQU1JLE1BQU1GLE1BQU1HLFVBQWxCO0FBQ0EsU0FBSyxJQUFJQyxJQUFJLENBQWIsRUFBZ0JBLElBQUlGLEdBQXBCLEVBQXlCRSxHQUF6QixFQUE4QjtBQUM1QkwsZ0JBQVVNLE9BQU9DLFlBQVAsQ0FBb0JOLE1BQU1JLENBQU4sQ0FBcEIsQ0FBVjtBQUNEO0FBQ0QsV0FBT0csT0FBT0MsSUFBUCxDQUFZVCxNQUFaLENBQVA7QUFDRCxHOzs7OztrQkFma0JILFciLCJmaWxlIjoiYmFzZTY0LXV0aWxzLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBAcHJpdmF0ZVxuICovXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBCYXNlNjRVdGlscyB7XG4gIC8qKlxuICAgKiBDb252ZXJ0IGFuIGFycmF5IGJ1ZmZlciB0byBhIGJhc2U2NCBlbmNvZGVkIGJpbmFyeSBzdHJpbmcuXG4gICAqXG4gICAqIEBwYXJhbSB7YnVmZmVyfSBidWZmZXIgLSBCdWZmZXIgb2YgZGF0YS5cbiAgICogQHJldHVybnMge3N0cmluZ30gQmFzZTY0IGVuY29kZWQgYmluYXJ5IHN0cmluZy5cbiAgICovXG4gIHN0YXRpYyBfYXJyYXlCdWZmZXJUb0Jhc2U2NChidWZmZXIpIHtcbiAgICBsZXQgYmluYXJ5ID0gJyc7XG4gICAgY29uc3QgYnl0ZXMgPSBuZXcgVWludDhBcnJheShidWZmZXIpO1xuICAgIGNvbnN0IGxlbiA9IGJ5dGVzLmJ5dGVMZW5ndGg7XG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCBsZW47IGkrKykge1xuICAgICAgYmluYXJ5ICs9IFN0cmluZy5mcm9tQ2hhckNvZGUoYnl0ZXNbaV0pO1xuICAgIH1cbiAgICByZXR1cm4gd2luZG93LmJ0b2EoYmluYXJ5KTtcbiAgfVxufVxuIl19 |
@@ -27,2 +27,2 @@ "use strict"; | ||
exports.default = Word; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9hZG1pbmlzdHJhdGl2ZS1zZGsvd29yZC93b3JkLmpzIl0sIm5hbWVzIjpbIldvcmQiLCJjaHVua3MiXSwibWFwcGluZ3MiOiI7Ozs7OztBQUFBOzs7SUFHdUJBLEk7QUFDckI7Ozs7O0FBS0UsY0FBWUMsTUFBWixFQUFvQjtBQUFBOztBQUNwQjs7OztBQUlFLE9BQUtBLE1BQUwsR0FBY0EsTUFBZDtBQUNELEM7O2tCQVprQkQsSSIsImZpbGUiOiJ3b3JkLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBAY2xhc3MgV29yZFxuICovXG4gIGV4cG9ydCBkZWZhdWx0IGNsYXNzIFdvcmQge1xuICAvKipcbiAgICogQ3JlYXRlIGEgd29yZCBkb21haW4gbW9kZWwuXG4gICAqXG4gICAqIEBwYXJhbSB7V29yZENodW5rW119IGNodW5rcyAtIFRoZSBzcG9rZW4gc2VudGVuY2UsIHNwbGl0IGluIGdyYXBoZW1lcyBwZXIgd29yZC5cbiAgICovXG4gICAgY29uc3RydWN0b3IoY2h1bmtzKSB7XG4gICAgLyoqXG4gICAgICogVGhlIHNwb2tlbiBzZW50ZW5jZSwgc3BsaXQgaW4gZ3JhcGhlbWVzIHBlciB3b3JkLlxuICAgICAqIEB0eXBlIHtXb3JkQ2h1bmtbXX1cbiAgICAgKi9cbiAgICAgIHRoaXMuY2h1bmtzID0gY2h1bmtzO1xuICAgIH1cbn1cbiJdfQ== | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9hZG1pbmlzdHJhdGl2ZS1zZGsvd29yZC93b3JkLmpzIl0sIm5hbWVzIjpbIldvcmQiLCJjaHVua3MiXSwibWFwcGluZ3MiOiI7Ozs7OztBQUFBOzs7SUFHcUJBLEk7QUFDbkI7Ozs7O0FBS0EsY0FBWUMsTUFBWixFQUFvQjtBQUFBOztBQUNsQjs7OztBQUlBLE9BQUtBLE1BQUwsR0FBY0EsTUFBZDtBQUNELEM7O2tCQVprQkQsSSIsImZpbGUiOiJ3b3JkLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBAY2xhc3MgV29yZFxuICovXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBXb3JkIHtcbiAgLyoqXG4gICAqIENyZWF0ZSBhIHdvcmQgZG9tYWluIG1vZGVsLlxuICAgKlxuICAgKiBAcGFyYW0ge1dvcmRDaHVua1tdfSBjaHVua3MgLSBUaGUgc3Bva2VuIHNlbnRlbmNlLCBzcGxpdCBpbiBncmFwaGVtZXMgcGVyIHdvcmQuXG4gICAqL1xuICBjb25zdHJ1Y3RvcihjaHVua3MpIHtcbiAgICAvKipcbiAgICAgKiBUaGUgc3Bva2VuIHNlbnRlbmNlLCBzcGxpdCBpbiBncmFwaGVtZXMgcGVyIHdvcmQuXG4gICAgICogQHR5cGUge1dvcmRDaHVua1tdfVxuICAgICAqL1xuICAgIHRoaXMuY2h1bmtzID0gY2h1bmtzO1xuICB9XG59XG4iXX0= |
@@ -145,4 +145,3 @@ 'use strict'; | ||
* | ||
* This function only allows to make calls to the ITSLanguage API using | ||
* partial URLs. | ||
* This function only allows to make calls to the ITSLanguage API. | ||
* | ||
@@ -164,5 +163,4 @@ * @param {string} method - The request METHOD ('GET', 'POST', 'PUT', 'DELETE'). | ||
// connection reference. | ||
if (url && !url.startsWith('/')) { | ||
console.warn('Complete URLs will soon be disallowed in authorised requests.'); | ||
// return Promise.reject('Only relative ITSLanguage API URLs are allowed.'); | ||
if (url && !url.startsWith('/') && !url.startsWith(settings.apiURL)) { | ||
return Promise.reject('Only relative ITSLanguage API URLs are allowed.'); | ||
} | ||
@@ -179,2 +177,2 @@ | ||
} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/api/communication/index.js"],"names":["updateSettings","request","authorisedRequest","CONTENT_TYPE","AUTHORIZATION","APPLICATION_JSON","settings","apiURL","wsURL","authorizationToken","newSettings","Object","Error","assign","handleResponse","response","responseContentType","headers","get","includes","json","then","ok","Promise","reject","status","statusText","method","url","body","requestHeaders","Headers","requestBody","URLSearchParams","FormData","set","JSON","stringify","requestURL","startsWith","requestOptions","fetch","getBearerToken","console","warn","unauthorised","message"],"mappings":"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","file":"index.js","sourcesContent":["/**\n * This file contains the settings and the communication mechanism for the\n * ITSLanguage REST API.\n */\n\n// Headers\nconst CONTENT_TYPE = 'Content-Type';\nconst AUTHORIZATION = 'Authorization';\n\n// Content-Types\nconst APPLICATION_JSON = 'application/json';\n\n\n/**\n * The settings to use for the communication with the ITSLanguage API.\n */\nexport const settings = {\n  apiURL: 'https://api.itslanguage.nl',\n  wsURL: null,\n  authorizationToken: null\n};\n\n\n/**\n * Update the settings with the `newSettings`.\n *\n * @param {Object} newSettings - The settings to inject/update.\n *\n * @throws {Error} - When the given `newSettings` is something other than a\n *                   object.\n */\nexport function updateSettings(newSettings) {\n  if (!newSettings || !(newSettings instanceof Object)) {\n    throw new Error('Please, only provide objects as settings.');\n  }\n\n  Object.assign(settings, newSettings);\n}\n\n\n/**\n * Parse the response of a fetch request.\n *\n * Try to parse the given response body as JSON, if it isn't Leave the\n * response as is.\n *\n * @param {Response} response - The response to parse.\n *\n * @throws {Promise.<String>} - When the requets was not okay and the contents\n *                              of the response isn't json.\n * @throws {Promise.<Object>} - When the requets was not okay and the contents\n *                              of the response is json.\n *\n * @returns {Promise.<Object>} - The contents of a JSON response or the\n *                               response itself if the body is something other\n *                               than JSON.\n */\nfunction handleResponse(response) {\n  const responseContentType = response.headers.get(CONTENT_TYPE);\n\n  // The ITSLanguage API should return JSON. If t\n  if (responseContentType && responseContentType.includes(APPLICATION_JSON)) {\n    return response.json().then(json => {\n      if (response.ok) {\n        return json;\n      }\n\n      return Promise.reject(json);\n    });\n  }\n\n  if (!response.ok) {\n    return Promise.reject(`${response.status}: ${response.statusText}`);\n  }\n\n  return response;\n}\n\n\n/**\n * Perform an HTTP request for the given method, url, body, and headers.\n *\n * In case the given url is a partial url, meaning it starts with a `/`, the\n * base URL to the ITSLanguage API is prepended.\n *\n * When a Object instance is provided as body, it'll be transformed into JSON.\n * Unless it is either a `URLSearchParams` or a `FormData` object. Anything\n * else is sent as plain text.\n *\n * @param {string} method - The request METHOD ('GET', 'POST', 'PUT', 'DELETE').\n * @param {string} url - The location to send the request to.\n * @param {*} [body] - Anything which needs to be sent somewhere.\n * @param {Headers} [headers] - Extra headers to send with the request.\n *\n * @returns {Promise.<Object>} The response of the made request.\n */\nexport function request(method, url, body, headers) {\n  const requestHeaders = headers || new Headers();\n\n  let requestBody = body;\n  if (!(body instanceof URLSearchParams || body instanceof FormData) && body instanceof Object) {\n    requestHeaders.set('Content-Type', 'application/json');\n    requestBody = JSON.stringify(body);\n  }\n\n  let requestURL = url;\n  // XXX remove the URL truthy check when all tests are properly written. Now\n  // it happens way to often that the URL is omitted without any good reason.\n  if (url && url.startsWith('/')) {\n    requestURL = `${settings.apiURL}${url}`;\n  }\n\n  const requestOptions = {\n    method,\n    headers: requestHeaders,\n    body: requestBody\n  };\n\n  return fetch(requestURL, requestOptions).then(handleResponse);\n}\n\n\n/**\n * Build a bearer token from the `authorizationToken` in the settings object.\n *\n * @throws {Error} When no authorizationToken is set.\n *\n * @returns {string} The generated bearer token.\n */\nfunction getBearerToken() {\n  if (!settings.authorizationToken) {\n    throw new Error('Please authenticate first.');\n  }\n\n  return `Bearer ${settings.authorizationToken}`;\n}\n\n\n/**\n * Perform an HTTP request with the desired method, body, and headers to the\n * given partial ITSLanguage API URL.\n *\n * This request will add the ``Authorization`` header to the request.\n *\n * This function only allows to make calls to the ITSLanguage API using\n * partial URLs.\n *\n * @param {string} method - The request METHOD ('GET', 'POST', 'PUT', 'DELETE').\n * @param {string} url - The location to send the request to.\n * @param {*} [body] - Anything which needs to be sent somewhere.\n * @param {Headers} [headers] - Extra headers to send with the request.\n *\n * @throws {Promise.<string>} - When the given `url` param is not a partial\n *                              URL, or when there is no authorisation token\n *                              availible.\n *\n * @returns {Promise.<Object>} - The response from the ITSLanguage API.\n */\nexport function authorisedRequest(method, url, body, headers) {\n  // XXX remove the URL truthy check when all parts of the SDK no longer build\n  // a complete url by themselves using the \"private\" settings object of their\n  // connection reference.\n  if (url && !url.startsWith('/')) {\n    console.warn('Complete URLs will soon be disallowed in authorised requests.');\n    // return Promise.reject('Only relative ITSLanguage API URLs are allowed.');\n  }\n\n  try {\n    const requestHeaders = headers || new Headers();\n    requestHeaders.set(AUTHORIZATION, getBearerToken());\n\n    return request(method, url, body, requestHeaders);\n  } catch (unauthorised) {\n    return Promise.reject(unauthorised.message);\n  }\n}\n"]} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/api/communication/index.js"],"names":["updateSettings","request","authorisedRequest","CONTENT_TYPE","AUTHORIZATION","APPLICATION_JSON","settings","apiURL","wsURL","authorizationToken","newSettings","Object","Error","assign","handleResponse","response","responseContentType","headers","get","includes","json","then","ok","Promise","reject","status","statusText","method","url","body","requestHeaders","Headers","requestBody","URLSearchParams","FormData","set","JSON","stringify","requestURL","startsWith","requestOptions","fetch","getBearerToken","unauthorised","message"],"mappings":"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","file":"index.js","sourcesContent":["/**\n * This file contains the settings and the communication mechanism for the\n * ITSLanguage REST API.\n */\n\n// Headers\nconst CONTENT_TYPE = 'Content-Type';\nconst AUTHORIZATION = 'Authorization';\n\n// Content-Types\nconst APPLICATION_JSON = 'application/json';\n\n\n/**\n * The settings to use for the communication with the ITSLanguage API.\n */\nexport const settings = {\n  apiURL: 'https://api.itslanguage.nl',\n  wsURL: null,\n  authorizationToken: null\n};\n\n\n/**\n * Update the settings with the `newSettings`.\n *\n * @param {Object} newSettings - The settings to inject/update.\n *\n * @throws {Error} - When the given `newSettings` is something other than a\n *                   object.\n */\nexport function updateSettings(newSettings) {\n  if (!newSettings || !(newSettings instanceof Object)) {\n    throw new Error('Please, only provide objects as settings.');\n  }\n\n  Object.assign(settings, newSettings);\n}\n\n\n/**\n * Parse the response of a fetch request.\n *\n * Try to parse the given response body as JSON, if it isn't Leave the\n * response as is.\n *\n * @param {Response} response - The response to parse.\n *\n * @throws {Promise.<String>} - When the requets was not okay and the contents\n *                              of the response isn't json.\n * @throws {Promise.<Object>} - When the requets was not okay and the contents\n *                              of the response is json.\n *\n * @returns {Promise.<Object>} - The contents of a JSON response or the\n *                               response itself if the body is something other\n *                               than JSON.\n */\nfunction handleResponse(response) {\n  const responseContentType = response.headers.get(CONTENT_TYPE);\n\n  // The ITSLanguage API should return JSON. If t\n  if (responseContentType && responseContentType.includes(APPLICATION_JSON)) {\n    return response.json().then(json => {\n      if (response.ok) {\n        return json;\n      }\n\n      return Promise.reject(json);\n    });\n  }\n\n  if (!response.ok) {\n    return Promise.reject(`${response.status}: ${response.statusText}`);\n  }\n\n  return response;\n}\n\n\n/**\n * Perform an HTTP request for the given method, url, body, and headers.\n *\n * In case the given url is a partial url, meaning it starts with a `/`, the\n * base URL to the ITSLanguage API is prepended.\n *\n * When a Object instance is provided as body, it'll be transformed into JSON.\n * Unless it is either a `URLSearchParams` or a `FormData` object. Anything\n * else is sent as plain text.\n *\n * @param {string} method - The request METHOD ('GET', 'POST', 'PUT', 'DELETE').\n * @param {string} url - The location to send the request to.\n * @param {*} [body] - Anything which needs to be sent somewhere.\n * @param {Headers} [headers] - Extra headers to send with the request.\n *\n * @returns {Promise.<Object>} The response of the made request.\n */\nexport function request(method, url, body, headers) {\n  const requestHeaders = headers || new Headers();\n\n  let requestBody = body;\n  if (!(body instanceof URLSearchParams || body instanceof FormData) && body instanceof Object) {\n    requestHeaders.set('Content-Type', 'application/json');\n    requestBody = JSON.stringify(body);\n  }\n\n  let requestURL = url;\n  // XXX remove the URL truthy check when all tests are properly written. Now\n  // it happens way to often that the URL is omitted without any good reason.\n  if (url && url.startsWith('/')) {\n    requestURL = `${settings.apiURL}${url}`;\n  }\n\n  const requestOptions = {\n    method,\n    headers: requestHeaders,\n    body: requestBody\n  };\n\n  return fetch(requestURL, requestOptions).then(handleResponse);\n}\n\n\n/**\n * Build a bearer token from the `authorizationToken` in the settings object.\n *\n * @throws {Error} When no authorizationToken is set.\n *\n * @returns {string} The generated bearer token.\n */\nfunction getBearerToken() {\n  if (!settings.authorizationToken) {\n    throw new Error('Please authenticate first.');\n  }\n\n  return `Bearer ${settings.authorizationToken}`;\n}\n\n\n/**\n * Perform an HTTP request with the desired method, body, and headers to the\n * given partial ITSLanguage API URL.\n *\n * This request will add the ``Authorization`` header to the request.\n *\n * This function only allows to make calls to the ITSLanguage API.\n *\n * @param {string} method - The request METHOD ('GET', 'POST', 'PUT', 'DELETE').\n * @param {string} url - The location to send the request to.\n * @param {*} [body] - Anything which needs to be sent somewhere.\n * @param {Headers} [headers] - Extra headers to send with the request.\n *\n * @throws {Promise.<string>} - When the given `url` param is not a partial\n *                              URL, or when there is no authorisation token\n *                              availible.\n *\n * @returns {Promise.<Object>} - The response from the ITSLanguage API.\n */\nexport function authorisedRequest(method, url, body, headers) {\n  // XXX remove the URL truthy check when all parts of the SDK no longer build\n  // a complete url by themselves using the \"private\" settings object of their\n  // connection reference.\n  if (url && (!url.startsWith('/') && !url.startsWith(settings.apiURL))) {\n    return Promise.reject('Only relative ITSLanguage API URLs are allowed.');\n  }\n\n  try {\n    const requestHeaders = headers || new Headers();\n    requestHeaders.set(AUTHORIZATION, getBearerToken());\n\n    return request(method, url, body, requestHeaders);\n  } catch (unauthorised) {\n    return Promise.reject(unauthorised.message);\n  }\n}\n"]} |
'use strict'; | ||
exports.__esModule = true; | ||
var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; /** | ||
* | ||
*/ | ||
exports.openWebsocketConnection = openWebsocketConnection; | ||
exports.getWebsocketConnection = getWebsocketConnection; | ||
exports.closeWebsocketConnection = closeWebsocketConnection; | ||
@@ -12,2 +18,6 @@ exports.makeWebsocketCall = makeWebsocketCall; | ||
var _debug = require('debug'); | ||
var _debug2 = _interopRequireDefault(_debug); | ||
var _index = require('./index'); | ||
@@ -17,2 +27,6 @@ | ||
var log = (0, _debug2.default)('its-sdk:WebSocket'); | ||
var error = (0, _debug2.default)('its-sdk:WebSocket'); | ||
log.log = console.log.bind(console); | ||
/** | ||
@@ -23,6 +37,2 @@ * Keep hold of the currently open autobahn connection. | ||
*/ | ||
/** | ||
* | ||
*/ | ||
var bundesautobahn = void 0; | ||
@@ -59,6 +69,8 @@ | ||
var bahn = new _autobahn2.default.Connection({ | ||
url: _index.settings.WS_URL, | ||
url: _index.settings.wsURL, | ||
realm: 'default', | ||
// Of course we want to use es6 promises if they are availbile. | ||
use_es6_promises: true, // eslint-disable-line camelcase | ||
// But, the backend sometimes spits out progress. For that we need | ||
// a When.JS promise.. | ||
use_es6_promises: false, // eslint-disable-line camelcase | ||
// The following options are required in order to authorise the | ||
@@ -86,2 +98,3 @@ // connection. | ||
bahn.onopen = function () { | ||
log('Successfully established a websocket connection.'); | ||
// Remove the `onclose` handler as it is no longer of interest to us. | ||
@@ -116,7 +129,7 @@ delete bahn.onclose; | ||
return establishNewBundesbahn(); | ||
} | ||
}) | ||
// `bundesautobahn` actually resolved with the `autobahn.Connection` | ||
// object. This is only meant for internal usage and therefore should not | ||
// be exposed to the users of the SDK. | ||
).then(function () { | ||
.then(function () { | ||
return 'Successfully established a websocket connection.'; | ||
@@ -156,3 +169,5 @@ }); | ||
bundesautobahn = null; | ||
return 'The websocket connection has been closed successfully.'; | ||
var message = 'The websocket connection has been closed successfully.'; | ||
log(message); | ||
return message; | ||
} catch (reason) { | ||
@@ -164,3 +179,5 @@ // `autobahn.Connection.close()` throws a string when the connection is | ||
bundesautobahn = null; | ||
return 'The websocket connection has already been closed.'; | ||
var _message = 'The websocket connection has already been closed.'; | ||
error(_message); | ||
return _message; | ||
} | ||
@@ -171,3 +188,3 @@ }); | ||
/** | ||
* Make a (raw) call to the its-ws server. | ||
* Make a rpc call to the ITSLanguage websocket server. | ||
* | ||
@@ -180,13 +197,42 @@ * This method will try to establish a websocket connection if there isn't one | ||
* when the RPC starts with that prefix. | ||
* @param {Array} [args] - The positional arguments to pass to the RPC. | ||
* @param {Object} [kwargs] - The key word mapped arguments to pass to the RPC. | ||
* @param {Object} [options] - The options to pass to the RPC. | ||
* @param {Object} [options] - Destructured object with options to pass to the websocket server. | ||
* @param {Array} [options.args] - An array with arguments to pass to the RPC. | ||
* @param {Object} [options.kwargs] - An object (dictionary) with arguments to pass to the RPC. | ||
* @param {Object} [options.options] - The options to pass to the RPC. | ||
* @param {Function} [options.progressCb] - Optional callback to receive progressed results. | ||
* | ||
* @returns {Promise.<*>} - The response of the websocket call. | ||
*/ | ||
function makeWebsocketCall(rpc, args, kwargs, options) { | ||
function makeWebsocketCall(rpc) { | ||
var _ref = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}, | ||
args = _ref.args, | ||
kwargs = _ref.kwargs, | ||
options = _ref.options, | ||
progressCb = _ref.progressCb; | ||
var mergedOptions = options; | ||
if (progressCb) { | ||
mergedOptions = _extends({}, options, { | ||
receive_progress: true // eslint-disable-line camelcase | ||
}); | ||
} | ||
return getWebsocketConnection().then(function (connection) { | ||
return connection.session.call('nl.itslanguage.' + rpc, args, kwargs, options); | ||
return connection.session.call('nl.itslanguage.' + rpc, args, kwargs, mergedOptions).progress(progressCb); | ||
}).catch(function (result) { | ||
var wssError = result.error, | ||
wssKwargs = result.kwargs, | ||
wssArgs = result.args; | ||
// Log the error to stderr | ||
error(result); | ||
// Return a slightly simplistic version of the error that occurred | ||
return Promise.reject(_extends({ | ||
error: wssError | ||
}, wssKwargs, { | ||
args: [].concat(wssArgs) | ||
})); | ||
}); | ||
} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/api/communication/websocket.js"],"names":["openWebsocketConnection","closeWebsocketConnection","makeWebsocketCall","bundesautobahn","handleWebsocketAuthorisationChallenge","session","method","authorizationToken","Error","establishNewBundesbahn","Promise","resolve","reject","bahn","Connection","url","WS_URL","realm","use_es6_promises","authmethods","authid","details","ticket","onchallenge","onclose","message","onopen","open","catch","reason","then","getWebsocketConnection","close","rpc","args","kwargs","options","connection","call"],"mappings":";;;QAuGgBA,uB,GAAAA,uB;QAgCAC,wB,GAAAA,wB;QAsCAC,iB,GAAAA,iB;;AAzKhB;;;;AACA;;;;AAGA;;;;;AARA;;;;AAaA,IAAIC,uBAAJ;;AAGA;;;;;;;;;AASA,SAASC,qCAAT,CAA+CC,OAA/C,EAAwDC,MAAxD,EAAgE;AAC9D,UAAQA,MAAR;AACE,SAAK,QAAL;AACE,aAAO,gBAASC,kBAAhB;AACF;AACE,YAAM,IAAIC,KAAJ,CAAU,oFAC8BF,MAD9B,OAAV,CAAN;AAJJ;AAOD;;AAGD;;;;;;;;AAQA,SAASG,sBAAT,GAAkC;AAChCN,mBAAiB,IAAIO,OAAJ,CAAY,UAACC,OAAD,EAAUC,MAAV,EAAqB;AAChD,QAAMC,OAAO,IAAI,mBAASC,UAAb,CAAwB;AACnCC,WAAK,gBAASC,MADqB;AAEnCC,aAAO,SAF4B;AAGnC;AACAC,wBAAkB,IAJiB,EAIX;AACxB;AACA;AACAC,mBAAa,CAAC,QAAD,CAPsB;AAQnCC,cAAQ,QAR2B;AASnCC,eAAS;AACPC,gBAAQ,gBAASf;AADV,OAT0B;AAYnCgB,mBAAanB;AAZsB,KAAxB,CAAb;;AAeA;AACA;AACAS,SAAKW,OAAL,GAAe,YAAC,qBAA0B;AACxC;AACA;AACA;AACA,UAAMC,UAAU,wDACA,wCADA,GAEA,uDAFA,GAGA,+CAHhB;AAIAb,aAAOa,OAAP;AACD,KATD;;AAWA;AACAZ,SAAKa,MAAL,GAAc,YAAM;AAClB;AACA,aAAOb,KAAKW,OAAZ;AACAb,cAAQE,IAAR;AACD,KAJD;;AAMAA,SAAKc,IAAL;AACD,GArCgB,CAAjB;;AAuCA;AACA;AACA;AACA,SAAOxB,eAAeyB,KAAf,CAAqB,kBAAU;AACpCzB,qBAAiB,IAAjB;AACA,WAAOO,QAAQE,MAAR,CAAeiB,MAAf,CAAP;AACD,GAHM,CAAP;AAID;;AAGD;;;;;;;;;AASO,SAAS7B,uBAAT,GAAmC;AACxC,SAAOC,2BACJ6B,IADI,CACC;AAAA,WAAMrB,wBAAN;AAAA;AACN;AACA;AACA;AAJK,IAKJqB,IALI,CAKC;AAAA,WAAM,kDAAN;AAAA,GALD,CAAP;AAMD;;AAGD;;;;;;;AAOA,SAASC,sBAAT,GAAkC;AAChC,MAAI,CAAC5B,cAAL,EAAqB;AACnB,WAAOM,wBAAP;AACD;;AAED,SAAON,cAAP;AACD;;AAGD;;;;;;AAMO,SAASF,wBAAT,GAAoC;AACzC,MAAI,CAACE,cAAL,EAAqB;AACnB,WAAOO,QAAQC,OAAR,CAAgB,4CAAhB,CAAP;AACD;;AAED,SAAOR,eACJ2B,IADI,CACC,gBAAQ;AACZ,QAAI;AACFjB,WAAKmB,KAAL;AACA7B,uBAAiB,IAAjB;AACA,aAAO,wDAAP;AACD,KAJD,CAIE,OAAO0B,MAAP,EAAe;AACf;AACA;AACA;AACA;AACA1B,uBAAiB,IAAjB;AACA,aAAO,mDAAP;AACD;AACF,GAdI,CAAP;AAeD;;AAGD;;;;;;;;;;;;;;;AAeO,SAASD,iBAAT,CAA2B+B,GAA3B,EAAgCC,IAAhC,EAAsCC,MAAtC,EAA8CC,OAA9C,EAAuD;AAC5D,SAAOL,yBACJD,IADI,CACC;AAAA,WAAcO,WAAWhC,OAAX,CAAmBiC,IAAnB,qBAA0CL,GAA1C,EAAiDC,IAAjD,EAAuDC,MAAvD,EAA+DC,OAA/D,CAAd;AAAA,GADD,CAAP;AAED","file":"websocket.js","sourcesContent":["/**\n *\n */\n\nimport autobahn from 'autobahn';\nimport {settings} from './index';\n\n\n/**\n * Keep hold of the currently open autobahn connection.\n *\n * @type {Promise.<autobahn.Connection>}\n */\nlet bundesautobahn;\n\n\n/**\n * Allow the `autobahn.Connection` to challenge the provided authentication.\n *\n * @param {autobahn.Session} session - The session of the current\n *                                     {@link autobahn.Connection}.\n * @param {string} method - The authentication method it tries to use.\n *\n * @throws {Error} - When the given `method` is unknown to the SDK.\n */\nfunction handleWebsocketAuthorisationChallenge(session, method) {\n  switch (method) {\n    case 'ticket':\n      return settings.authorizationToken;\n    default:\n      throw new Error('The websocket server tried to use the unknown ' +\n                      `authentication challenge: \"${method}\"`);\n  }\n}\n\n\n/**\n * Set {@link bundesautobahn} to a new Promise which resolves into a\n * `autobahn.Connection` object when a connection was successfully established.\n *\n * @returns {Promise.<autobahn.Connection>} - A promise which resolves when the\n *                                            connection was successfully\n *                                            created and opened.\n */\nfunction establishNewBundesbahn() {\n  bundesautobahn = new Promise((resolve, reject) => {\n    const bahn = new autobahn.Connection({\n      url: settings.WS_URL,\n      realm: 'default',\n      // Of course we want to use es6 promises if they are availbile.\n      use_es6_promises: true, // eslint-disable-line camelcase\n      // The following options are required in order to authorise the\n      // connection.\n      authmethods: ['ticket'],\n      authid: 'oauth2',\n      details: {\n        ticket: settings.authorizationToken\n      },\n      onchallenge: handleWebsocketAuthorisationChallenge\n    });\n\n    // `autobahn.Connection` calls its `onclose` method, if it exists, when it\n    // was not able to open a connection.\n    bahn.onclose = (/* reason, details */) => {\n      // When the connection faild to open a reason is given with some details.\n      // Sadly these are very undescriptive. Therefore hint/warn the developer\n      // about potential erroneous settings or to contact us.\n      const message = 'The connection is erroneous; check if all required ' +\n                      'settings have been injected using the ' +\n                      '`updateSettings()` function. If the problem persists ' +\n                      'please post a issue on our GitHub repository.';\n      reject(message);\n    };\n\n    // Connection got established; lets us it.\n    bahn.onopen = () => {\n      // Remove the `onclose` handler as it is no longer of interest to us.\n      delete bahn.onclose;\n      resolve(bahn);\n    };\n\n    bahn.open();\n  });\n\n  // Return the promise to make it this function chainable. In case the\n  // `bundesautobahn` is rejected; remove the reference so we can use simple\n  // falsy checks to detemine if there is a connection.\n  return bundesautobahn.catch(reason => {\n    bundesautobahn = null;\n    return Promise.reject(reason);\n  });\n}\n\n\n/**\n * Open a new websocket connection.\n *\n * There there currently is a open connection, close it and open a new\n * connection.\n *\n * @returns {Promise.<string>} - A resolved promise which resolves when the\n *                               connection was successfully created and opened.\n */\nexport function openWebsocketConnection() {\n  return closeWebsocketConnection()\n    .then(() => establishNewBundesbahn())\n    // `bundesautobahn` actually resolved with the `autobahn.Connection`\n    // object. This is only meant for internal usage and therefore should not\n    // be exposed to the users of the SDK.\n    .then(() => 'Successfully established a websocket connection.');\n}\n\n\n/**\n * Get the current websocket connection, or open a new one.\n *\n * If there is no current connection, open one and return that in stead.\n *\n * @returns {Promise.<autobahn.Connection>} - The current websocket connection.\n */\nfunction getWebsocketConnection() {\n  if (!bundesautobahn) {\n    return establishNewBundesbahn();\n  }\n\n  return bundesautobahn;\n}\n\n\n/**\n * Close the current websocket connection.\n *\n * @returns {Promise.<string>} - A promise which will resolve as soon as the\n *                               connection was successfully closed.\n */\nexport function closeWebsocketConnection() {\n  if (!bundesautobahn) {\n    return Promise.resolve('There is no websocket connection to close.');\n  }\n\n  return bundesautobahn\n    .then(bahn => {\n      try {\n        bahn.close();\n        bundesautobahn = null;\n        return 'The websocket connection has been closed successfully.';\n      } catch (reason) {\n        // `autobahn.Connection.close()` throws a string when the connection is\n        // already closed. The connection is not exposed and therefore cannot be\n        // closed by anyone using the SDK. Regardless, when it happens just\n        // return a resolved promise.\n        bundesautobahn = null;\n        return 'The websocket connection has already been closed.';\n      }\n    });\n}\n\n\n/**\n * Make a (raw) call to the its-ws server.\n *\n * This method will try to establish a websocket connection if there isn't one\n * already.\n *\n * @param {string} rpc - The RPC to make. This be prepended by `nl.itslanguage`\n *                       as the websocket server only handles websocket calls\n *                       when the RPC starts with that prefix.\n * @param {Array} [args] - The positional arguments to pass to the RPC.\n * @param {Object} [kwargs] - The key word mapped arguments to pass to the RPC.\n * @param {Object} [options] - The options to pass to the RPC.\n *\n * @returns {Promise.<*>} - The response of the websocket call.\n */\nexport function makeWebsocketCall(rpc, args, kwargs, options) {\n  return getWebsocketConnection()\n    .then(connection => connection.session.call(`nl.itslanguage.${rpc}`, args, kwargs, options));\n}\n"]} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/api/communication/websocket.js"],"names":["openWebsocketConnection","getWebsocketConnection","closeWebsocketConnection","makeWebsocketCall","log","error","console","bind","bundesautobahn","handleWebsocketAuthorisationChallenge","session","method","authorizationToken","Error","establishNewBundesbahn","Promise","resolve","reject","bahn","Connection","url","wsURL","realm","use_es6_promises","authmethods","authid","details","ticket","onchallenge","onclose","message","onopen","open","catch","reason","then","close","rpc","args","kwargs","options","progressCb","mergedOptions","receive_progress","connection","call","progress","wssError","result","wssKwargs","wssArgs"],"mappings":";;;;kQAAA;;;;QA8GgBA,uB,GAAAA,uB;QAiBAC,sB,GAAAA,sB;QAeAC,wB,GAAAA,wB;QA4CAC,iB,GAAAA,iB;;AAtLhB;;;;AACA;;;;AACA;;;;AAEA,IAAMC,MAAM,qBAAM,mBAAN,CAAZ;AACA,IAAMC,QAAQ,qBAAM,mBAAN,CAAd;AACAD,IAAIA,GAAJ,GAAUE,QAAQF,GAAR,CAAYG,IAAZ,CAAiBD,OAAjB,CAAV;;AAEA;;;;;AAKA,IAAIE,uBAAJ;;AAGA;;;;;;;;;AASA,SAASC,qCAAT,CAA+CC,OAA/C,EAAwDC,MAAxD,EAAgE;AAC9D,UAAQA,MAAR;AACE,SAAK,QAAL;AACE,aAAO,gBAASC,kBAAhB;AACF;AACE,YAAM,IAAIC,KAAJ,CAAU,oFAC8BF,MAD9B,OAAV,CAAN;AAJJ;AAOD;;AAGD;;;;;;;;AAQA,SAASG,sBAAT,GAAkC;AAChCN,mBAAiB,IAAIO,OAAJ,CAAY,UAACC,OAAD,EAAUC,MAAV,EAAqB;AAChD,QAAMC,OAAO,IAAI,mBAASC,UAAb,CAAwB;AACnCC,WAAK,gBAASC,KADqB;AAEnCC,aAAO,SAF4B;AAGnC;AACA;AACA;AACAC,wBAAkB,KANiB,EAMV;AACzB;AACA;AACAC,mBAAa,CAAC,QAAD,CATsB;AAUnCC,cAAQ,QAV2B;AAWnCC,eAAS;AACPC,gBAAQ,gBAASf;AADV,OAX0B;AAcnCgB,mBAAanB;AAdsB,KAAxB,CAAb;;AAiBA;AACA;AACAS,SAAKW,OAAL,GAAe,YAAC,qBAA0B;AACxC;AACA;AACA;AACA,UAAMC,UAAU,wDACA,wCADA,GAEA,uDAFA,GAGA,+CAHhB;AAIAb,aAAOa,OAAP;AACD,KATD;;AAWA;AACAZ,SAAKa,MAAL,GAAc,YAAM;AAClB3B,UAAI,kDAAJ;AACA;AACA,aAAOc,KAAKW,OAAZ;AACAb,cAAQE,IAAR;AACD,KALD;;AAOAA,SAAKc,IAAL;AACD,GAxCgB,CAAjB;;AA0CA;AACA;AACA;AACA,SAAOxB,eAAeyB,KAAf,CAAqB,kBAAU;AACpCzB,qBAAiB,IAAjB;AACA,WAAOO,QAAQE,MAAR,CAAeiB,MAAf,CAAP;AACD,GAHM,CAAP;AAID;;AAGD;;;;;;;;;AASO,SAASlC,uBAAT,GAAmC;AACxC,SAAOE,2BACJiC,IADI,CACC;AAAA,WAAMrB,wBAAN;AAAA,GADD;AAEL;AACA;AACA;AAJK,GAKJqB,IALI,CAKC;AAAA,WAAM,kDAAN;AAAA,GALD,CAAP;AAMD;;AAGD;;;;;;;AAOO,SAASlC,sBAAT,GAAkC;AACvC,MAAI,CAACO,cAAL,EAAqB;AACnB,WAAOM,wBAAP;AACD;;AAED,SAAON,cAAP;AACD;;AAGD;;;;;;AAMO,SAASN,wBAAT,GAAoC;AACzC,MAAI,CAACM,cAAL,EAAqB;AACnB,WAAOO,QAAQC,OAAR,CAAgB,4CAAhB,CAAP;AACD;;AAED,SAAOR,eACJ2B,IADI,CACC,gBAAQ;AACZ,QAAI;AACFjB,WAAKkB,KAAL;AACA5B,uBAAiB,IAAjB;AACA,UAAMsB,UAAU,wDAAhB;AACA1B,UAAI0B,OAAJ;AACA,aAAOA,OAAP;AACD,KAND,CAME,OAAOI,MAAP,EAAe;AACf;AACA;AACA;AACA;AACA1B,uBAAiB,IAAjB;AACA,UAAMsB,WAAU,mDAAhB;AACAzB,YAAMyB,QAAN;AACA,aAAOA,QAAP;AACD;AACF,GAlBI,CAAP;AAmBD;;AAGD;;;;;;;;;;;;;;;;;AAiBO,SAAS3B,iBAAT,CAA2BkC,GAA3B,EAA0E;AAAA,iFAAJ,EAAI;AAAA,MAAzCC,IAAyC,QAAzCA,IAAyC;AAAA,MAAnCC,MAAmC,QAAnCA,MAAmC;AAAA,MAA3BC,OAA2B,QAA3BA,OAA2B;AAAA,MAAlBC,UAAkB,QAAlBA,UAAkB;;AAC/E,MAAIC,gBAAgBF,OAApB;AACA,MAAIC,UAAJ,EAAgB;AACdC,iCACKF,OADL;AAEEG,wBAAkB,IAFpB,CAEyB;AAFzB;AAID;AACD,SAAO1C,yBACJkC,IADI,CACC;AAAA,WACJS,WAAWlC,OAAX,CAAmBmC,IAAnB,qBAA0CR,GAA1C,EAAiDC,IAAjD,EAAuDC,MAAvD,EAA+DG,aAA/D,EACGI,QADH,CACYL,UADZ,CADI;AAAA,GADD,EAKJR,KALI,CAKE,kBAAU;AAAA,QACDc,QADC,GAC6CC,MAD7C,CACR3C,KADQ;AAAA,QACiB4C,SADjB,GAC6CD,MAD7C,CACST,MADT;AAAA,QACkCW,OADlC,GAC6CF,MAD7C,CAC4BV,IAD5B;;AAGf;;AACAjC,UAAM2C,MAAN;;AAEA;AACA,WAAOjC,QAAQE,MAAR;AACLZ,aAAO0C;AADF,OAEFE,SAFE;AAGLX,sBAAUY,OAAV;AAHK,OAAP;AAKD,GAjBI,CAAP;AAkBD","file":"websocket.js","sourcesContent":["/**\n *\n */\n\nimport autobahn from 'autobahn';\nimport debug from 'debug';\nimport {settings} from './index';\n\nconst log = debug('its-sdk:WebSocket');\nconst error = debug('its-sdk:WebSocket');\nlog.log = console.log.bind(console);\n\n/**\n * Keep hold of the currently open autobahn connection.\n *\n * @type {Promise.<autobahn.Connection>}\n */\nlet bundesautobahn;\n\n\n/**\n * Allow the `autobahn.Connection` to challenge the provided authentication.\n *\n * @param {autobahn.Session} session - The session of the current\n *                                     {@link autobahn.Connection}.\n * @param {string} method - The authentication method it tries to use.\n *\n * @throws {Error} - When the given `method` is unknown to the SDK.\n */\nfunction handleWebsocketAuthorisationChallenge(session, method) {\n  switch (method) {\n    case 'ticket':\n      return settings.authorizationToken;\n    default:\n      throw new Error('The websocket server tried to use the unknown ' +\n                      `authentication challenge: \"${method}\"`);\n  }\n}\n\n\n/**\n * Set {@link bundesautobahn} to a new Promise which resolves into a\n * `autobahn.Connection` object when a connection was successfully established.\n *\n * @returns {Promise.<autobahn.Connection>} - A promise which resolves when the\n *                                            connection was successfully\n *                                            created and opened.\n */\nfunction establishNewBundesbahn() {\n  bundesautobahn = new Promise((resolve, reject) => {\n    const bahn = new autobahn.Connection({\n      url: settings.wsURL,\n      realm: 'default',\n      // Of course we want to use es6 promises if they are availbile.\n      // But, the backend sometimes spits out progress. For that we need\n      // a When.JS promise..\n      use_es6_promises: false, // eslint-disable-line camelcase\n      // The following options are required in order to authorise the\n      // connection.\n      authmethods: ['ticket'],\n      authid: 'oauth2',\n      details: {\n        ticket: settings.authorizationToken\n      },\n      onchallenge: handleWebsocketAuthorisationChallenge\n    });\n\n    // `autobahn.Connection` calls its `onclose` method, if it exists, when it\n    // was not able to open a connection.\n    bahn.onclose = (/* reason, details */) => {\n      // When the connection faild to open a reason is given with some details.\n      // Sadly these are very undescriptive. Therefore hint/warn the developer\n      // about potential erroneous settings or to contact us.\n      const message = 'The connection is erroneous; check if all required ' +\n                      'settings have been injected using the ' +\n                      '`updateSettings()` function. If the problem persists ' +\n                      'please post a issue on our GitHub repository.';\n      reject(message);\n    };\n\n    // Connection got established; lets us it.\n    bahn.onopen = () => {\n      log('Successfully established a websocket connection.');\n      // Remove the `onclose` handler as it is no longer of interest to us.\n      delete bahn.onclose;\n      resolve(bahn);\n    };\n\n    bahn.open();\n  });\n\n  // Return the promise to make it this function chainable. In case the\n  // `bundesautobahn` is rejected; remove the reference so we can use simple\n  // falsy checks to detemine if there is a connection.\n  return bundesautobahn.catch(reason => {\n    bundesautobahn = null;\n    return Promise.reject(reason);\n  });\n}\n\n\n/**\n * Open a new websocket connection.\n *\n * There there currently is a open connection, close it and open a new\n * connection.\n *\n * @returns {Promise.<string>} - A resolved promise which resolves when the\n *                               connection was successfully created and opened.\n */\nexport function openWebsocketConnection() {\n  return closeWebsocketConnection()\n    .then(() => establishNewBundesbahn())\n    // `bundesautobahn` actually resolved with the `autobahn.Connection`\n    // object. This is only meant for internal usage and therefore should not\n    // be exposed to the users of the SDK.\n    .then(() => 'Successfully established a websocket connection.');\n}\n\n\n/**\n * Get the current websocket connection, or open a new one.\n *\n * If there is no current connection, open one and return that in stead.\n *\n * @returns {Promise.<autobahn.Connection>} - The current websocket connection.\n */\nexport function getWebsocketConnection() {\n  if (!bundesautobahn) {\n    return establishNewBundesbahn();\n  }\n\n  return bundesautobahn;\n}\n\n\n/**\n * Close the current websocket connection.\n *\n * @returns {Promise.<string>} - A promise which will resolve as soon as the\n *                               connection was successfully closed.\n */\nexport function closeWebsocketConnection() {\n  if (!bundesautobahn) {\n    return Promise.resolve('There is no websocket connection to close.');\n  }\n\n  return bundesautobahn\n    .then(bahn => {\n      try {\n        bahn.close();\n        bundesautobahn = null;\n        const message = 'The websocket connection has been closed successfully.';\n        log(message);\n        return message;\n      } catch (reason) {\n        // `autobahn.Connection.close()` throws a string when the connection is\n        // already closed. The connection is not exposed and therefore cannot be\n        // closed by anyone using the SDK. Regardless, when it happens just\n        // return a resolved promise.\n        bundesautobahn = null;\n        const message = 'The websocket connection has already been closed.';\n        error(message);\n        return message;\n      }\n    });\n}\n\n\n/**\n * Make a rpc call to the ITSLanguage websocket server.\n *\n * This method will try to establish a websocket connection if there isn't one\n * already.\n *\n * @param {string} rpc - The RPC to make. This be prepended by `nl.itslanguage`\n *                       as the websocket server only handles websocket calls\n *                       when the RPC starts with that prefix.\n * @param {Object} [options] - Destructured object with options to pass to the websocket server.\n * @param {Array} [options.args] - An array with arguments to pass to the RPC.\n * @param {Object} [options.kwargs] - An object (dictionary) with arguments to pass to the RPC.\n * @param {Object} [options.options] - The options to pass to the RPC.\n * @param {Function} [options.progressCb] - Optional callback to receive progressed results.\n *\n * @returns {Promise.<*>} - The response of the websocket call.\n */\nexport function makeWebsocketCall(rpc, {args, kwargs, options, progressCb} = {}) {\n  let mergedOptions = options;\n  if (progressCb) {\n    mergedOptions = {\n      ...options,\n      receive_progress: true // eslint-disable-line camelcase\n    };\n  }\n  return getWebsocketConnection()\n    .then(connection =>\n      connection.session.call(`nl.itslanguage.${rpc}`, args, kwargs, mergedOptions)\n        .progress(progressCb)\n    )\n    .catch(result => {\n      const {error: wssError, kwargs: wssKwargs, args: wssArgs} = result;\n\n      // Log the error to stderr\n      error(result);\n\n      // Return a slightly simplistic version of the error that occurred\n      return Promise.reject({\n        error: wssError,\n        ...wssKwargs,\n        args: [...wssArgs]\n      });\n    });\n}\n"]} |
@@ -5,6 +5,2 @@ 'use strict'; | ||
var _cordovaMediaPlayer = require('./cordova-media-player'); | ||
var _cordovaMediaPlayer2 = _interopRequireDefault(_cordovaMediaPlayer); | ||
var _tools = require('./tools'); | ||
@@ -84,5 +80,5 @@ | ||
/** | ||
* @type {CordovaMediaPlayer|WebAudioPlayer} player - Specific audio player. | ||
* @type {WebAudioPlayer} player - Specific audio player. | ||
*/ | ||
this._player = this._getBestPlayer(callbacks); | ||
this._player = this._getPlayer(callbacks); | ||
this._emitter = (0, _eventEmitter2.default)({}); | ||
@@ -143,9 +139,3 @@ this._stopwatch = null; | ||
// Detect Cordova Media Playback | ||
// It allows playing audio using the native bridge inside WebView Apps. | ||
// https://github.com/apache/cordova-plugin-media/blob/master/doc/index.md | ||
this.canUseCordovaMedia = Boolean(window.Media); | ||
console.log('Cordova Media playback capability:', this.canUseCordovaMedia); | ||
if (!this.canUseAudio && !this.canUseCordovaMedia) { | ||
if (!this.canUseAudio) { | ||
throw new Error('Some form of audio playback capability is required'); | ||
@@ -190,13 +180,6 @@ } | ||
AudioPlayer.prototype._getBestPlayer = function _getBestPlayer(callbacks) { | ||
AudioPlayer.prototype._getPlayer = function _getPlayer(callbacks) { | ||
var player = null; | ||
// Start by checking for a Cordova environment. | ||
// When running under a debugger like Ripple, both Cordova and WebAudio | ||
// environments get detected. While this is technically valid -Ripple is | ||
// running in Chrome, which supports WebAudio-, it's not a sandbox that | ||
// also disables functionality that would not be available on a device. | ||
if (this.canUseCordovaMedia) { | ||
// Use Cordova audio encoding (used codec depends on the platform). | ||
player = new _cordovaMediaPlayer2.default(callbacks); | ||
} else if (this.canUseAudio) { | ||
if (this.canUseAudio) { | ||
// Use the recorder with MediaRecorder implementation. | ||
@@ -510,2 +493,2 @@ player = new _webAudioPlayer2.default(callbacks); | ||
exports.default = AudioPlayer; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/audio/audio-player.js"],"names":["AudioPlayer","options","_settings","Object","assign","_playbackCompatibility","self","callbacks","playingCb","_emitter","emit","timeupdateCb","durationchangeCb","canplayCb","endedCb","pauseCb","stoppedCb","playbackStoppedCb","_stopwatch","stop","progressCb","errorCb","_player","_getBestPlayer","_audioLevel","_audioMuted","resetEventListeners","addEventListener","name","handler","on","removeEventListener","off","canUseAudio","Boolean","Audio","console","log","canUseCordovaMedia","window","Media","Error","_audio","canPlayType","Function","canPlayOggVorbis","canPlayOggOpus","canPlayWave","canPlayMP3","canPlayAAC","canPlay3GPP","player","load","url","preload","loadedCb","reset","play","position","isPlaying","time","Math","round","getCurrentTime","_value","start","pause","togglePlayback","scrub","percentage","getBufferFill","getDuration","canPlay","setPlaybackRate","rate","getPlaybackRate","bindStopwatch","tickCb","duration","sound","playbackRate","setAudioVolume","volume","getAudioVolume","toggleAudioMute","setAudioMute","shouldMute","isAudioMuted"],"mappings":"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","file":"audio-player.js","sourcesContent":["import CordovaMediaPlayer from './cordova-media-player';\nimport Stopwatch from './tools';\nimport WebAudioPlayer from './web-audio-player';\nimport allOff from 'event-emitter/all-off';\nimport ee from 'event-emitter';\n/**\n * ITSLanguage AudioPlayer non-graphical component.\n */\nexport default class AudioPlayer {\n  /**\n   * Construct an AudioPlayer for playing .wav or .mp3 files.\n   *\n   * @param {?Object} options - Override any of the default settings.\n   * @emits {Event} 'playbackstopped' When playback has ended, been stopped or been paused.\n   * @emits {Event} All events the HTML5 Audio also fires. {@link http://www.w3schools.com/tags/ref_av_dom.asp}\n   */\n  constructor(options) {\n    this._settings = Object.assign({}, options);\n\n    this._playbackCompatibility();\n    const self = this;\n    const callbacks = {\n      playingCb() {\n        self._emitter.emit('playing', []);\n      },\n      timeupdateCb() {\n        self._emitter.emit('timeupdate', []);\n      },\n      durationchangeCb() {\n        self._emitter.emit('durationchange', []);\n      },\n      canplayCb() {\n        self._emitter.emit('canplay', []);\n      },\n      endedCb() {\n        self._emitter.emit('ended', []);\n      },\n      pauseCb() {\n        self._emitter.emit('pause', []);\n      },\n      stoppedCb() {\n        self._emitter.emit('stopped', []);\n      },\n      playbackStoppedCb() {\n        self._emitter.emit('playbackstopped', []);\n        if (self._stopwatch) {\n          self._stopwatch.stop();\n        }\n      },\n      progressCb() {\n        self._emitter.emit('progress', []);\n      },\n      errorCb() {\n        self._emitter.emit('error', []);\n      }\n    };\n    /**\n     * @type {CordovaMediaPlayer|WebAudioPlayer} player - Specific audio player.\n     */\n    this._player = this._getBestPlayer(callbacks);\n    this._emitter = ee({});\n    this._stopwatch = null;\n    this._audioLevel = 1;\n    this._audioMuted = false;\n  }\n\n  /**\n   * Turn off all event listeners for this player.\n   */\n  resetEventListeners() {\n    allOff(this._emitter);\n  }\n\n  /**\n   * Add an event listener. Listens to events emitted from the player.\n   *\n   * @param {string} name - Name of the event.\n   * @param {Function} handler - Handler function to add.\n   */\n  addEventListener(name, handler) {\n    this._emitter.on(name, handler);\n  }\n\n  /**\n   * Remove an event listener of the player.\n   *\n   * @param {string} name - Name of the event.\n   * @param {Function} handler - Handler function to remove.\n   */\n  removeEventListener(name, handler) {\n    this._emitter.off(name, handler);\n  }\n\n  /**\n   * Check for mandatory browser compatibility.\n   * Logs detailed browser compatibilities related to for audio playback.\n   *\n   * @throws {Error} If no native wave or MP3 playback is available.\n   */\n  _playbackCompatibility() {\n    // Detect audio playback capabilities.\n\n    // Detect HTML5 Audio playback.\n    // http://caniuse.com/#feat=audio\n    this.canUseAudio = Boolean(Audio);\n    console.log('Native HTML5 Audio playback capability:', this.canUseAudio);\n\n    // Detect Cordova Media Playback\n    // It allows playing audio using the native bridge inside WebView Apps.\n    // https://github.com/apache/cordova-plugin-media/blob/master/doc/index.md\n    this.canUseCordovaMedia = Boolean(window.Media);\n    console.log('Cordova Media playback capability:', this.canUseCordovaMedia);\n\n    if (!this.canUseAudio && !this.canUseCordovaMedia) {\n      throw new Error(\n        'Some form of audio playback capability is required');\n    }\n    if (this.canUseAudio) {\n      const _audio = new Audio();\n      if (!(_audio.canPlayType && _audio.canPlayType instanceof Function)) {\n        throw new Error(\n          'Unable to detect audio playback capabilities');\n      }\n      const canPlayOggVorbis = _audio.canPlayType(\n          'audio/ogg; codecs=\"vorbis\"') !== '';\n      const canPlayOggOpus = _audio.canPlayType(\n          'audio/ogg; codecs=\"opus\"') !== '';\n      const canPlayWave = _audio.canPlayType('audio/wav') !== '';\n      const canPlayMP3 = _audio.canPlayType('audio/mpeg; codecs=\"mp3\"') !== '';\n      const canPlayAAC = _audio.canPlayType(\n          'audio/mp4; codecs=\"mp4a.40.2\"') !== '';\n      const canPlay3GPP = _audio.canPlayType(\n          'audio/3gpp; codecs=\"samr\"') !== '';\n\n      console.log('Native Vorbis audio in Ogg container playback capability:', canPlayOggVorbis);\n      console.log('Native Opus audio in Ogg container playback capability:', canPlayOggOpus);\n      console.log('Native PCM audio in Waveform Audio File Format (WAVE) playback capability:', canPlayWave);\n      console.log('Native MPEG Audio Layer 3 (MP3) playback capability:', canPlayMP3);\n      console.log('Native Low-Complexity AAC audio in MP4 container playback capability:', canPlayAAC);\n      console.log('Native AMR audio in 3GPP container playback capability:', canPlay3GPP);\n\n      if (!(canPlayWave || canPlayMP3)) {\n        throw new Error(\n          'Native Wave or MP3 playback is required');\n      }\n    }\n  }\n\n  /**\n   * Get a player object that performs audio compression, when available.\n   *\n   * Using the Media Stream Recording API for recording is the preferred\n   * solution. It allows recording compressed audio which makes it quicker to\n   * submit. If not available, use a default createScriptProcessor is used.\n   *\n   * @param {Function} callbacks - Callbacks to add to the chosen player.\n   * @private\n   */\n  _getBestPlayer(callbacks) {\n    let player = null;\n    // Start by checking for a Cordova environment.\n    // When running under a debugger like Ripple, both Cordova and WebAudio\n    // environments get detected. While this is technically valid -Ripple is\n    // running in Chrome, which supports WebAudio-, it's not a sandbox that\n    // also disables functionality that would not be available on a device.\n    if (this.canUseCordovaMedia) {\n      // Use Cordova audio encoding (used codec depends on the platform).\n      player = new CordovaMediaPlayer(callbacks);\n    } else if (this.canUseAudio) {\n      // Use the recorder with MediaRecorder implementation.\n      player = new WebAudioPlayer(callbacks);\n    } else {\n      throw new Error('Unable to find a proper player.');\n    }\n    console.log('Player initialised.');\n    return player;\n  }\n\n  /**\n   * Preload audio from an URL.\n   *\n   * @param {string} url - The URL that contains the audio.\n   * @param {boolean} [preload=true] - Try preloading metadata and possible some audio. Set to false to not download\n   * anything until playing.\n   * @param {?Function} loadedCb - The callback that is invoked when the duration of the audio file\n   * is first known.\n   * @emits {Event} 'canplay' When the player is ready to play.\n   */\n  load(url, preload, loadedCb) {\n    this.reset();\n    this._player.load(url, preload, loadedCb);\n    this._audioLevel = 1;\n\n    // If preloading is disabled, the 'canplay' event won't be triggered.\n    // In that case, fire it manually.\n    if (!preload) {\n      this._emitter.emit('canplay', []);\n    }\n  }\n\n  /**\n   * Unload previously loaded audio. Stops the player and any stopwatch.\n   *\n   * @emits {Event} 'unloaded'\n   */\n  reset() {\n    this.stop();\n    this._player.reset();\n    this._emitter.emit('unloaded', []);\n  }\n\n  /**\n   * Start or continue playback of audio. Also starts the stopwatch at the given position.\n   *\n   * @param {?number} position - When position is given, start playing from this position (seconds).\n   */\n  play(position) {\n    if (this._player.isPlaying()) {\n      return;\n    }\n    this._player.play(position);\n    if (this._stopwatch) {\n      const time = Math.round(this._player.getCurrentTime() * 10);\n      this._stopwatch._value = time;\n      this._stopwatch.start();\n    }\n  }\n\n  /**\n   * Stop playback of audio. Stops and resets the stopwatch.\n   */\n  stop() {\n    if (this._stopwatch) {\n      this._stopwatch.reset();\n      this._stopwatch.stop();\n    }\n    this._player.stop();\n  }\n\n  /**\n   * Pause playback of audio. Stops the stopwatch.\n   */\n  pause() {\n    if (this._stopwatch) {\n      this._stopwatch.stop();\n    }\n    this._player.pause();\n  }\n\n  /**\n   * Toggle audio playback. Switch from playing to paused state and back.\n   */\n  togglePlayback() {\n    if (this._player.isPlaying()) {\n      this.pause();\n    } else {\n      this.play();\n    }\n  }\n\n  /**\n   * Start preloading audio.\n   */\n  preload() {\n    this._player.preload();\n  }\n\n  /**\n   * Start playing audio at the given offset. Corrects a percentage under 0 or above 100 to the respective values.\n   *\n   * @param {number} percentage - Start at this percentage (0..100) of the audio stream.\n   */\n  scrub(percentage) {\n    if (percentage < 0) {\n      this._player.scrub(0);\n    } else if (percentage > 100) {\n      this._player.scrub(100);\n    } else {\n      this._player.scrub(percentage);\n    }\n    if (this._stopwatch) {\n      this._stopwatch._value = Math.round(this._player.getCurrentTime() * 10);\n    }\n  }\n\n  /**\n   * Returns the percentage of which the buffer is filled.\n   *\n   * @returns {number} Percentage of buffer fill.\n   */\n  getBufferFill() {\n    return this._player.getBufferFill();\n  }\n\n  /**\n   * Returns the current playing time as offset in seconds from the start.\n   *\n   * @returns {number} Time in seconds as offset from the start.\n   */\n  getCurrentTime() {\n    return this._player.getCurrentTime();\n  }\n\n  /**\n   * Returns the total duration in seconds.\n   *\n   * @returns {number} Time in seconds of fragment duration.\n   */\n  getDuration() {\n    return this._player.getDuration();\n  }\n\n  /**\n   * Check if there is playback in progress.\n   *\n   * @returns {boolean} True if user is currently playing audio. False otherwise.\n   */\n  isPlaying() {\n    return this._player.isPlaying();\n  }\n\n  /**\n   * Returns ready state of the player.\n   *\n   * @returns {boolean} True when player is ready to start loading data or play. False when no audio is loaded\n   * or the player is preparing.\n   */\n  canPlay() {\n    return this._player.canPlay();\n  }\n\n  /**\n   * Set the playback rate of the audio. Values are used according to HTML5 Audio.\n   * Example values:\n   * *1.0 is normal speed.\n   * *0.5 is half speed (slower).\n   * *2.0 is double speed (faster).\n   * *-1.0 is backwards, normal speed.\n   * *-0.5 is backwards, half speed.\n   *\n   * @param {number} rate - Rate at which to change the audio playback.\n   */\n  setPlaybackRate(rate) {\n    this._player.setPlaybackRate(rate);\n  }\n\n  /**\n   * Get the playback rate of the current loaded audio.\n   *\n   * @returns {number} Playback rate of the audio.\n   */\n  getPlaybackRate() {\n    return this._player.getPlaybackRate();\n  }\n\n  /**\n   * Bind a stopwatch to sync with the playing and stopping functionality of the player.\n   *\n   * @param {Function} tickCb - Callback to invoke on every tick. A tick occurs once every 100 ms.\n   * @throws {Error} If _tickCb is null.\n   * @returns {Stopwatch} New Stopwatch object.\n   */\n  bindStopwatch(tickCb) {\n    this._stopwatch = new Stopwatch(time => {\n      const duration = this.getDuration() * 10 / this._player.sound.playbackRate;\n      if (time > duration) {\n        tickCb(duration);\n      } else {\n        tickCb(time);\n      }\n    });\n    return this._stopwatch;\n  }\n\n  /**\n   * Sets the audio level of the current loaded audio. Valid values are from 0 (0%) to 1 (100%).\n   *\n   * @param {number} volume - Volume value from 0 to 1.\n   */\n  setAudioVolume(volume) {\n    if (volume !== 0) {\n      this._audioMuted = false;\n    }\n    if (volume === 0) {\n      this._audioMuted = true;\n    }\n    this._player.setAudioVolume(volume);\n  }\n\n  /**\n   * Gets the audio level of the current loaded audio. Valid values are from 0 (0%) to 1 (100%).\n   *\n   * @returns {number} Volume level of the current loaded audio.\n   */\n  getAudioVolume() {\n    return this._player.getAudioVolume();\n  }\n\n  /**\n   * Toggle the current playing audio to be muted or not. If the audio will be muted, the current audio level\n   * is remembered and can be unmuted to continue at this same audio level.\n   */\n  toggleAudioMute() {\n    this.setAudioMute(!this._audioMuted);\n  }\n\n  /**\n   * Manually set the muted state of the current loaded audio.\n   *\n   * @param {boolean} shouldMute - Whether the audio should be muted or unmuted.\n   */\n  setAudioMute(shouldMute) {\n    if (shouldMute) {\n      this._audioLevel = this.getAudioVolume();\n      this.setAudioVolume(0);\n    } else {\n      this.setAudioVolume(this._audioLevel);\n    }\n  }\n\n  /**\n   * Return the muted state of the current loaded audio.\n   *\n   * @returns {boolean} The muted state of the current loaded audio.\n   */\n  isAudioMuted() {\n    return this._audioMuted;\n  }\n}\n"]} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/audio/audio-player.js"],"names":["AudioPlayer","options","_settings","Object","assign","_playbackCompatibility","self","callbacks","playingCb","_emitter","emit","timeupdateCb","durationchangeCb","canplayCb","endedCb","pauseCb","stoppedCb","playbackStoppedCb","_stopwatch","stop","progressCb","errorCb","_player","_getPlayer","_audioLevel","_audioMuted","resetEventListeners","addEventListener","name","handler","on","removeEventListener","off","canUseAudio","Boolean","Audio","console","log","Error","_audio","canPlayType","Function","canPlayOggVorbis","canPlayOggOpus","canPlayWave","canPlayMP3","canPlayAAC","canPlay3GPP","player","load","url","preload","loadedCb","reset","play","position","isPlaying","time","Math","round","getCurrentTime","_value","start","pause","togglePlayback","scrub","percentage","getBufferFill","getDuration","canPlay","setPlaybackRate","rate","getPlaybackRate","bindStopwatch","tickCb","duration","sound","playbackRate","setAudioVolume","volume","getAudioVolume","toggleAudioMute","setAudioMute","shouldMute","isAudioMuted"],"mappings":"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","file":"audio-player.js","sourcesContent":["import Stopwatch from './tools';\nimport WebAudioPlayer from './web-audio-player';\nimport allOff from 'event-emitter/all-off';\nimport ee from 'event-emitter';\n/**\n * ITSLanguage AudioPlayer non-graphical component.\n */\nexport default class AudioPlayer {\n  /**\n   * Construct an AudioPlayer for playing .wav or .mp3 files.\n   *\n   * @param {?Object} options - Override any of the default settings.\n   * @emits {Event} 'playbackstopped' When playback has ended, been stopped or been paused.\n   * @emits {Event} All events the HTML5 Audio also fires. {@link http://www.w3schools.com/tags/ref_av_dom.asp}\n   */\n  constructor(options) {\n    this._settings = Object.assign({}, options);\n\n    this._playbackCompatibility();\n    const self = this;\n    const callbacks = {\n      playingCb() {\n        self._emitter.emit('playing', []);\n      },\n      timeupdateCb() {\n        self._emitter.emit('timeupdate', []);\n      },\n      durationchangeCb() {\n        self._emitter.emit('durationchange', []);\n      },\n      canplayCb() {\n        self._emitter.emit('canplay', []);\n      },\n      endedCb() {\n        self._emitter.emit('ended', []);\n      },\n      pauseCb() {\n        self._emitter.emit('pause', []);\n      },\n      stoppedCb() {\n        self._emitter.emit('stopped', []);\n      },\n      playbackStoppedCb() {\n        self._emitter.emit('playbackstopped', []);\n        if (self._stopwatch) {\n          self._stopwatch.stop();\n        }\n      },\n      progressCb() {\n        self._emitter.emit('progress', []);\n      },\n      errorCb() {\n        self._emitter.emit('error', []);\n      }\n    };\n    /**\n     * @type {WebAudioPlayer} player - Specific audio player.\n     */\n    this._player = this._getPlayer(callbacks);\n    this._emitter = ee({});\n    this._stopwatch = null;\n    this._audioLevel = 1;\n    this._audioMuted = false;\n  }\n\n  /**\n   * Turn off all event listeners for this player.\n   */\n  resetEventListeners() {\n    allOff(this._emitter);\n  }\n\n  /**\n   * Add an event listener. Listens to events emitted from the player.\n   *\n   * @param {string} name - Name of the event.\n   * @param {Function} handler - Handler function to add.\n   */\n  addEventListener(name, handler) {\n    this._emitter.on(name, handler);\n  }\n\n  /**\n   * Remove an event listener of the player.\n   *\n   * @param {string} name - Name of the event.\n   * @param {Function} handler - Handler function to remove.\n   */\n  removeEventListener(name, handler) {\n    this._emitter.off(name, handler);\n  }\n\n  /**\n   * Check for mandatory browser compatibility.\n   * Logs detailed browser compatibilities related to for audio playback.\n   *\n   * @throws {Error} If no native wave or MP3 playback is available.\n   */\n  _playbackCompatibility() {\n    // Detect audio playback capabilities.\n\n    // Detect HTML5 Audio playback.\n    // http://caniuse.com/#feat=audio\n    this.canUseAudio = Boolean(Audio);\n    console.log('Native HTML5 Audio playback capability:', this.canUseAudio);\n\n    if (!this.canUseAudio) {\n      throw new Error(\n        'Some form of audio playback capability is required');\n    }\n    if (this.canUseAudio) {\n      const _audio = new Audio();\n      if (!(_audio.canPlayType && _audio.canPlayType instanceof Function)) {\n        throw new Error(\n          'Unable to detect audio playback capabilities');\n      }\n      const canPlayOggVorbis = _audio.canPlayType(\n        'audio/ogg; codecs=\"vorbis\"') !== '';\n      const canPlayOggOpus = _audio.canPlayType(\n        'audio/ogg; codecs=\"opus\"') !== '';\n      const canPlayWave = _audio.canPlayType('audio/wav') !== '';\n      const canPlayMP3 = _audio.canPlayType('audio/mpeg; codecs=\"mp3\"') !== '';\n      const canPlayAAC = _audio.canPlayType(\n        'audio/mp4; codecs=\"mp4a.40.2\"') !== '';\n      const canPlay3GPP = _audio.canPlayType(\n        'audio/3gpp; codecs=\"samr\"') !== '';\n\n      console.log('Native Vorbis audio in Ogg container playback capability:', canPlayOggVorbis);\n      console.log('Native Opus audio in Ogg container playback capability:', canPlayOggOpus);\n      console.log('Native PCM audio in Waveform Audio File Format (WAVE) playback capability:', canPlayWave);\n      console.log('Native MPEG Audio Layer 3 (MP3) playback capability:', canPlayMP3);\n      console.log('Native Low-Complexity AAC audio in MP4 container playback capability:', canPlayAAC);\n      console.log('Native AMR audio in 3GPP container playback capability:', canPlay3GPP);\n\n      if (!(canPlayWave || canPlayMP3)) {\n        throw new Error(\n          'Native Wave or MP3 playback is required');\n      }\n    }\n  }\n\n  /**\n   * Get a player object that performs audio compression, when available.\n   *\n   * Using the Media Stream Recording API for recording is the preferred\n   * solution. It allows recording compressed audio which makes it quicker to\n   * submit. If not available, use a default createScriptProcessor is used.\n   *\n   * @param {Function} callbacks - Callbacks to add to the chosen player.\n   * @private\n   */\n  _getPlayer(callbacks) {\n    let player = null;\n\n    if (this.canUseAudio) {\n      // Use the recorder with MediaRecorder implementation.\n      player = new WebAudioPlayer(callbacks);\n    } else {\n      throw new Error('Unable to find a proper player.');\n    }\n    console.log('Player initialised.');\n    return player;\n  }\n\n  /**\n   * Preload audio from an URL.\n   *\n   * @param {string} url - The URL that contains the audio.\n   * @param {boolean} [preload=true] - Try preloading metadata and possible some audio. Set to false to not download\n   * anything until playing.\n   * @param {?Function} loadedCb - The callback that is invoked when the duration of the audio file\n   * is first known.\n   * @emits {Event} 'canplay' When the player is ready to play.\n   */\n  load(url, preload, loadedCb) {\n    this.reset();\n    this._player.load(url, preload, loadedCb);\n    this._audioLevel = 1;\n\n    // If preloading is disabled, the 'canplay' event won't be triggered.\n    // In that case, fire it manually.\n    if (!preload) {\n      this._emitter.emit('canplay', []);\n    }\n  }\n\n  /**\n   * Unload previously loaded audio. Stops the player and any stopwatch.\n   *\n   * @emits {Event} 'unloaded'\n   */\n  reset() {\n    this.stop();\n    this._player.reset();\n    this._emitter.emit('unloaded', []);\n  }\n\n  /**\n   * Start or continue playback of audio. Also starts the stopwatch at the given position.\n   *\n   * @param {?number} position - When position is given, start playing from this position (seconds).\n   */\n  play(position) {\n    if (this._player.isPlaying()) {\n      return;\n    }\n    this._player.play(position);\n    if (this._stopwatch) {\n      const time = Math.round(this._player.getCurrentTime() * 10);\n      this._stopwatch._value = time;\n      this._stopwatch.start();\n    }\n  }\n\n  /**\n   * Stop playback of audio. Stops and resets the stopwatch.\n   */\n  stop() {\n    if (this._stopwatch) {\n      this._stopwatch.reset();\n      this._stopwatch.stop();\n    }\n    this._player.stop();\n  }\n\n  /**\n   * Pause playback of audio. Stops the stopwatch.\n   */\n  pause() {\n    if (this._stopwatch) {\n      this._stopwatch.stop();\n    }\n    this._player.pause();\n  }\n\n  /**\n   * Toggle audio playback. Switch from playing to paused state and back.\n   */\n  togglePlayback() {\n    if (this._player.isPlaying()) {\n      this.pause();\n    } else {\n      this.play();\n    }\n  }\n\n  /**\n   * Start preloading audio.\n   */\n  preload() {\n    this._player.preload();\n  }\n\n  /**\n   * Start playing audio at the given offset. Corrects a percentage under 0 or above 100 to the respective values.\n   *\n   * @param {number} percentage - Start at this percentage (0..100) of the audio stream.\n   */\n  scrub(percentage) {\n    if (percentage < 0) {\n      this._player.scrub(0);\n    } else if (percentage > 100) {\n      this._player.scrub(100);\n    } else {\n      this._player.scrub(percentage);\n    }\n    if (this._stopwatch) {\n      this._stopwatch._value = Math.round(this._player.getCurrentTime() * 10);\n    }\n  }\n\n  /**\n   * Returns the percentage of which the buffer is filled.\n   *\n   * @returns {number} Percentage of buffer fill.\n   */\n  getBufferFill() {\n    return this._player.getBufferFill();\n  }\n\n  /**\n   * Returns the current playing time as offset in seconds from the start.\n   *\n   * @returns {number} Time in seconds as offset from the start.\n   */\n  getCurrentTime() {\n    return this._player.getCurrentTime();\n  }\n\n  /**\n   * Returns the total duration in seconds.\n   *\n   * @returns {number} Time in seconds of fragment duration.\n   */\n  getDuration() {\n    return this._player.getDuration();\n  }\n\n  /**\n   * Check if there is playback in progress.\n   *\n   * @returns {boolean} True if user is currently playing audio. False otherwise.\n   */\n  isPlaying() {\n    return this._player.isPlaying();\n  }\n\n  /**\n   * Returns ready state of the player.\n   *\n   * @returns {boolean} True when player is ready to start loading data or play. False when no audio is loaded\n   * or the player is preparing.\n   */\n  canPlay() {\n    return this._player.canPlay();\n  }\n\n  /**\n   * Set the playback rate of the audio. Values are used according to HTML5 Audio.\n   * Example values:\n   * *1.0 is normal speed.\n   * *0.5 is half speed (slower).\n   * *2.0 is double speed (faster).\n   * *-1.0 is backwards, normal speed.\n   * *-0.5 is backwards, half speed.\n   *\n   * @param {number} rate - Rate at which to change the audio playback.\n   */\n  setPlaybackRate(rate) {\n    this._player.setPlaybackRate(rate);\n  }\n\n  /**\n   * Get the playback rate of the current loaded audio.\n   *\n   * @returns {number} Playback rate of the audio.\n   */\n  getPlaybackRate() {\n    return this._player.getPlaybackRate();\n  }\n\n  /**\n   * Bind a stopwatch to sync with the playing and stopping functionality of the player.\n   *\n   * @param {Function} tickCb - Callback to invoke on every tick. A tick occurs once every 100 ms.\n   * @throws {Error} If _tickCb is null.\n   * @returns {Stopwatch} New Stopwatch object.\n   */\n  bindStopwatch(tickCb) {\n    this._stopwatch = new Stopwatch(time => {\n      const duration = this.getDuration() * 10 / this._player.sound.playbackRate;\n      if (time > duration) {\n        tickCb(duration);\n      } else {\n        tickCb(time);\n      }\n    });\n    return this._stopwatch;\n  }\n\n  /**\n   * Sets the audio level of the current loaded audio. Valid values are from 0 (0%) to 1 (100%).\n   *\n   * @param {number} volume - Volume value from 0 to 1.\n   */\n  setAudioVolume(volume) {\n    if (volume !== 0) {\n      this._audioMuted = false;\n    }\n    if (volume === 0) {\n      this._audioMuted = true;\n    }\n    this._player.setAudioVolume(volume);\n  }\n\n  /**\n   * Gets the audio level of the current loaded audio. Valid values are from 0 (0%) to 1 (100%).\n   *\n   * @returns {number} Volume level of the current loaded audio.\n   */\n  getAudioVolume() {\n    return this._player.getAudioVolume();\n  }\n\n  /**\n   * Toggle the current playing audio to be muted or not. If the audio will be muted, the current audio level\n   * is remembered and can be unmuted to continue at this same audio level.\n   */\n  toggleAudioMute() {\n    this.setAudioMute(!this._audioMuted);\n  }\n\n  /**\n   * Manually set the muted state of the current loaded audio.\n   *\n   * @param {boolean} shouldMute - Whether the audio should be muted or unmuted.\n   */\n  setAudioMute(shouldMute) {\n    if (shouldMute) {\n      this._audioLevel = this.getAudioVolume();\n      this.setAudioVolume(0);\n    } else {\n      this.setAudioVolume(this._audioLevel);\n    }\n  }\n\n  /**\n   * Return the muted state of the current loaded audio.\n   *\n   * @returns {boolean} The muted state of the current loaded audio.\n   */\n  isAudioMuted() {\n    return this._audioMuted;\n  }\n}\n"]} |
@@ -5,10 +5,2 @@ 'use strict'; | ||
var _cordovaMediaRecorder = require('./cordova-media-recorder'); | ||
var _cordovaMediaRecorder2 = _interopRequireDefault(_cordovaMediaRecorder); | ||
var _mediaRecorder = require('./media-recorder'); | ||
var _mediaRecorder2 = _interopRequireDefault(_mediaRecorder); | ||
var _tools = require('./tools'); | ||
@@ -40,4 +32,5 @@ | ||
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } | ||
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } // import MediaRecorder from './media-recorder'; | ||
/** | ||
@@ -65,3 +58,3 @@ * Audio recording component. | ||
* | ||
* @type {CordovaMediaRecorder|WebAudioRecorder|MediaRecorder} The specific recorder type. | ||
* @type {WebAudioRecorder|MediaRecorder} The specific recorder type. | ||
*/ | ||
@@ -72,8 +65,2 @@ this._recorder = null; | ||
if (this.canUseCordovaMedia) { | ||
// Through the App permissions, access to the microphone was | ||
// already granted. | ||
this.userMediaApproval = true; | ||
this._recorder = this._getBestRecorder(); | ||
} | ||
this._stopwatch = null; | ||
@@ -155,5 +142,8 @@ } | ||
// https://developer.mozilla.org/en-US/docs/Web/API/Navigator.getUserMedia | ||
navigator.getUserMedia = navigator.getUserMedia || navigator.webkitGetUserMedia || navigator.mozGetUserMedia || navigator.msGetUserMedia; | ||
this.canGetUserMedia = Boolean(navigator.getUserMedia); | ||
console.log('Native deprecated navigator.getUserMedia API capability:', this.canGetUserMedia); | ||
// navigator.getUserMedia = navigator.getUserMedia || | ||
// navigator.webkitGetUserMedia || | ||
// navigator.mozGetUserMedia || | ||
// navigator.msGetUserMedia; | ||
// this.canGetUserMedia = Boolean(navigator.getUserMedia); | ||
// console.log('Native deprecated navigator.getUserMedia API capability:', this.canGetUserMedia); | ||
@@ -172,4 +162,4 @@ // https://developer.mozilla.org/en-US/docs/Web/API/MediaDevices/mediaDevices.getUserMedia | ||
// http://www.w3.org/TR/mediastream-recording/ | ||
this.canUseMediaRecorder = Boolean(window.MediaRecorder); | ||
console.log('Native MediaRecorder recording capability:', this.canUseMediaRecorder); | ||
// this.canUseMediaRecorder = Boolean(window.MediaRecorder); | ||
// console.log('Native MediaRecorder recording capability:', this.canUseMediaRecorder); | ||
@@ -183,11 +173,3 @@ // Web Audio API | ||
// Detect Cordova Media Recording | ||
// It allows recording audio using the native bridge inside WebView Apps. | ||
// Note that it may also require native playback when codecs were used for | ||
// recording that are not yet supported in the WebView. | ||
// https://github.com/apache/cordova-plugin-media/blob/master/doc/index.md | ||
this.canUseCordovaMedia = Boolean(window.Media); | ||
console.log('Cordova Media recording capability:', this.canUseCordovaMedia); | ||
if (!(this.canGetUserMedia || this.canUseCordovaMedia)) { | ||
if (!this.canGetUserMedia && !this.canMediaDevicesGetUserMedia) { | ||
throw new Error('Some form of audio recording capability is required'); | ||
@@ -297,14 +279,8 @@ } | ||
var recorder = null; | ||
// Start by checking for a Cordova environment. | ||
// When running under a debugger like Ripple, both Cordova and WebAudio | ||
// environments get detected. While this is technically valid -Ripple is | ||
// running in Chrome, which supports WebAudio-, it's not a sandbox that | ||
// also disables functionality that would not be available on a device. | ||
if (this.canUseCordovaMedia && !this._settings.forceWave) { | ||
// Use Cordova audio encoding (used codec depends on the platform). | ||
recorder = new _cordovaMediaRecorder2.default(); | ||
} else if (this.canUserMediaRecorder && !this._settings.forceWave) { | ||
// Use the recorder with MediaRecorder implementation. | ||
recorder = new _mediaRecorder2.default(micInputGain); | ||
} else if (this.canGetUserMedia) { | ||
// Start by checking for a MediaRecorder. | ||
// if (this.canUserMediaRecorder && !this._settings.forceWave) { | ||
// // Use the recorder with MediaRecorder implementation. | ||
// recorder = new MediaRecorder(micInputGain); | ||
// } else if (this.canGetUserMedia) { | ||
if (this.canMediaDevicesGetUserMedia) { | ||
// Fall back to raw (WAVE) audio encoding. | ||
@@ -480,2 +456,2 @@ var self = this; | ||
exports.default = AudioRecorder; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/audio/audio-recorder.js"],"names":["AudioRecorder","options","_settings","Object","assign","_recordingCompatibility","userMediaApproval","_recorder","_emitter","canUseCordovaMedia","_getBestRecorder","_stopwatch","removeAllEventListeners","addEventListener","name","handler","on","removeEventListener","off","fireEvent","args","emit","hasUserMediaApproval","navigator","getUserMedia","webkitGetUserMedia","mozGetUserMedia","msGetUserMedia","canGetUserMedia","Boolean","console","log","canMediaDevicesGetUserMedia","mediaDevices","canUseMediaRecorder","window","MediaRecorder","AudioContext","webkitAudioContext","mozAudioContext","canCreateAudioContext","Media","Error","URL","webkitURL","hasWindowURL","requestUserMedia","self","success","stream","getAudioTracks","length","micInputGain","_startUserMedia","audioContext","failure","e","audio","then","catch","createMediaStreamSource","micInput","createGain","connect","recorder","forceWave","canUserMediaRecorder","streamCallback","data","chunk","_requireGetUserMedia","startRecordingSession","id","uuid_","undefined","v4","activeRecordingId","record","cb","isRecording","_value","start","stop","forced","getEncodedAudio","blob","type","toggleRecording","getAudioSpecs","bindStopwatch","tickCb"],"mappings":"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","file":"audio-recorder.js","sourcesContent":["import CordovaMediaRecorder from './cordova-media-recorder';\nimport MediaRecorder from './media-recorder';\nimport Stopwatch from './tools';\nimport WavePacker from './wave-packer';\nimport WebAudioRecorder from './web-audio-recorder';\nimport allOff from 'event-emitter/all-off';\nimport ee from 'event-emitter';\nimport uuid from 'uuid';\n\n/**\n * Audio recording component.\n*/\n\nexport default class AudioRecorder {\n  /**\n   * ITSLanguage AudioRecorder.\n   *\n   * @param {?Object} options - Override any of the default settings.\n   *\n   */\n  constructor(options) {\n    this._settings = Object.assign({}, options);\n\n    this._recordingCompatibility();\n\n    this.userMediaApproval = false;\n\n    /**\n     *\n     * @type {CordovaMediaRecorder|WebAudioRecorder|MediaRecorder} The specific recorder type.\n     */\n    this._recorder = null;\n\n    this._emitter = ee({});\n\n\n    if (this.canUseCordovaMedia) {\n      // Through the App permissions, access to the microphone was\n      // already granted.\n      this.userMediaApproval = true;\n      this._recorder = this._getBestRecorder();\n    }\n    this._stopwatch = null;\n  }\n\n  /**\n   * Turn off all event listeners for this recorder.\n   */\n  removeAllEventListeners() {\n    allOff(this._emitter);\n  }\n\n  /**\n   * Add an event listener. Listens to events emitted from the recorder.\n   *\n   * @param {string} name - Name of the event.\n   * @param {Function} handler - Handler function to add.\n   */\n  addEventListener(name, handler) {\n    this._emitter.on(name, handler);\n  }\n\n  /**\n   * Remove an event listener of the recorder.\n   *\n   * @param {string} name - Name of the event.\n   * @param {Function} handler - Handler function to remove.\n   */\n  removeEventListener(name, handler) {\n    this._emitter.off(name, handler);\n  }\n\n  /**\n   * Fire an event.\n   *\n   * @param {string} name - Name of the event.\n   * @param {Object[]} args - Arguments.\n   * @private\n   */\n  fireEvent(name, args = []) {\n    this._emitter.emit(name, ...args);\n  }\n\n  /**\n   * Check if the user has already given permission to access the microphone.\n   *\n   * @returns {boolean} True if user has granted access to the microphone. False otherwise.\n   */\n  hasUserMediaApproval() {\n    return this.userMediaApproval || false;\n  }\n\n  /**\n   * Logs browser compatibility for audio recording.\n   * In case of compatibility issues, an error is thrown.\n   *\n   * @private\n   */\n  _recordingCompatibility/* istanbul ignore next */() {\n    // Detect audio recording capabilities.\n    // http://caniuse.com/#feat=stream\n    // https://developer.mozilla.org/en-US/docs/Web/API/Navigator.getUserMedia\n    navigator.getUserMedia = navigator.getUserMedia ||\n      navigator.webkitGetUserMedia ||\n      navigator.mozGetUserMedia ||\n      navigator.msGetUserMedia;\n    this.canGetUserMedia = Boolean(navigator.getUserMedia);\n    console.log('Native deprecated navigator.getUserMedia API capability:', this.canGetUserMedia);\n\n    // https://developer.mozilla.org/en-US/docs/Web/API/MediaDevices/mediaDevices.getUserMedia\n    this.canMediaDevicesGetUserMedia = false;\n    if (navigator.mediaDevices) {\n      navigator.mediaDevices.getUserMedia = navigator.mediaDevices.getUserMedia ||\n        navigator.mediaDevices.webkitGetUserMedia ||\n        navigator.mediaDevices.mozGetUserMedia;\n      this.canMediaDevicesGetUserMedia = Boolean(navigator.mediaDevices.getUserMedia);\n    }\n    console.log('Native navigator.mediaDevices.getUserMedia API capability:', this.canMediaDevicesGetUserMedia);\n\n    // Detect MediaStream Recording\n    // It allows recording audio using the MediaStream from the above\n    // getUserMedia directly with a native codec better than Wave.\n    // http://www.w3.org/TR/mediastream-recording/\n    this.canUseMediaRecorder = Boolean(window.MediaRecorder);\n    console.log('Native MediaRecorder recording capability:', this.canUseMediaRecorder);\n\n    // Web Audio API\n    // High-level JavaScript API for processing and synthesizing audio\n    // http://caniuse.com/#feat=audio-api\n    window.AudioContext = window.AudioContext ||\n      window.webkitAudioContext || window.mozAudioContext;\n    const canCreateAudioContext = Boolean(window.AudioContext);\n    console.log('Native Web Audio API (AudioContext) processing capability:', canCreateAudioContext);\n\n    // Detect Cordova Media Recording\n    // It allows recording audio using the native bridge inside WebView Apps.\n    // Note that it may also require native playback when codecs were used for\n    // recording that are not yet supported in the WebView.\n    // https://github.com/apache/cordova-plugin-media/blob/master/doc/index.md\n    this.canUseCordovaMedia = Boolean(window.Media);\n    console.log('Cordova Media recording capability:', this.canUseCordovaMedia);\n\n    if (!(this.canGetUserMedia || this.canUseCordovaMedia)) {\n      throw new Error(\n        'Some form of audio recording capability is required');\n    }\n\n    window.URL = window.URL || window.webkitURL;\n    const hasWindowURL = Boolean(window.URL);\n    console.log('Native window.URL capability:', hasWindowURL);\n    if (!hasWindowURL) {\n      throw new Error(\n        'No window.URL blob conversion capabilities');\n    }\n  }\n\n\n  /**\n   * Request microphone access.\n   *\n   * Calling this function may result in thrown exceptions when browser\n   * doesn't support provide live audio input.\n   *\n   * @throws {Error} If no live audio input is available or permitted.\n   */\n  requestUserMedia() {\n    const self = this;\n    function success(stream) {\n      console.log('Got getUserMedia stream');\n\n      // checking audio presence\n      if (self.canMediaDevicesGetUserMedia) {\n        if (stream.getAudioTracks().length) {\n          console.log('Got audio tracks:', stream.getAudioTracks().length);\n        }\n      }\n\n      // Modify state of userMediaApproval now access is granted.\n      self.userMediaApproval = true;\n\n      const micInputGain = self._startUserMedia(stream);\n      self.fireEvent('ready', [self.audioContext, micInputGain]);\n    }\n    function failure(e) {\n      console.log(e);\n      throw new Error('No live audio input available or permitted');\n    }\n\n    if (this.canMediaDevicesGetUserMedia) {\n      // Use of promises is required.\n      navigator.mediaDevices.getUserMedia({audio: true}).then(success).catch(failure);\n    } else if (this.canGetUserMedia) {\n      navigator.getUserMedia({audio: true}, success, failure);\n    }\n  }\n\n  /**\n   * Audio access was granted, start analysing.\n   *\n   * @param {MediaStream} stream - Media Stream.\n   * @private\n   */\n  _startUserMedia(stream) {\n    if (!this.audioContext) {\n      // Initialize the context once, and only when getUserMedia was\n      // successful.\n      this.audioContext = new window.AudioContext();\n    }\n\n    if (!this.audioContext.createMediaStreamSource) {\n      throw new Error('AudioContext has no property createMediaStreamSource');\n    }\n\n    // Creates an audio node from the microphone incoming stream.\n    const micInput = this.audioContext.createMediaStreamSource(stream);\n\n    // This is a workaround for a bug in Firefox that would otherwise lead to\n    // the sound input stopping after ~5 seconds.\n    // https://bugzilla.mozilla.org/show_bug.cgi?id=934512\n    // the important thing is to save a reference to the MediaStreamAudioSourceNode\n    this.micInput = micInput;\n\n    // Create a gain node\n    const micInputGain = this.audioContext.createGain();\n    // Connect the microphone source to a gain node.\n    micInput.connect(micInputGain);\n\n    this._recorder = this._getBestRecorder(micInputGain);\n\n    return micInputGain;\n  }\n\n  /**\n   * Get a recorder object that performs audio compression, when available.\n   *\n   * Using the Media Stream Recording API for recording is the prefered\n   * solution. It allows recording compressed audio which makes it quicker to\n   * submit. If not available, use a default createScriptProcessor is used.\n   *\n   * @param {GainNode} micInputGain - The GainNode to analyze.\n   * @private\n   */\n  _getBestRecorder(micInputGain) {\n    let recorder = null;\n    // Start by checking for a Cordova environment.\n    // When running under a debugger like Ripple, both Cordova and WebAudio\n    // environments get detected. While this is technically valid -Ripple is\n    // running in Chrome, which supports WebAudio-, it's not a sandbox that\n    // also disables functionality that would not be available on a device.\n    if (this.canUseCordovaMedia && !this._settings.forceWave) {\n      // Use Cordova audio encoding (used codec depends on the platform).\n      recorder = new CordovaMediaRecorder();\n    } else if (this.canUserMediaRecorder && !this._settings.forceWave) {\n      // Use the recorder with MediaRecorder implementation.\n      recorder = new MediaRecorder(micInputGain);\n    } else if (this.canGetUserMedia) {\n      // Fall back to raw (WAVE) audio encoding.\n      const self = this;\n      recorder = new WebAudioRecorder(micInputGain, data => {\n        self.streamCallback(data);\n      }, new WavePacker());\n    } else {\n      throw new Error('Unable to find a proper recorder.');\n    }\n    console.log('Recorder initialised.');\n    return recorder;\n  }\n\n  /**\n   * Called when a chunk of audio becomes available.\n   *\n   * @param {ArrayBuffer} chunk - A chunk of audio (Int16 formatted).\n   */\n  streamCallback(chunk) {\n    this.fireEvent('dataavailable', [chunk]);\n  }\n\n  /**\n   * Throw an error if the user is not yet logged in.\n   *\n   * @returns {boolean} True when permission was already granted. False otherwise.\n   */\n  _requireGetUserMedia() {\n    if (this._recorder) {\n      return true;\n    }\n    console.log('Requesting getUserMedia permission first.');\n    this.requestUserMedia();\n    return false;\n  }\n\n  /**\n   * Set a new recording session id.\n   *\n   * @param {number} id - When defined, stick this id to the recorded blob.\n   *\n   * @returns {number} The id that was given or a unique generated one.\n   */\n  startRecordingSession(id) {\n    // Generate a uuid to remember this recording by (locally).\n    const uuid_ = id === undefined ? uuid.v4() : id;\n    this.activeRecordingId = uuid_;\n    return uuid_;\n  }\n\n  /**\n   * Start recording microphone input until stopped.\n   *\n   * @param {?Function} cb - The callback that provides a piece of raw audio when\n   * it becomes available. It may be used for streaming.\n   * @emits {Event} 'recording' With arguments: [recording ID].\n   */\n  record(cb) {\n    if (!this._requireGetUserMedia()) {\n      return;\n    }\n\n    if (this.isRecording()) {\n      throw new Error('Already recording, stop recording first.');\n    }\n\n    this._recorder.record();\n    if (this._stopwatch) {\n      this._stopwatch._value = 0;\n      this._stopwatch.start();\n    }\n\n    if (!this.activeRecordingId) {\n      this.startRecordingSession();\n    }\n    console.log('Recording as id: ' + this.activeRecordingId);\n\n    this.fireEvent('recording', [this.activeRecordingId]);\n    return cb;\n  }\n\n  /**\n   * Stop recording microphone input.\n   *\n   * @param {boolean} [forced=false] - Set whether to force the microphone to stop recording or let it end normally.\n   * @emits {Event} 'recorded' With arguments: [recording ID, audio Blob, forced].\n   */\n  stop(forced) {\n    if (!this._recorder.isRecording()) {\n      return;\n    }\n    this._recorder.stop();\n    if (this._stopwatch) {\n      this._stopwatch.stop();\n    }\n    console.log('Stopped recording for id: ' + this.activeRecordingId);\n\n    const self = this;\n    this._recorder.getEncodedAudio(blob => {\n      console.log('Received encoded audio of type: ' + blob.type);\n      // Allow direct playback from local blob.\n      self.fireEvent('recorded', [self.activeRecordingId, blob, Boolean(forced)]);\n    });\n  }\n\n  /**\n   * Check if there is a recording in progress.\n   *\n   * @returns {boolean} True if user is currently recording audio. False` otherwise.\n   */\n  isRecording() {\n    if (!this._recorder) {\n      return false;\n    }\n    return this._recorder.isRecording();\n  }\n\n  /**\n   * Toggle audio playback. Switch from playing to paused state and back.\n   */\n  toggleRecording() {\n    if (this.isRecording()) {\n      this.stop();\n    } else {\n      this.record();\n    }\n  }\n\n  /**\n   * Get the recorded audio specifications.\n   *\n   * @returns {Object} Containing audioFormat and audioParameters describing the format.\n   */\n  getAudioSpecs() {\n    return this._recorder.getAudioSpecs();\n  }\n\n  /**\n   * Bind a stopwatch to sync with the playing and stopping functionality of the recorder.\n   *\n   * @param {Function} tickCb - Callback to invoke on every tick. A tick occurs once every 100 ms.\n   * @throws {Error} If _tickCb is null.\n   * @returns {Stopwatch} New Stopwatch object.\n   */\n  bindStopwatch(tickCb) {\n    this._stopwatch = new Stopwatch(tickCb);\n    return this._stopwatch;\n  }\n}\n"]} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/audio/audio-recorder.js"],"names":["AudioRecorder","options","_settings","Object","assign","_recordingCompatibility","userMediaApproval","_recorder","_emitter","_stopwatch","removeAllEventListeners","addEventListener","name","handler","on","removeEventListener","off","fireEvent","args","emit","hasUserMediaApproval","canMediaDevicesGetUserMedia","navigator","mediaDevices","getUserMedia","webkitGetUserMedia","mozGetUserMedia","Boolean","console","log","window","AudioContext","webkitAudioContext","mozAudioContext","canCreateAudioContext","canGetUserMedia","Error","URL","webkitURL","hasWindowURL","requestUserMedia","self","success","stream","getAudioTracks","length","micInputGain","_startUserMedia","audioContext","failure","e","audio","then","catch","createMediaStreamSource","micInput","createGain","connect","_getBestRecorder","recorder","streamCallback","data","chunk","_requireGetUserMedia","startRecordingSession","id","uuid_","undefined","v4","activeRecordingId","record","cb","isRecording","_value","start","stop","forced","getEncodedAudio","blob","type","toggleRecording","getAudioSpecs","bindStopwatch","tickCb"],"mappings":";;;;AACA;;;;AACA;;;;AACA;;;;AACA;;;;AACA;;;;AACA;;;;;;0JANA;;;AAQA;;;;IAIqBA,a;AACnB;;;;;;AAMA,yBAAYC,OAAZ,EAAqB;AAAA;;AACnB,SAAKC,SAAL,GAAiBC,OAAOC,MAAP,CAAc,EAAd,EAAkBH,OAAlB,CAAjB;;AAEA,SAAKI,uBAAL;;AAEA,SAAKC,iBAAL,GAAyB,KAAzB;;AAEA;;;;AAIA,SAAKC,SAAL,GAAiB,IAAjB;;AAEA,SAAKC,QAAL,GAAgB,4BAAG,EAAH,CAAhB;;AAEA,SAAKC,UAAL,GAAkB,IAAlB;AACD;;AAED;;;;;0BAGAC,uB,sCAA0B;AACxB,0BAAO,KAAKF,QAAZ;AACD,G;;AAED;;;;;;;;0BAMAG,gB,6BAAiBC,I,EAAMC,O,EAAS;AAC9B,SAAKL,QAAL,CAAcM,EAAd,CAAiBF,IAAjB,EAAuBC,OAAvB;AACD,G;;AAED;;;;;;;;0BAMAE,mB,gCAAoBH,I,EAAMC,O,EAAS;AACjC,SAAKL,QAAL,CAAcQ,GAAd,CAAkBJ,IAAlB,EAAwBC,OAAxB;AACD,G;;AAED;;;;;;;;;0BAOAI,S,sBAAUL,I,EAAiB;AAAA;;AAAA,QAAXM,IAAW,uEAAJ,EAAI;;AACzB,qBAAKV,QAAL,EAAcW,IAAd,kBAAmBP,IAAnB,SAA4BM,IAA5B;AACD,G;;AAED;;;;;;;0BAKAE,oB,mCAAuB;AACrB,WAAO,KAAKd,iBAAL,IAA0B,KAAjC;AACD,G;;AAED;;;;;;;;0BAMAD,uB,CAAuB,0B,sCAA6B;AAClD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,SAAKgB,2BAAL,GAAmC,KAAnC;AACA,QAAIC,UAAUC,YAAd,EAA4B;AAC1BD,gBAAUC,YAAV,CAAuBC,YAAvB,GAAsCF,UAAUC,YAAV,CAAuBC,YAAvB,IACpCF,UAAUC,YAAV,CAAuBE,kBADa,IAEpCH,UAAUC,YAAV,CAAuBG,eAFzB;AAGA,WAAKL,2BAAL,GAAmCM,QAAQL,UAAUC,YAAV,CAAuBC,YAA/B,CAAnC;AACD;AACDI,YAAQC,GAAR,CAAY,4DAAZ,EAA0E,KAAKR,2BAA/E;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACAS,WAAOC,YAAP,GAAsBD,OAAOC,YAAP,IACpBD,OAAOE,kBADa,IACSF,OAAOG,eADtC;AAEA,QAAMC,wBAAwBP,QAAQG,OAAOC,YAAf,CAA9B;AACAH,YAAQC,GAAR,CAAY,4DAAZ,EAA0EK,qBAA1E;;AAEA,QAAI,CAAC,KAAKC,eAAN,IAAyB,CAAC,KAAKd,2BAAnC,EAAgE;AAC9D,YAAM,IAAIe,KAAJ,CACJ,qDADI,CAAN;AAED;;AAEDN,WAAOO,GAAP,GAAaP,OAAOO,GAAP,IAAcP,OAAOQ,SAAlC;AACA,QAAMC,eAAeZ,QAAQG,OAAOO,GAAf,CAArB;AACAT,YAAQC,GAAR,CAAY,+BAAZ,EAA6CU,YAA7C;AACA,QAAI,CAACA,YAAL,EAAmB;AACjB,YAAM,IAAIH,KAAJ,CACJ,4CADI,CAAN;AAED;AACF,G;;AAGD;;;;;;;;;;0BAQAI,gB,+BAAmB;AACjB,QAAMC,OAAO,IAAb;AACA,aAASC,OAAT,CAAiBC,MAAjB,EAAyB;AACvBf,cAAQC,GAAR,CAAY,yBAAZ;;AAEA;AACA,UAAIY,KAAKpB,2BAAT,EAAsC;AACpC,YAAIsB,OAAOC,cAAP,GAAwBC,MAA5B,EAAoC;AAClCjB,kBAAQC,GAAR,CAAY,mBAAZ,EAAiCc,OAAOC,cAAP,GAAwBC,MAAzD;AACD;AACF;;AAED;AACAJ,WAAKnC,iBAAL,GAAyB,IAAzB;;AAEA,UAAMwC,eAAeL,KAAKM,eAAL,CAAqBJ,MAArB,CAArB;AACAF,WAAKxB,SAAL,CAAe,OAAf,EAAwB,CAACwB,KAAKO,YAAN,EAAoBF,YAApB,CAAxB;AACD;AACD,aAASG,OAAT,CAAiBC,CAAjB,EAAoB;AAClBtB,cAAQC,GAAR,CAAYqB,CAAZ;AACA,YAAM,IAAId,KAAJ,CAAU,4CAAV,CAAN;AACD;;AAED,QAAI,KAAKf,2BAAT,EAAsC;AACpC;AACAC,gBAAUC,YAAV,CAAuBC,YAAvB,CAAoC,EAAC2B,OAAO,IAAR,EAApC,EAAmDC,IAAnD,CAAwDV,OAAxD,EAAiEW,KAAjE,CAAuEJ,OAAvE;AACD,KAHD,MAGO,IAAI,KAAKd,eAAT,EAA0B;AAC/Bb,gBAAUE,YAAV,CAAuB,EAAC2B,OAAO,IAAR,EAAvB,EAAsCT,OAAtC,EAA+CO,OAA/C;AACD;AACF,G;;AAED;;;;;;;;0BAMAF,e,4BAAgBJ,M,EAAQ;AACtB,QAAI,CAAC,KAAKK,YAAV,EAAwB;AACtB;AACA;AACA,WAAKA,YAAL,GAAoB,IAAIlB,OAAOC,YAAX,EAApB;AACD;;AAED,QAAI,CAAC,KAAKiB,YAAL,CAAkBM,uBAAvB,EAAgD;AAC9C,YAAM,IAAIlB,KAAJ,CAAU,sDAAV,CAAN;AACD;;AAED;AACA,QAAMmB,WAAW,KAAKP,YAAL,CAAkBM,uBAAlB,CAA0CX,MAA1C,CAAjB;;AAEA;AACA;AACA;AACA;AACA,SAAKY,QAAL,GAAgBA,QAAhB;;AAEA;AACA,QAAMT,eAAe,KAAKE,YAAL,CAAkBQ,UAAlB,EAArB;AACA;AACAD,aAASE,OAAT,CAAiBX,YAAjB;;AAEA,SAAKvC,SAAL,GAAiB,KAAKmD,gBAAL,CAAsBZ,YAAtB,CAAjB;;AAEA,WAAOA,YAAP;AACD,G;;AAED;;;;;;;;;;;;0BAUAY,gB,6BAAiBZ,Y,EAAc;AAC7B,QAAIa,WAAW,IAAf;AACA;AACA;AACA;AACA;AACA;AACA,QAAI,KAAKtC,2BAAT,EAAsC;AACpC;AACA,UAAMoB,OAAO,IAAb;AACAkB,iBAAW,+BAAqBb,YAArB,EAAmC,gBAAQ;AACpDL,aAAKmB,cAAL,CAAoBC,IAApB;AACD,OAFU,EAER,0BAFQ,CAAX;AAGD,KAND,MAMO;AACL,YAAM,IAAIzB,KAAJ,CAAU,mCAAV,CAAN;AACD;AACDR,YAAQC,GAAR,CAAY,uBAAZ;AACA,WAAO8B,QAAP;AACD,G;;AAED;;;;;;;0BAKAC,c,2BAAeE,K,EAAO;AACpB,SAAK7C,SAAL,CAAe,eAAf,EAAgC,CAAC6C,KAAD,CAAhC;AACD,G;;AAED;;;;;;;0BAKAC,oB,mCAAuB;AACrB,QAAI,KAAKxD,SAAT,EAAoB;AAClB,aAAO,IAAP;AACD;AACDqB,YAAQC,GAAR,CAAY,2CAAZ;AACA,SAAKW,gBAAL;AACA,WAAO,KAAP;AACD,G;;AAED;;;;;;;;;0BAOAwB,qB,kCAAsBC,E,EAAI;AACxB;AACA,QAAMC,QAAQD,OAAOE,SAAP,GAAmB,eAAKC,EAAL,EAAnB,GAA+BH,EAA7C;AACA,SAAKI,iBAAL,GAAyBH,KAAzB;AACA,WAAOA,KAAP;AACD,G;;AAED;;;;;;;;;0BAOAI,M,mBAAOC,E,EAAI;AACT,QAAI,CAAC,KAAKR,oBAAL,EAAL,EAAkC;AAChC;AACD;;AAED,QAAI,KAAKS,WAAL,EAAJ,EAAwB;AACtB,YAAM,IAAIpC,KAAJ,CAAU,0CAAV,CAAN;AACD;;AAED,SAAK7B,SAAL,CAAe+D,MAAf;AACA,QAAI,KAAK7D,UAAT,EAAqB;AACnB,WAAKA,UAAL,CAAgBgE,MAAhB,GAAyB,CAAzB;AACA,WAAKhE,UAAL,CAAgBiE,KAAhB;AACD;;AAED,QAAI,CAAC,KAAKL,iBAAV,EAA6B;AAC3B,WAAKL,qBAAL;AACD;AACDpC,YAAQC,GAAR,CAAY,sBAAsB,KAAKwC,iBAAvC;;AAEA,SAAKpD,SAAL,CAAe,WAAf,EAA4B,CAAC,KAAKoD,iBAAN,CAA5B;AACA,WAAOE,EAAP;AACD,G;;AAED;;;;;;;;0BAMAI,I,iBAAKC,M,EAAQ;AACX,QAAI,CAAC,KAAKrE,SAAL,CAAeiE,WAAf,EAAL,EAAmC;AACjC;AACD;AACD,SAAKjE,SAAL,CAAeoE,IAAf;AACA,QAAI,KAAKlE,UAAT,EAAqB;AACnB,WAAKA,UAAL,CAAgBkE,IAAhB;AACD;AACD/C,YAAQC,GAAR,CAAY,+BAA+B,KAAKwC,iBAAhD;;AAEA,QAAM5B,OAAO,IAAb;AACA,SAAKlC,SAAL,CAAesE,eAAf,CAA+B,gBAAQ;AACrCjD,cAAQC,GAAR,CAAY,qCAAqCiD,KAAKC,IAAtD;AACA;AACAtC,WAAKxB,SAAL,CAAe,UAAf,EAA2B,CAACwB,KAAK4B,iBAAN,EAAyBS,IAAzB,EAA+BnD,QAAQiD,MAAR,CAA/B,CAA3B;AACD,KAJD;AAKD,G;;AAED;;;;;;;0BAKAJ,W,0BAAc;AACZ,QAAI,CAAC,KAAKjE,SAAV,EAAqB;AACnB,aAAO,KAAP;AACD;AACD,WAAO,KAAKA,SAAL,CAAeiE,WAAf,EAAP;AACD,G;;AAED;;;;;0BAGAQ,e,8BAAkB;AAChB,QAAI,KAAKR,WAAL,EAAJ,EAAwB;AACtB,WAAKG,IAAL;AACD,KAFD,MAEO;AACL,WAAKL,MAAL;AACD;AACF,G;;AAED;;;;;;;0BAKAW,a,4BAAgB;AACd,WAAO,KAAK1E,SAAL,CAAe0E,aAAf,EAAP;AACD,G;;AAED;;;;;;;;;0BAOAC,a,0BAAcC,M,EAAQ;AACpB,SAAK1E,UAAL,GAAkB,oBAAc0E,MAAd,CAAlB;AACA,WAAO,KAAK1E,UAAZ;AACD,G;;;;;kBAhXkBT,a","file":"audio-recorder.js","sourcesContent":["// import MediaRecorder from './media-recorder';\nimport Stopwatch from './tools';\nimport WavePacker from './wave-packer';\nimport WebAudioRecorder from './web-audio-recorder';\nimport allOff from 'event-emitter/all-off';\nimport ee from 'event-emitter';\nimport uuid from 'uuid';\n\n/**\n * Audio recording component.\n*/\n\nexport default class AudioRecorder {\n  /**\n   * ITSLanguage AudioRecorder.\n   *\n   * @param {?Object} options - Override any of the default settings.\n   *\n   */\n  constructor(options) {\n    this._settings = Object.assign({}, options);\n\n    this._recordingCompatibility();\n\n    this.userMediaApproval = false;\n\n    /**\n     *\n     * @type {WebAudioRecorder|MediaRecorder} The specific recorder type.\n     */\n    this._recorder = null;\n\n    this._emitter = ee({});\n\n    this._stopwatch = null;\n  }\n\n  /**\n   * Turn off all event listeners for this recorder.\n   */\n  removeAllEventListeners() {\n    allOff(this._emitter);\n  }\n\n  /**\n   * Add an event listener. Listens to events emitted from the recorder.\n   *\n   * @param {string} name - Name of the event.\n   * @param {Function} handler - Handler function to add.\n   */\n  addEventListener(name, handler) {\n    this._emitter.on(name, handler);\n  }\n\n  /**\n   * Remove an event listener of the recorder.\n   *\n   * @param {string} name - Name of the event.\n   * @param {Function} handler - Handler function to remove.\n   */\n  removeEventListener(name, handler) {\n    this._emitter.off(name, handler);\n  }\n\n  /**\n   * Fire an event.\n   *\n   * @param {string} name - Name of the event.\n   * @param {Object[]} args - Arguments.\n   * @private\n   */\n  fireEvent(name, args = []) {\n    this._emitter.emit(name, ...args);\n  }\n\n  /**\n   * Check if the user has already given permission to access the microphone.\n   *\n   * @returns {boolean} True if user has granted access to the microphone. False otherwise.\n   */\n  hasUserMediaApproval() {\n    return this.userMediaApproval || false;\n  }\n\n  /**\n   * Logs browser compatibility for audio recording.\n   * In case of compatibility issues, an error is thrown.\n   *\n   * @private\n   */\n  _recordingCompatibility/* istanbul ignore next */() {\n    // Detect audio recording capabilities.\n    // http://caniuse.com/#feat=stream\n    // https://developer.mozilla.org/en-US/docs/Web/API/Navigator.getUserMedia\n    // navigator.getUserMedia = navigator.getUserMedia ||\n    //   navigator.webkitGetUserMedia ||\n    //   navigator.mozGetUserMedia ||\n    //   navigator.msGetUserMedia;\n    // this.canGetUserMedia = Boolean(navigator.getUserMedia);\n    // console.log('Native deprecated navigator.getUserMedia API capability:', this.canGetUserMedia);\n\n    // https://developer.mozilla.org/en-US/docs/Web/API/MediaDevices/mediaDevices.getUserMedia\n    this.canMediaDevicesGetUserMedia = false;\n    if (navigator.mediaDevices) {\n      navigator.mediaDevices.getUserMedia = navigator.mediaDevices.getUserMedia ||\n        navigator.mediaDevices.webkitGetUserMedia ||\n        navigator.mediaDevices.mozGetUserMedia;\n      this.canMediaDevicesGetUserMedia = Boolean(navigator.mediaDevices.getUserMedia);\n    }\n    console.log('Native navigator.mediaDevices.getUserMedia API capability:', this.canMediaDevicesGetUserMedia);\n\n    // Detect MediaStream Recording\n    // It allows recording audio using the MediaStream from the above\n    // getUserMedia directly with a native codec better than Wave.\n    // http://www.w3.org/TR/mediastream-recording/\n    // this.canUseMediaRecorder = Boolean(window.MediaRecorder);\n    // console.log('Native MediaRecorder recording capability:', this.canUseMediaRecorder);\n\n    // Web Audio API\n    // High-level JavaScript API for processing and synthesizing audio\n    // http://caniuse.com/#feat=audio-api\n    window.AudioContext = window.AudioContext ||\n      window.webkitAudioContext || window.mozAudioContext;\n    const canCreateAudioContext = Boolean(window.AudioContext);\n    console.log('Native Web Audio API (AudioContext) processing capability:', canCreateAudioContext);\n\n    if (!this.canGetUserMedia && !this.canMediaDevicesGetUserMedia) {\n      throw new Error(\n        'Some form of audio recording capability is required');\n    }\n\n    window.URL = window.URL || window.webkitURL;\n    const hasWindowURL = Boolean(window.URL);\n    console.log('Native window.URL capability:', hasWindowURL);\n    if (!hasWindowURL) {\n      throw new Error(\n        'No window.URL blob conversion capabilities');\n    }\n  }\n\n\n  /**\n   * Request microphone access.\n   *\n   * Calling this function may result in thrown exceptions when browser\n   * doesn't support provide live audio input.\n   *\n   * @throws {Error} If no live audio input is available or permitted.\n   */\n  requestUserMedia() {\n    const self = this;\n    function success(stream) {\n      console.log('Got getUserMedia stream');\n\n      // checking audio presence\n      if (self.canMediaDevicesGetUserMedia) {\n        if (stream.getAudioTracks().length) {\n          console.log('Got audio tracks:', stream.getAudioTracks().length);\n        }\n      }\n\n      // Modify state of userMediaApproval now access is granted.\n      self.userMediaApproval = true;\n\n      const micInputGain = self._startUserMedia(stream);\n      self.fireEvent('ready', [self.audioContext, micInputGain]);\n    }\n    function failure(e) {\n      console.log(e);\n      throw new Error('No live audio input available or permitted');\n    }\n\n    if (this.canMediaDevicesGetUserMedia) {\n      // Use of promises is required.\n      navigator.mediaDevices.getUserMedia({audio: true}).then(success).catch(failure);\n    } else if (this.canGetUserMedia) {\n      navigator.getUserMedia({audio: true}, success, failure);\n    }\n  }\n\n  /**\n   * Audio access was granted, start analysing.\n   *\n   * @param {MediaStream} stream - Media Stream.\n   * @private\n   */\n  _startUserMedia(stream) {\n    if (!this.audioContext) {\n      // Initialize the context once, and only when getUserMedia was\n      // successful.\n      this.audioContext = new window.AudioContext();\n    }\n\n    if (!this.audioContext.createMediaStreamSource) {\n      throw new Error('AudioContext has no property createMediaStreamSource');\n    }\n\n    // Creates an audio node from the microphone incoming stream.\n    const micInput = this.audioContext.createMediaStreamSource(stream);\n\n    // This is a workaround for a bug in Firefox that would otherwise lead to\n    // the sound input stopping after ~5 seconds.\n    // https://bugzilla.mozilla.org/show_bug.cgi?id=934512\n    // the important thing is to save a reference to the MediaStreamAudioSourceNode\n    this.micInput = micInput;\n\n    // Create a gain node\n    const micInputGain = this.audioContext.createGain();\n    // Connect the microphone source to a gain node.\n    micInput.connect(micInputGain);\n\n    this._recorder = this._getBestRecorder(micInputGain);\n\n    return micInputGain;\n  }\n\n  /**\n   * Get a recorder object that performs audio compression, when available.\n   *\n   * Using the Media Stream Recording API for recording is the prefered\n   * solution. It allows recording compressed audio which makes it quicker to\n   * submit. If not available, use a default createScriptProcessor is used.\n   *\n   * @param {GainNode} micInputGain - The GainNode to analyze.\n   * @private\n   */\n  _getBestRecorder(micInputGain) {\n    let recorder = null;\n    // Start by checking for a MediaRecorder.\n    // if (this.canUserMediaRecorder && !this._settings.forceWave) {\n    //   // Use the recorder with MediaRecorder implementation.\n    //   recorder = new MediaRecorder(micInputGain);\n    // } else if (this.canGetUserMedia) {\n    if (this.canMediaDevicesGetUserMedia) {\n      // Fall back to raw (WAVE) audio encoding.\n      const self = this;\n      recorder = new WebAudioRecorder(micInputGain, data => {\n        self.streamCallback(data);\n      }, new WavePacker());\n    } else {\n      throw new Error('Unable to find a proper recorder.');\n    }\n    console.log('Recorder initialised.');\n    return recorder;\n  }\n\n  /**\n   * Called when a chunk of audio becomes available.\n   *\n   * @param {ArrayBuffer} chunk - A chunk of audio (Int16 formatted).\n   */\n  streamCallback(chunk) {\n    this.fireEvent('dataavailable', [chunk]);\n  }\n\n  /**\n   * Throw an error if the user is not yet logged in.\n   *\n   * @returns {boolean} True when permission was already granted. False otherwise.\n   */\n  _requireGetUserMedia() {\n    if (this._recorder) {\n      return true;\n    }\n    console.log('Requesting getUserMedia permission first.');\n    this.requestUserMedia();\n    return false;\n  }\n\n  /**\n   * Set a new recording session id.\n   *\n   * @param {number} id - When defined, stick this id to the recorded blob.\n   *\n   * @returns {number} The id that was given or a unique generated one.\n   */\n  startRecordingSession(id) {\n    // Generate a uuid to remember this recording by (locally).\n    const uuid_ = id === undefined ? uuid.v4() : id;\n    this.activeRecordingId = uuid_;\n    return uuid_;\n  }\n\n  /**\n   * Start recording microphone input until stopped.\n   *\n   * @param {?Function} cb - The callback that provides a piece of raw audio when\n   * it becomes available. It may be used for streaming.\n   * @emits {Event} 'recording' With arguments: [recording ID].\n   */\n  record(cb) {\n    if (!this._requireGetUserMedia()) {\n      return;\n    }\n\n    if (this.isRecording()) {\n      throw new Error('Already recording, stop recording first.');\n    }\n\n    this._recorder.record();\n    if (this._stopwatch) {\n      this._stopwatch._value = 0;\n      this._stopwatch.start();\n    }\n\n    if (!this.activeRecordingId) {\n      this.startRecordingSession();\n    }\n    console.log('Recording as id: ' + this.activeRecordingId);\n\n    this.fireEvent('recording', [this.activeRecordingId]);\n    return cb;\n  }\n\n  /**\n   * Stop recording microphone input.\n   *\n   * @param {boolean} [forced=false] - Set whether to force the microphone to stop recording or let it end normally.\n   * @emits {Event} 'recorded' With arguments: [recording ID, audio Blob, forced].\n   */\n  stop(forced) {\n    if (!this._recorder.isRecording()) {\n      return;\n    }\n    this._recorder.stop();\n    if (this._stopwatch) {\n      this._stopwatch.stop();\n    }\n    console.log('Stopped recording for id: ' + this.activeRecordingId);\n\n    const self = this;\n    this._recorder.getEncodedAudio(blob => {\n      console.log('Received encoded audio of type: ' + blob.type);\n      // Allow direct playback from local blob.\n      self.fireEvent('recorded', [self.activeRecordingId, blob, Boolean(forced)]);\n    });\n  }\n\n  /**\n   * Check if there is a recording in progress.\n   *\n   * @returns {boolean} True if user is currently recording audio. False` otherwise.\n   */\n  isRecording() {\n    if (!this._recorder) {\n      return false;\n    }\n    return this._recorder.isRecording();\n  }\n\n  /**\n   * Toggle audio playback. Switch from playing to paused state and back.\n   */\n  toggleRecording() {\n    if (this.isRecording()) {\n      this.stop();\n    } else {\n      this.record();\n    }\n  }\n\n  /**\n   * Get the recorded audio specifications.\n   *\n   * @returns {Object} Containing audioFormat and audioParameters describing the format.\n   */\n  getAudioSpecs() {\n    return this._recorder.getAudioSpecs();\n  }\n\n  /**\n   * Bind a stopwatch to sync with the playing and stopping functionality of the recorder.\n   *\n   * @param {Function} tickCb - Callback to invoke on every tick. A tick occurs once every 100 ms.\n   * @throws {Error} If _tickCb is null.\n   * @returns {Stopwatch} New Stopwatch object.\n   */\n  bindStopwatch(tickCb) {\n    this._stopwatch = new Stopwatch(tickCb);\n    return this._stopwatch;\n  }\n}\n"]} |
@@ -117,2 +117,2 @@ 'use strict'; | ||
exports.default = Stopwatch; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9hdWRpby90b29scy5qcyJdLCJuYW1lcyI6WyJlZSIsInJlcXVpcmUiLCJTdG9wd2F0Y2giLCJ0aWNrQ2IiLCJFcnJvciIsIl9pbnRlcnZhbCIsIl92YWx1ZSIsIl90aWNrQ2IiLCJfZW1pdHRlciIsInN0YXJ0IiwiY29uc29sZSIsImRlYnVnIiwic2VsZiIsInNldEludGVydmFsIiwidXBkYXRlIiwic3RvcCIsImNsZWFySW50ZXJ2YWwiLCJ0aWNrIiwicmVzZXQiLCJlbWl0IiwicmVnaXN0ZXJMaXN0ZW5lciIsIm9uIiwic3RvcExpc3RlbmluZyIsIm9mZiJdLCJtYXBwaW5ncyI6Ijs7Ozs7O0FBQUE7Ozs7Ozs7O0FBUUEsSUFBTUEsS0FBS0MsUUFBUSxlQUFSLENBQVg7O0FBRUE7Ozs7O0FBS0E7Ozs7Ozs7OztJQVFxQkMsUztBQUNuQjs7OztBQUlBLHFCQUFZQyxNQUFaLEVBQW9CO0FBQUE7O0FBQ2xCLFFBQUksQ0FBQ0EsTUFBTCxFQUFhO0FBQ1gsWUFBTSxJQUFJQyxLQUFKLENBQVUsMkJBQVYsQ0FBTjtBQUNEO0FBQ0QsU0FBS0MsU0FBTCxHQUFpQixJQUFqQjtBQUNBLFNBQUtDLE1BQUwsR0FBYyxDQUFkO0FBQ0EsU0FBS0MsT0FBTCxHQUFlSixNQUFmO0FBQ0EsU0FBS0ssUUFBTCxHQUFnQlIsR0FBRyxFQUFILENBQWhCO0FBQ0Q7O0FBRUQ7Ozs7O3NCQUdBUyxLLG9CQUFRO0FBQ05DLFlBQVFDLEtBQVIsQ0FBYyxnQkFBZDtBQUNBO0FBQ0EsUUFBTUMsT0FBTyxJQUFiO0FBQ0EsU0FBS1AsU0FBTCxHQUFpQlEsWUFBWSxZQUFNO0FBQ2pDRCxXQUFLRSxNQUFMO0FBQ0QsS0FGZ0IsRUFFZCxHQUZjLENBQWpCO0FBR0QsRzs7QUFFRDs7Ozs7c0JBR0FDLEksbUJBQU87QUFDTEwsWUFBUUMsS0FBUixDQUFjLGVBQWQ7QUFDQUssa0JBQWMsS0FBS1gsU0FBbkI7QUFDQSxTQUFLWSxJQUFMO0FBQ0EsU0FBS1osU0FBTCxHQUFpQixJQUFqQjtBQUNELEc7O0FBRUQ7Ozs7O3NCQUdBYSxLLG9CQUFRO0FBQ05SLFlBQVFDLEtBQVIsQ0FBYyxhQUFkO0FBQ0EsU0FBS0wsTUFBTCxHQUFjLENBQWQ7QUFDQSxTQUFLVyxJQUFMO0FBQ0QsRzs7QUFFRDs7Ozs7c0JBR0FILE0scUJBQVM7QUFDUCxTQUFLRyxJQUFMO0FBQ0EsU0FBS1gsTUFBTDtBQUNELEc7O0FBRUQ7Ozs7O3NCQUdBVyxJLG1CQUFPO0FBQ0wsU0FBS1YsT0FBTCxDQUFhLEtBQUtELE1BQWxCO0FBQ0EsU0FBS0UsUUFBTCxDQUFjVyxJQUFkLENBQW1CLE1BQW5CLEVBQTJCLEtBQUtiLE1BQWhDO0FBQ0QsRzs7c0JBRURjLGdCLDZCQUFpQmpCLE0sRUFBUTtBQUN2QixTQUFLSyxRQUFMLENBQWNhLEVBQWQsQ0FBaUIsTUFBakIsRUFBeUJsQixNQUF6QjtBQUNELEc7O3NCQUVEbUIsYSwwQkFBY25CLE0sRUFBUTtBQUNwQixTQUFLSyxRQUFMLENBQWNlLEdBQWQsQ0FBa0IsTUFBbEIsRUFBMEJwQixNQUExQjtBQUNELEc7Ozs7O2tCQXBFa0JELFMiLCJmaWxlIjoidG9vbHMuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEB0aXRsZSBJVFNMYW5ndWFnZSBKYXZhc2NyaXB0XG4gKiBAb3ZlcnZpZXcgVGhpcyBpcyBwYXJ0IG9mIHRoZSBJVFNMYW5ndWFnZSBKYXZhc2NyaXB0IFNESyB0byBwZXJmb3JtIGhlbHBlciBmdW5jdGlvbnMuXG4gKiBAY29weXJpZ2h0IChjKSAyMDE0IElUU0xhbmd1YWdlXG4gKiBAbGljZW5zZSBNSVRcbiAqIEBhdXRob3IgZC1jZW50cmFsaXplXG4gKi9cblxuY29uc3QgZWUgPSByZXF1aXJlKCdldmVudC1lbWl0dGVyJyk7XG5cbi8qKlxuQG1vZHVsZSBUb29sc1xuSVRTTGFuZ3VhZ2UgaGVscGVyIHRvb2xzLlxuKi9cblxuLyoqXG4gKiBBIHNpbXBsZSBzdG9wd2F0Y2ggdGhhdCB0aWNrcyBldmVyeSAxMDAgbXMuIEl0IGNhbiBiZSBib3VuZCB0byBhbiB7QGxpbmsgQXVkaW9QbGF5ZXJ9IG9yIHtAbGluayBBdWRpb1JlY29yZGVyfVxuICogd2hpY2ggYmluZHMgdGhlIHN0b3AgYW5kIHBsYXlpbmcgZnVuY3Rpb25hbGl0eSB0byB0aGUgc3RvcHBpbmcgYW5kIHN0YXJ0aW5nIG9mIHRoZSBXYXRjaC5cbiAqIEl0IGNhbiBhbHNvIGJlIGxpc3RlbmVkIHRvIGJ5IG90aGVyIGVudGl0aWVzLlxuICogQGV4cGVyaW1lbnRhbCBXaGVuIGJpbmRpbmcgdG8gYW4gQXVkaW8gUmVjb3JkZXIsIHRoZSBzdG9wd2F0Y2ggaXMgbm90IGFsd2F5cyBzeW5jZWQgcHJvcGVybHkuXG4gKiBBIGRpZmZlcmVuY2Ugb2YgMC4xcyBvciAwLjJzIHRvbyBoaWdoIG1heSBvY2N1ciB3aGVuIGNvdW50aW5nLiBXaGVuIGJpbmRpbmcgdG8gYW4gQXVkaW8gUGxheWVyIGhvd2V2ZXIsIHRoZSB0aW1lclxuICogd2lsbCBzeW5jIHByb3Blcmx5IGFuZCBzaG93IHRoZSBjb3JyZWN0IGR1cmF0aW9uIG9mIHRoZSBsb2FkZWQgYXVkaW8gZmlsZS5cbiAqL1xuZXhwb3J0IGRlZmF1bHQgY2xhc3MgU3RvcHdhdGNoIHtcbiAgLyoqXG4gICAqIEBwYXJhbSB7RnVuY3Rpb259IHRpY2tDYiAtIFRoZSBjYWxsYmFjayB0aGF0IGlzIGludm9rZWQgb24gZXZlcnkgdGljayAoZXZlcnkgMTAwbXMpLlxuICAgKiBAdGhyb3dzIHtFcnJvcn0gSWYgdGlja0NiIGlzIG1pc3NpbmcuXG4gICAqL1xuICBjb25zdHJ1Y3Rvcih0aWNrQ2IpIHtcbiAgICBpZiAoIXRpY2tDYikge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKCd0aWNrQ2IgcGFyYW1ldGVyIHJlcXVpcmVkJyk7XG4gICAgfVxuICAgIHRoaXMuX2ludGVydmFsID0gbnVsbDtcbiAgICB0aGlzLl92YWx1ZSA9IDA7XG4gICAgdGhpcy5fdGlja0NiID0gdGlja0NiO1xuICAgIHRoaXMuX2VtaXR0ZXIgPSBlZSh7fSk7XG4gIH1cblxuICAvKipcbiAgICogU3RhcnQgY291bnRpbmcgYW5kIHRpY2sgZXZlcnkgMTAwIG1zLlxuICAgKi9cbiAgc3RhcnQoKSB7XG4gICAgY29uc29sZS5kZWJ1ZygnU3RhcnQgY291bnRpbmcnKTtcbiAgICAvLyBUaWNrIGV2ZXJ5IDEwMG1zICgwLjFzKVxuICAgIGNvbnN0IHNlbGYgPSB0aGlzO1xuICAgIHRoaXMuX2ludGVydmFsID0gc2V0SW50ZXJ2YWwoKCkgPT4ge1xuICAgICAgc2VsZi51cGRhdGUoKTtcbiAgICB9LCAxMDApO1xuICB9XG5cbiAgLyoqXG4gICAqIFN0b3AgY291bnRpbmcuXG4gICAqL1xuICBzdG9wKCkge1xuICAgIGNvbnNvbGUuZGVidWcoJ1N0b3AgY291bnRpbmcnKTtcbiAgICBjbGVhckludGVydmFsKHRoaXMuX2ludGVydmFsKTtcbiAgICB0aGlzLnRpY2soKTtcbiAgICB0aGlzLl9pbnRlcnZhbCA9IG51bGw7XG4gIH1cblxuICAvKipcbiAgICogUmVzZXQgY291bnQgdG8gMC5cbiAgICovXG4gIHJlc2V0KCkge1xuICAgIGNvbnNvbGUuZGVidWcoJ1Jlc2V0IGNvdW50Jyk7XG4gICAgdGhpcy5fdmFsdWUgPSAwO1xuICAgIHRoaXMudGljaygpO1xuICB9XG5cbiAgLyoqXG4gICAqIFRpY2sgb25jZSBhbmQgaW5jcmVtZW50IHRoZSB2YWx1ZSBieSAxLlxuICAgKi9cbiAgdXBkYXRlKCkge1xuICAgIHRoaXMudGljaygpO1xuICAgIHRoaXMuX3ZhbHVlKys7XG4gIH1cblxuICAvKipcbiAgICogSW52b2tlIHRoZSB0aWNrIGNhbGxiYWNrIHdpdGggdGhlIGN1cnJlbnQgdmFsdWUuXG4gICAqL1xuICB0aWNrKCkge1xuICAgIHRoaXMuX3RpY2tDYih0aGlzLl92YWx1ZSk7XG4gICAgdGhpcy5fZW1pdHRlci5lbWl0KCd0aWNrJywgdGhpcy5fdmFsdWUpO1xuICB9XG5cbiAgcmVnaXN0ZXJMaXN0ZW5lcih0aWNrQ2IpIHtcbiAgICB0aGlzLl9lbWl0dGVyLm9uKCd0aWNrJywgdGlja0NiKTtcbiAgfVxuXG4gIHN0b3BMaXN0ZW5pbmcodGlja0NiKSB7XG4gICAgdGhpcy5fZW1pdHRlci5vZmYoJ3RpY2snLCB0aWNrQ2IpO1xuICB9XG5cbn1cbiJdfQ== | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9hdWRpby90b29scy5qcyJdLCJuYW1lcyI6WyJlZSIsInJlcXVpcmUiLCJTdG9wd2F0Y2giLCJ0aWNrQ2IiLCJFcnJvciIsIl9pbnRlcnZhbCIsIl92YWx1ZSIsIl90aWNrQ2IiLCJfZW1pdHRlciIsInN0YXJ0IiwiY29uc29sZSIsImRlYnVnIiwic2VsZiIsInNldEludGVydmFsIiwidXBkYXRlIiwic3RvcCIsImNsZWFySW50ZXJ2YWwiLCJ0aWNrIiwicmVzZXQiLCJlbWl0IiwicmVnaXN0ZXJMaXN0ZW5lciIsIm9uIiwic3RvcExpc3RlbmluZyIsIm9mZiJdLCJtYXBwaW5ncyI6Ijs7Ozs7O0FBQUE7Ozs7Ozs7O0FBUUEsSUFBTUEsS0FBS0MsUUFBUSxlQUFSLENBQVg7O0FBRUE7Ozs7O0FBS0E7Ozs7Ozs7OztJQVFxQkMsUztBQUNuQjs7OztBQUlBLHFCQUFZQyxNQUFaLEVBQW9CO0FBQUE7O0FBQ2xCLFFBQUksQ0FBQ0EsTUFBTCxFQUFhO0FBQ1gsWUFBTSxJQUFJQyxLQUFKLENBQVUsMkJBQVYsQ0FBTjtBQUNEO0FBQ0QsU0FBS0MsU0FBTCxHQUFpQixJQUFqQjtBQUNBLFNBQUtDLE1BQUwsR0FBYyxDQUFkO0FBQ0EsU0FBS0MsT0FBTCxHQUFlSixNQUFmO0FBQ0EsU0FBS0ssUUFBTCxHQUFnQlIsR0FBRyxFQUFILENBQWhCO0FBQ0Q7O0FBRUQ7Ozs7O3NCQUdBUyxLLG9CQUFRO0FBQ05DLFlBQVFDLEtBQVIsQ0FBYyxnQkFBZDtBQUNBO0FBQ0EsUUFBTUMsT0FBTyxJQUFiO0FBQ0EsU0FBS1AsU0FBTCxHQUFpQlEsWUFBWSxZQUFNO0FBQ2pDRCxXQUFLRSxNQUFMO0FBQ0QsS0FGZ0IsRUFFZCxHQUZjLENBQWpCO0FBR0QsRzs7QUFFRDs7Ozs7c0JBR0FDLEksbUJBQU87QUFDTEwsWUFBUUMsS0FBUixDQUFjLGVBQWQ7QUFDQUssa0JBQWMsS0FBS1gsU0FBbkI7QUFDQSxTQUFLWSxJQUFMO0FBQ0EsU0FBS1osU0FBTCxHQUFpQixJQUFqQjtBQUNELEc7O0FBRUQ7Ozs7O3NCQUdBYSxLLG9CQUFRO0FBQ05SLFlBQVFDLEtBQVIsQ0FBYyxhQUFkO0FBQ0EsU0FBS0wsTUFBTCxHQUFjLENBQWQ7QUFDQSxTQUFLVyxJQUFMO0FBQ0QsRzs7QUFFRDs7Ozs7c0JBR0FILE0scUJBQVM7QUFDUCxTQUFLRyxJQUFMO0FBQ0EsU0FBS1gsTUFBTDtBQUNELEc7O0FBRUQ7Ozs7O3NCQUdBVyxJLG1CQUFPO0FBQ0wsU0FBS1YsT0FBTCxDQUFhLEtBQUtELE1BQWxCO0FBQ0EsU0FBS0UsUUFBTCxDQUFjVyxJQUFkLENBQW1CLE1BQW5CLEVBQTJCLEtBQUtiLE1BQWhDO0FBQ0QsRzs7c0JBRURjLGdCLDZCQUFpQmpCLE0sRUFBUTtBQUN2QixTQUFLSyxRQUFMLENBQWNhLEVBQWQsQ0FBaUIsTUFBakIsRUFBeUJsQixNQUF6QjtBQUNELEc7O3NCQUVEbUIsYSwwQkFBY25CLE0sRUFBUTtBQUNwQixTQUFLSyxRQUFMLENBQWNlLEdBQWQsQ0FBa0IsTUFBbEIsRUFBMEJwQixNQUExQjtBQUNELEc7Ozs7O2tCQXBFa0JELFMiLCJmaWxlIjoidG9vbHMuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEB0aXRsZSBJVFNMYW5ndWFnZSBKYXZhc2NyaXB0XG4gKiBAb3ZlcnZpZXcgVGhpcyBpcyBwYXJ0IG9mIHRoZSBJVFNMYW5ndWFnZSBKYXZhc2NyaXB0IFNESyB0byBwZXJmb3JtIGhlbHBlciBmdW5jdGlvbnMuXG4gKiBAY29weXJpZ2h0IChjKSAyMDE0IElUU0xhbmd1YWdlXG4gKiBAbGljZW5zZSBNSVRcbiAqIEBhdXRob3IgZC1jZW50cmFsaXplXG4gKi9cblxuY29uc3QgZWUgPSByZXF1aXJlKCdldmVudC1lbWl0dGVyJyk7XG5cbi8qKlxuQG1vZHVsZSBUb29sc1xuSVRTTGFuZ3VhZ2UgaGVscGVyIHRvb2xzLlxuKi9cblxuLyoqXG4gKiBBIHNpbXBsZSBzdG9wd2F0Y2ggdGhhdCB0aWNrcyBldmVyeSAxMDAgbXMuIEl0IGNhbiBiZSBib3VuZCB0byBhbiB7QGxpbmsgQXVkaW9QbGF5ZXJ9IG9yIHtAbGluayBBdWRpb1JlY29yZGVyfVxuICogd2hpY2ggYmluZHMgdGhlIHN0b3AgYW5kIHBsYXlpbmcgZnVuY3Rpb25hbGl0eSB0byB0aGUgc3RvcHBpbmcgYW5kIHN0YXJ0aW5nIG9mIHRoZSBXYXRjaC5cbiAqIEl0IGNhbiBhbHNvIGJlIGxpc3RlbmVkIHRvIGJ5IG90aGVyIGVudGl0aWVzLlxuICogQGV4cGVyaW1lbnRhbCBXaGVuIGJpbmRpbmcgdG8gYW4gQXVkaW8gUmVjb3JkZXIsIHRoZSBzdG9wd2F0Y2ggaXMgbm90IGFsd2F5cyBzeW5jZWQgcHJvcGVybHkuXG4gKiBBIGRpZmZlcmVuY2Ugb2YgMC4xcyBvciAwLjJzIHRvbyBoaWdoIG1heSBvY2N1ciB3aGVuIGNvdW50aW5nLiBXaGVuIGJpbmRpbmcgdG8gYW4gQXVkaW8gUGxheWVyIGhvd2V2ZXIsIHRoZSB0aW1lclxuICogd2lsbCBzeW5jIHByb3Blcmx5IGFuZCBzaG93IHRoZSBjb3JyZWN0IGR1cmF0aW9uIG9mIHRoZSBsb2FkZWQgYXVkaW8gZmlsZS5cbiAqL1xuZXhwb3J0IGRlZmF1bHQgY2xhc3MgU3RvcHdhdGNoIHtcbiAgLyoqXG4gICAqIEBwYXJhbSB7RnVuY3Rpb259IHRpY2tDYiAtIFRoZSBjYWxsYmFjayB0aGF0IGlzIGludm9rZWQgb24gZXZlcnkgdGljayAoZXZlcnkgMTAwbXMpLlxuICAgKiBAdGhyb3dzIHtFcnJvcn0gSWYgdGlja0NiIGlzIG1pc3NpbmcuXG4gICAqL1xuICBjb25zdHJ1Y3Rvcih0aWNrQ2IpIHtcbiAgICBpZiAoIXRpY2tDYikge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKCd0aWNrQ2IgcGFyYW1ldGVyIHJlcXVpcmVkJyk7XG4gICAgfVxuICAgIHRoaXMuX2ludGVydmFsID0gbnVsbDtcbiAgICB0aGlzLl92YWx1ZSA9IDA7XG4gICAgdGhpcy5fdGlja0NiID0gdGlja0NiO1xuICAgIHRoaXMuX2VtaXR0ZXIgPSBlZSh7fSk7XG4gIH1cblxuICAvKipcbiAgICogU3RhcnQgY291bnRpbmcgYW5kIHRpY2sgZXZlcnkgMTAwIG1zLlxuICAgKi9cbiAgc3RhcnQoKSB7XG4gICAgY29uc29sZS5kZWJ1ZygnU3RhcnQgY291bnRpbmcnKTtcbiAgICAvLyBUaWNrIGV2ZXJ5IDEwMG1zICgwLjFzKVxuICAgIGNvbnN0IHNlbGYgPSB0aGlzO1xuICAgIHRoaXMuX2ludGVydmFsID0gc2V0SW50ZXJ2YWwoKCkgPT4ge1xuICAgICAgc2VsZi51cGRhdGUoKTtcbiAgICB9LCAxMDApO1xuICB9XG5cbiAgLyoqXG4gICAqIFN0b3AgY291bnRpbmcuXG4gICAqL1xuICBzdG9wKCkge1xuICAgIGNvbnNvbGUuZGVidWcoJ1N0b3AgY291bnRpbmcnKTtcbiAgICBjbGVhckludGVydmFsKHRoaXMuX2ludGVydmFsKTtcbiAgICB0aGlzLnRpY2soKTtcbiAgICB0aGlzLl9pbnRlcnZhbCA9IG51bGw7XG4gIH1cblxuICAvKipcbiAgICogUmVzZXQgY291bnQgdG8gMC5cbiAgICovXG4gIHJlc2V0KCkge1xuICAgIGNvbnNvbGUuZGVidWcoJ1Jlc2V0IGNvdW50Jyk7XG4gICAgdGhpcy5fdmFsdWUgPSAwO1xuICAgIHRoaXMudGljaygpO1xuICB9XG5cbiAgLyoqXG4gICAqIFRpY2sgb25jZSBhbmQgaW5jcmVtZW50IHRoZSB2YWx1ZSBieSAxLlxuICAgKi9cbiAgdXBkYXRlKCkge1xuICAgIHRoaXMudGljaygpO1xuICAgIHRoaXMuX3ZhbHVlKys7XG4gIH1cblxuICAvKipcbiAgICogSW52b2tlIHRoZSB0aWNrIGNhbGxiYWNrIHdpdGggdGhlIGN1cnJlbnQgdmFsdWUuXG4gICAqL1xuICB0aWNrKCkge1xuICAgIHRoaXMuX3RpY2tDYih0aGlzLl92YWx1ZSk7XG4gICAgdGhpcy5fZW1pdHRlci5lbWl0KCd0aWNrJywgdGhpcy5fdmFsdWUpO1xuICB9XG5cbiAgcmVnaXN0ZXJMaXN0ZW5lcih0aWNrQ2IpIHtcbiAgICB0aGlzLl9lbWl0dGVyLm9uKCd0aWNrJywgdGlja0NiKTtcbiAgfVxuXG4gIHN0b3BMaXN0ZW5pbmcodGlja0NiKSB7XG4gICAgdGhpcy5fZW1pdHRlci5vZmYoJ3RpY2snLCB0aWNrQ2IpO1xuICB9XG59XG4iXX0= |
@@ -357,2 +357,2 @@ 'use strict'; | ||
exports.default = WebAudioPlayer; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/audio/web-audio-player.js"],"names":["WebAudioPlayer","options","_settings","Object","assign","_initPlayer","sound","window","Audio","_pauseIsStop","self","addEventListener","playingCb","timeupdateCb","durationchangeCb","canplayCb","endedCb","pauseCb","stoppedCb","playbackStoppedCb","progressCb","e","target","error","code","MEDIA_ERR_ABORTED","console","MEDIA_ERR_NETWORK","MEDIA_ERR_DECODE","MEDIA_ERR_SRC_NOT_SUPPORTED","errorCb","load","url","preload","loadedCb","undefined","autobuffer","Boolean","log","duration","src","play","position","readyState","HAVE_METADATA","warn","debug","currentTime","reset","stop","pause","info","scrub","percentage","newTime","getBufferFill","buffered","length","probableEnd","i","start","end","loaded","Math","round","getCurrentTime","getDuration","isPlaying","paused","setPlaybackRate","rate","playbackRate","getPlaybackRate","canPlay","setAudioVolume","volume","getAudioVolume"],"mappings":"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","file":"web-audio-player.js","sourcesContent":["/**\n * @title ITSLanguage Javascript Audio\n * @overview This is part of the ITSLanguage Javascript SDK to perform audio related functions.\n * @copyright (c) 2014 ITSLanguage\n * @license MIT\n * @author d-centralize\n *\n * This class fires the same events as the HTML5 Audio does. {@link http://www.w3schools.com/tags/ref_av_dom.asp}\n * @private\n */\n\nexport default class WebAudioPlayer {\n  /**\n   * ITSLanguage WebAudioPlayer non-graphical component.\n   *\n   * This player uses the HTML5 Audio component for playback.\n   *\n   * @param {?Object} options - Override any of the default settings.\n   */\n  constructor(options) {\n    this._settings = Object.assign({}, options);\n\n    this._initPlayer();\n  }\n\n  _initPlayer() {\n    this.sound = new window.Audio();\n    this._pauseIsStop = false;\n\n    // The its.AudioPlayer API is based upon the same API calls as the\n    // HTML5 Audio element itself, therefore, just bubble up all events.\n    const self = this;\n\n    this.sound.addEventListener('playing', () => {\n      if (self._settings.playingCb) {\n        self._settings.playingCb();\n      }\n    });\n\n    this.sound.addEventListener('timeupdate', () => {\n      if (self._settings.timeupdateCb) {\n        self._settings.timeupdateCb();\n      }\n    });\n\n    this.sound.addEventListener('durationchange', () => {\n      if (self._settings.durationchangeCb) {\n        self._settings.durationchangeCb();\n      }\n    });\n\n    this.sound.addEventListener('canplay', () => {\n      if (self._settings.canplayCb) {\n        self._settings.canplayCb();\n      }\n    });\n\n    this.sound.addEventListener('ended', () => {\n      if (self._settings.endedCb) {\n        self._settings.endedCb();\n      }\n    });\n\n    this.sound.addEventListener('pause', () => {\n      // The HTML5 audio player only has a pause(), no stop().\n      // To differentiate between the two, a flag is set in case the user\n      // explicitly stopped (not paused) the audio.\n      if (self._pauseIsStop === true) {\n        self._pauseIsStop = false;\n        if (self._settings.pauseCb) {\n          self._settings.pauseCb();\n        }\n      } else if (self._settings.stoppedCb) {\n        self._settings.stoppedCb();\n      }\n      if (self._settings.playbackStoppedCb) {\n        self._settings.playbackStoppedCb();\n      }\n    });\n\n    this.sound.addEventListener('progress', () => {\n      if (self._settings.progressCb) {\n        self._settings.progressCb();\n      }\n    });\n\n    this.sound.addEventListener('error', e => {\n      switch (e.target.error.code) {\n        case e.target.error.MEDIA_ERR_ABORTED:\n          console.error('You aborted the playback.');\n          break;\n        case e.target.error.MEDIA_ERR_NETWORK:\n          console.error(\n          'A network error caused the audio download to fail.');\n          break;\n        case e.target.error.MEDIA_ERR_DECODE:\n          console.error(\n          'The audio playback was aborted due to a corruption ' +\n          'problem or because the media used features your ' +\n          'browser did not support.');\n          break;\n        case e.target.error.MEDIA_ERR_SRC_NOT_SUPPORTED:\n          console.error(\n          'The audio could not be loaded, either because the ' +\n          'server or network failed or because the format is ' +\n          'not supported.');\n          break;\n        default:\n          console.error('An unknown error occurred.');\n          break;\n      }\n      if (self._settings.errorCb) {\n        self._settings.errorCb();\n      }\n    });\n  }\n\n\n  /**\n   * Preload audio from an URL.\n   *\n   * @param {string} url - The URL that contains the audio.\n   * @param {boolean} [preload=true] - Try preloading metadata and possible some audio (default).\n   * Set to false to not download anything until playing.\n   * @param {?Function} loadedCb - The callback that is invoked when the duration of the audio file\n   * is first known.\n   */\n  load(url, preload, loadedCb) {\n    preload = preload === undefined ? true : preload;\n\n    // Automatically begin buffering the file, even if autoplay is off.\n    this.sound.autobuffer = Boolean(preload);\n\n    // Preloading options:\n    // none - Do not preload any media.\n    // Wait for a play event before downloading anything.\n    // metadata - Preload just the metadata. Grab the start and the end of\n    // the file via range-request and determine the duration.\n    // auto - Preload the whole file. Grab the start and the end of the\n    // file to determine duration, then seek back to the start\n    // again for the preload proper.\n    this.sound.preload = preload ? 'auto' : 'none';\n\n    const self = this;\n    if (loadedCb) {\n      this.sound.addEventListener('durationchange', () => {\n        console.log('Duration change for ' + url + ' to : ' +\n          self.sound.duration);\n        loadedCb(self.sound);\n      });\n    }\n\n    this.sound.src = url;\n  }\n\n\n  /**\n   * Start or continue playback of audio.\n   *\n   * @param {?number} position - When position is given, start playing from this position (seconds).\n   */\n  play(position) {\n    if (position !== undefined) {\n      if (this.sound.readyState < this.sound.HAVE_METADATA) {\n        // In case the audio wasn't already preloaded, do it now.\n        this.sound.preload = 'auto';\n        console.warn('Playing from a given position is not possible. ' +\n          'Audio was not yet loaded. Try again.');\n      } else {\n        console.debug('Scrub position to: ' + position);\n        this.sound.currentTime = position;\n      }\n    }\n    this.sound.play();\n    console.debug('Start playing from position: ' + this.sound.currentTime);\n  }\n\n  /**\n   * Unload previously loaded audio.\n   */\n  reset() {\n    this._initPlayer();\n  }\n\n  /**\n   * Stop playback of audio.\n   */\n  stop() {\n    // The HTML5 audio player only has a pause(), no stop().\n    // To differentiate between the two, set a flag.\n    this.sound.pause();\n    this.sound.currentTime = 0;\n  }\n\n  /**\n   * Pause playback of audio.\n   */\n  pause() {\n    this._pauseIsStop = true;\n    this.sound.pause();\n  }\n\n  /**\n   * Start preloading audio.\n   */\n  preload() {\n    // In case the audio wasn't already preloaded, do it now.\n    if (this.sound.preload !== 'auto') {\n      console.info('Start preloading audio.');\n      this.sound.preload = 'auto';\n    }\n  }\n\n\n  /**\n   * Start playing audio at the given offset.\n   *\n   * @param {number} percentage - Start at this percentage (0..100) of the audio stream.\n   */\n  scrub(percentage) {\n    // In case the audio wasn't already preloaded, do it now.\n    if (this.sound.readyState < this.sound.HAVE_METADATA) {\n      this.preload();\n      console.warn('Scrubbing not possible. Audio was not yet loaded. ' +\n        'Try again.');\n      return;\n    }\n\n    const newTime = this.sound.duration / 100 * percentage;\n    console.log('Moving audio position to: ' + percentage + '%: ' +\n      newTime + 's of total playing time: ' + this.sound.duration);\n    this.sound.currentTime = newTime;\n  }\n\n  /**\n   * Returns the percentage of which the buffer is filled.\n   *\n   * @returns {number} Percentage of buffer fill.\n   */\n  getBufferFill() {\n    if (this.sound.buffered === undefined ||\n      this.sound.buffered.length === 0) {\n      // Nothing buffered yet.\n      return 0;\n    }\n\n    // The fact that there's not one buffer segment is ignored here.\n    // Truely representing the buffered state requires multiple\n    // loading bars.\n    // Usually, when user didn't seek yet, there are two segments:\n    // Got segment from: 0 to: 187.63999938964844\n    // Got segment from: 222.44700622558594 to: 228.1140899658203\n    // The latter is gained when the HTML5 audio component tries to find\n    // the total audio duration.\n    // More info:\n    // http://html5doctor.com/html5-audio-the-state-of-play/#time-ranges\n    let probableEnd = 0;\n    for (let i = 0; i < this.sound.buffered.length; i++) {\n      const start = this.sound.buffered.start(i);\n      const end = this.sound.buffered.end(i);\n      // console.log('Got segment from: ' + start + ' to: ' + end);\n      // Often, the segment that starts from 0 keeps growing and\n      // indicates -most likely- the biggest buffer.\n      if (start === 0) {\n        probableEnd = end;\n      }\n    }\n\n    // Round up,so the buffer won't get stuck on 99% when\n    // duration and buffer are equal, except for some far decimal.\n    const loaded = Math.round(probableEnd * 100 / this.sound.duration);\n    console.log('Buffer filled to ' + loaded + '%');\n    return loaded;\n  }\n\n  /**\n   * Returns the current playing time as offset in seconds from the start.\n   *\n   * @returns {number} Time in seconds as offset from the start.\n   */\n  getCurrentTime() {\n    return this.sound.currentTime;\n  }\n\n  /**\n   * Returns the total duration in seconds.\n   *\n   * @returns {number} Time in seconds of fragment duration. 0 if no audio is loaded.\n   */\n  getDuration() {\n    let duration = this.sound.duration;\n    // When no audio is loaded, the duration may be NaN\n    if (!duration) {\n      duration = 0;\n    }\n    return duration;\n  }\n\n  /**\n   * Returns state of the player.\n   *\n   * @returns {boolean} True when player is currently playing. False when paused or stopped.\n   */\n  isPlaying() {\n    return !this.sound.paused;\n  }\n\n  setPlaybackRate(rate) {\n    this.sound.playbackRate = rate;\n  }\n\n  getPlaybackRate() {\n    return this.sound.playbackRate;\n  }\n\n  /**\n   * Returns ready state of the player.\n   *\n   * @returns {boolean} True when player is ready to start loading data or play. False when no audio is loaded\n   * or preparing.\n   */\n  canPlay() {\n    // Either the player is in a valid readyState (preloaded), or\n    // the player has a source attached and doesn't show any loading error (non-preloaded).\n    return this.sound.readyState >= this.sound.HAVE_METADATA ||\n      this.sound.src && !this.sound.error;\n  }\n\n  setAudioVolume(volume) {\n    this.sound.volume = volume;\n  }\n\n  getAudioVolume() {\n    return this.sound.volume;\n  }\n}\n"]} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/audio/web-audio-player.js"],"names":["WebAudioPlayer","options","_settings","Object","assign","_initPlayer","sound","window","Audio","_pauseIsStop","self","addEventListener","playingCb","timeupdateCb","durationchangeCb","canplayCb","endedCb","pauseCb","stoppedCb","playbackStoppedCb","progressCb","e","target","error","code","MEDIA_ERR_ABORTED","console","MEDIA_ERR_NETWORK","MEDIA_ERR_DECODE","MEDIA_ERR_SRC_NOT_SUPPORTED","errorCb","load","url","preload","loadedCb","undefined","autobuffer","Boolean","log","duration","src","play","position","readyState","HAVE_METADATA","warn","debug","currentTime","reset","stop","pause","info","scrub","percentage","newTime","getBufferFill","buffered","length","probableEnd","i","start","end","loaded","Math","round","getCurrentTime","getDuration","isPlaying","paused","setPlaybackRate","rate","playbackRate","getPlaybackRate","canPlay","setAudioVolume","volume","getAudioVolume"],"mappings":"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","file":"web-audio-player.js","sourcesContent":["/**\n * @title ITSLanguage Javascript Audio\n * @overview This is part of the ITSLanguage Javascript SDK to perform audio related functions.\n * @copyright (c) 2014 ITSLanguage\n * @license MIT\n * @author d-centralize\n *\n * This class fires the same events as the HTML5 Audio does. {@link http://www.w3schools.com/tags/ref_av_dom.asp}\n * @private\n */\n\nexport default class WebAudioPlayer {\n  /**\n   * ITSLanguage WebAudioPlayer non-graphical component.\n   *\n   * This player uses the HTML5 Audio component for playback.\n   *\n   * @param {?Object} options - Override any of the default settings.\n   */\n  constructor(options) {\n    this._settings = Object.assign({}, options);\n\n    this._initPlayer();\n  }\n\n  _initPlayer() {\n    this.sound = new window.Audio();\n    this._pauseIsStop = false;\n\n    // The its.AudioPlayer API is based upon the same API calls as the\n    // HTML5 Audio element itself, therefore, just bubble up all events.\n    const self = this;\n\n    this.sound.addEventListener('playing', () => {\n      if (self._settings.playingCb) {\n        self._settings.playingCb();\n      }\n    });\n\n    this.sound.addEventListener('timeupdate', () => {\n      if (self._settings.timeupdateCb) {\n        self._settings.timeupdateCb();\n      }\n    });\n\n    this.sound.addEventListener('durationchange', () => {\n      if (self._settings.durationchangeCb) {\n        self._settings.durationchangeCb();\n      }\n    });\n\n    this.sound.addEventListener('canplay', () => {\n      if (self._settings.canplayCb) {\n        self._settings.canplayCb();\n      }\n    });\n\n    this.sound.addEventListener('ended', () => {\n      if (self._settings.endedCb) {\n        self._settings.endedCb();\n      }\n    });\n\n    this.sound.addEventListener('pause', () => {\n      // The HTML5 audio player only has a pause(), no stop().\n      // To differentiate between the two, a flag is set in case the user\n      // explicitly stopped (not paused) the audio.\n      if (self._pauseIsStop === true) {\n        self._pauseIsStop = false;\n        if (self._settings.pauseCb) {\n          self._settings.pauseCb();\n        }\n      } else if (self._settings.stoppedCb) {\n        self._settings.stoppedCb();\n      }\n      if (self._settings.playbackStoppedCb) {\n        self._settings.playbackStoppedCb();\n      }\n    });\n\n    this.sound.addEventListener('progress', () => {\n      if (self._settings.progressCb) {\n        self._settings.progressCb();\n      }\n    });\n\n    this.sound.addEventListener('error', e => {\n      switch (e.target.error.code) {\n        case e.target.error.MEDIA_ERR_ABORTED:\n          console.error('You aborted the playback.');\n          break;\n        case e.target.error.MEDIA_ERR_NETWORK:\n          console.error(\n            'A network error caused the audio download to fail.');\n          break;\n        case e.target.error.MEDIA_ERR_DECODE:\n          console.error(\n            'The audio playback was aborted due to a corruption ' +\n          'problem or because the media used features your ' +\n          'browser did not support.');\n          break;\n        case e.target.error.MEDIA_ERR_SRC_NOT_SUPPORTED:\n          console.error(\n            'The audio could not be loaded, either because the ' +\n          'server or network failed or because the format is ' +\n          'not supported.');\n          break;\n        default:\n          console.error('An unknown error occurred.');\n          break;\n      }\n      if (self._settings.errorCb) {\n        self._settings.errorCb();\n      }\n    });\n  }\n\n\n  /**\n   * Preload audio from an URL.\n   *\n   * @param {string} url - The URL that contains the audio.\n   * @param {boolean} [preload=true] - Try preloading metadata and possible some audio (default).\n   * Set to false to not download anything until playing.\n   * @param {?Function} loadedCb - The callback that is invoked when the duration of the audio file\n   * is first known.\n   */\n  load(url, preload, loadedCb) {\n    preload = preload === undefined ? true : preload;\n\n    // Automatically begin buffering the file, even if autoplay is off.\n    this.sound.autobuffer = Boolean(preload);\n\n    // Preloading options:\n    // none - Do not preload any media.\n    // Wait for a play event before downloading anything.\n    // metadata - Preload just the metadata. Grab the start and the end of\n    // the file via range-request and determine the duration.\n    // auto - Preload the whole file. Grab the start and the end of the\n    // file to determine duration, then seek back to the start\n    // again for the preload proper.\n    this.sound.preload = preload ? 'auto' : 'none';\n\n    const self = this;\n    if (loadedCb) {\n      this.sound.addEventListener('durationchange', () => {\n        console.log('Duration change for ' + url + ' to : ' +\n          self.sound.duration);\n        loadedCb(self.sound);\n      });\n    }\n\n    this.sound.src = url;\n  }\n\n\n  /**\n   * Start or continue playback of audio.\n   *\n   * @param {?number} position - When position is given, start playing from this position (seconds).\n   */\n  play(position) {\n    if (position !== undefined) {\n      if (this.sound.readyState < this.sound.HAVE_METADATA) {\n        // In case the audio wasn't already preloaded, do it now.\n        this.sound.preload = 'auto';\n        console.warn('Playing from a given position is not possible. ' +\n          'Audio was not yet loaded. Try again.');\n      } else {\n        console.debug('Scrub position to: ' + position);\n        this.sound.currentTime = position;\n      }\n    }\n    this.sound.play();\n    console.debug('Start playing from position: ' + this.sound.currentTime);\n  }\n\n  /**\n   * Unload previously loaded audio.\n   */\n  reset() {\n    this._initPlayer();\n  }\n\n  /**\n   * Stop playback of audio.\n   */\n  stop() {\n    // The HTML5 audio player only has a pause(), no stop().\n    // To differentiate between the two, set a flag.\n    this.sound.pause();\n    this.sound.currentTime = 0;\n  }\n\n  /**\n   * Pause playback of audio.\n   */\n  pause() {\n    this._pauseIsStop = true;\n    this.sound.pause();\n  }\n\n  /**\n   * Start preloading audio.\n   */\n  preload() {\n    // In case the audio wasn't already preloaded, do it now.\n    if (this.sound.preload !== 'auto') {\n      console.info('Start preloading audio.');\n      this.sound.preload = 'auto';\n    }\n  }\n\n\n  /**\n   * Start playing audio at the given offset.\n   *\n   * @param {number} percentage - Start at this percentage (0..100) of the audio stream.\n   */\n  scrub(percentage) {\n    // In case the audio wasn't already preloaded, do it now.\n    if (this.sound.readyState < this.sound.HAVE_METADATA) {\n      this.preload();\n      console.warn('Scrubbing not possible. Audio was not yet loaded. ' +\n        'Try again.');\n      return;\n    }\n\n    const newTime = this.sound.duration / 100 * percentage;\n    console.log('Moving audio position to: ' + percentage + '%: ' +\n      newTime + 's of total playing time: ' + this.sound.duration);\n    this.sound.currentTime = newTime;\n  }\n\n  /**\n   * Returns the percentage of which the buffer is filled.\n   *\n   * @returns {number} Percentage of buffer fill.\n   */\n  getBufferFill() {\n    if (this.sound.buffered === undefined ||\n      this.sound.buffered.length === 0) {\n      // Nothing buffered yet.\n      return 0;\n    }\n\n    // The fact that there's not one buffer segment is ignored here.\n    // Truely representing the buffered state requires multiple\n    // loading bars.\n    // Usually, when user didn't seek yet, there are two segments:\n    // Got segment from: 0 to: 187.63999938964844\n    // Got segment from: 222.44700622558594 to: 228.1140899658203\n    // The latter is gained when the HTML5 audio component tries to find\n    // the total audio duration.\n    // More info:\n    // http://html5doctor.com/html5-audio-the-state-of-play/#time-ranges\n    let probableEnd = 0;\n    for (let i = 0; i < this.sound.buffered.length; i++) {\n      const start = this.sound.buffered.start(i);\n      const end = this.sound.buffered.end(i);\n      // console.log('Got segment from: ' + start + ' to: ' + end);\n      // Often, the segment that starts from 0 keeps growing and\n      // indicates -most likely- the biggest buffer.\n      if (start === 0) {\n        probableEnd = end;\n      }\n    }\n\n    // Round up,so the buffer won't get stuck on 99% when\n    // duration and buffer are equal, except for some far decimal.\n    const loaded = Math.round(probableEnd * 100 / this.sound.duration);\n    console.log('Buffer filled to ' + loaded + '%');\n    return loaded;\n  }\n\n  /**\n   * Returns the current playing time as offset in seconds from the start.\n   *\n   * @returns {number} Time in seconds as offset from the start.\n   */\n  getCurrentTime() {\n    return this.sound.currentTime;\n  }\n\n  /**\n   * Returns the total duration in seconds.\n   *\n   * @returns {number} Time in seconds of fragment duration. 0 if no audio is loaded.\n   */\n  getDuration() {\n    let duration = this.sound.duration;\n    // When no audio is loaded, the duration may be NaN\n    if (!duration) {\n      duration = 0;\n    }\n    return duration;\n  }\n\n  /**\n   * Returns state of the player.\n   *\n   * @returns {boolean} True when player is currently playing. False when paused or stopped.\n   */\n  isPlaying() {\n    return !this.sound.paused;\n  }\n\n  setPlaybackRate(rate) {\n    this.sound.playbackRate = rate;\n  }\n\n  getPlaybackRate() {\n    return this.sound.playbackRate;\n  }\n\n  /**\n   * Returns ready state of the player.\n   *\n   * @returns {boolean} True when player is ready to start loading data or play. False when no audio is loaded\n   * or preparing.\n   */\n  canPlay() {\n    // Either the player is in a valid readyState (preloaded), or\n    // the player has a source attached and doesn't show any loading error (non-preloaded).\n    return this.sound.readyState >= this.sound.HAVE_METADATA ||\n      this.sound.src && !this.sound.error;\n  }\n\n  setAudioVolume(volume) {\n    this.sound.volume = volume;\n  }\n\n  getAudioVolume() {\n    return this.sound.volume;\n  }\n}\n"]} |
@@ -147,2 +147,2 @@ 'use strict'; | ||
exports.default = WebAudioRecorder; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/audio/web-audio-recorder.js"],"names":["WebAudioRecorder","source","streamingCallback","packer","recording","context","recordedSampleRate","sampleRate","channels","console","log","getAudioSpecs","audioParameters","init","bufferSize","recorder","createScriptProcessor","_recorder","self","onaudioprocess","e","left","inputBuffer","getChannelData","right","leftClone","Float32Array","rightClone","record","recordStreaming","connect","destination","audioFormat","sampleWidth","frameRate","clear","getEncodedAudio","callback","exportWAV","isRecording","stop"],"mappings":";;;;;;AAAA;;;;IAIqBA,gB;AACnB;;;;;;;;;;AAUA,4BAAYC,MAAZ,EAAoBC,iBAApB,EAAuCC,MAAvC,EAA+C;AAAA;;AAC7C,SAAKC,SAAL,GAAiB,KAAjB;;AAEA,QAAMC,UAAUJ,OAAOI,OAAvB;AACA;AACA,SAAKC,kBAAL,GAA0BD,QAAQE,UAAlC;AACA;AACA;AACA;AACA,SAAKA,UAAL,GAAkB,KAAKD,kBAAL,GAA0B,CAA5C;AACA;AACA,SAAKC,UAAL,GAAkBL,oBAAoB,KAAKI,kBAAzB,GACE,KAAKC,UADzB;;AAGA;AACA,SAAKC,QAAL,GAAgB,CAAhB;AACAC,YAAQC,GAAR,CAAY,mBACE,KAAKC,aAAL,GAAqBC,eAArB,CAAqCL,UADnD;;AAGA,SAAKJ,MAAL,GAAcA,MAAd;AACA,SAAKA,MAAL,CAAYU,IAAZ,CAAiB,KAAKP,kBAAtB,EAA0C,KAAKC,UAA/C,EAA2D,KAAKC,QAAhE;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,QAAMM,aAAa,IAAnB;AACA,QAAMC,WAAWV,QAAQW,qBAAR,CAA8BF,UAA9B,EAA0C,CAA1C,EAA6C,CAA7C,CAAjB;AACA;AACA;AACA;AACA;AACA,SAAKG,SAAL,GAAiBF,QAAjB;;AAEA,QAAMG,OAAO,IAAb;AACAH,aAASI,cAAT,GAA0B,UAASC,CAAT,EAAY;AACpC,UAAI,CAACF,KAAKd,SAAV,EAAqB;AACnB;AACD;AACD,UAAMiB,OAAOD,EAAEE,WAAF,CAAcC,cAAd,CAA6B,CAA7B,CAAb;AACA,UAAMC,QAAQJ,EAAEE,WAAF,CAAcC,cAAd,CAA6B,CAA7B,CAAd;AACA;AACA;AACA;AACA,UAAME,YAAY,IAAIC,YAAJ,CAAiBL,IAAjB,CAAlB;AACA,UAAMM,aAAa,IAAID,YAAJ,CAAiBF,KAAjB,CAAnB;AACAN,WAAKf,MAAL,CAAYyB,MAAZ,CAAmBH,SAAnB,EAA8BE,UAA9B;AACA,UAAIzB,iBAAJ,EAAuB;AACrBgB,aAAKf,MAAL,CAAY0B,eAAZ,CAA4BJ,SAA5B,EAAuCE,UAAvC,EAAmDzB,iBAAnD;AACD;AACF,KAfD;;AAiBAD,WAAO6B,OAAP,CAAef,QAAf;AACA;AACA;AACAA,aAASe,OAAT,CAAiBzB,QAAQ0B,WAAzB;AACD;;AAED;;;;;;;6BAKApB,a,4BAAgB;AACd,WAAO;AACLqB,mBAAa,YADR;AAELpB,uBAAiB;AACfJ,kBAAU,KAAKA,QADA;AAEfyB,qBAAa,EAFE;AAGfC,mBAAW,KAAK3B,UAHD;AAIfA,oBAAY,KAAKA;AAJF;AAFZ,KAAP;AASD,G;;AAED;;;;;;6BAIAqB,M,qBAAS;AACP,SAAKzB,MAAL,CAAYgC,KAAZ;AACA,SAAK/B,SAAL,GAAiB,IAAjB;AACD,G;;AAED;;;;;;;6BAKAgC,e,4BAAgBC,Q,EAAU;AACxB,SAAKlC,MAAL,CAAYmC,SAAZ,CAAsBD,QAAtB;AACD,G;;AAED;;;;;;;6BAKAE,W,0BAAc;AACZ,WAAO,KAAKnC,SAAZ;AACD,G;;AAED;;;;;6BAGAoC,I,mBAAO;AACL,SAAKpC,SAAL,GAAiB,KAAjB;AACD,G;;;;;kBAxHkBJ,gB","file":"web-audio-recorder.js","sourcesContent":["/**\n * WebAudioRecorder.\n * @private\n */\nexport default class WebAudioRecorder {\n  /**\n   * Use 'low level' processing tooling to record audio and get a Wave\n   * (audio/wav) encoded recording.\n   *\n   * Currently supported in all modern HTML5/WebAudio browsers.\n   *\n   * @param {GainNode} source - The source to record.\n   * @param {?Function} streamingCallback - The callback to deliver audio chunks to.\n   * @param {WavePacker} packer - Packer to use.\n   */\n  constructor(source, streamingCallback, packer) {\n    this.recording = false;\n\n    const context = source.context;\n    // For the best quality, use the samplerate in which audio is recorded.\n    this.recordedSampleRate = context.sampleRate;\n    // var sampleRate = recordedSampleRate;\n    // 48000hz -> 24000hz recording, 44100hz -> 22050hz recording.\n    // Sheffield determined the minimum to be 16000hz, so /4 is too low.\n    this.sampleRate = this.recordedSampleRate / 2;\n    // Streaming doesn't yet downsample: #1302.\n    this.sampleRate = streamingCallback ? this.recordedSampleRate :\n                        this.sampleRate;\n\n    // Always record audio in mono.\n    this.channels = 1;\n    console.log('Recording at: ' +\n                  this.getAudioSpecs().audioParameters.sampleRate);\n\n    this.packer = packer;\n    this.packer.init(this.recordedSampleRate, this.sampleRate, this.channels);\n\n    // From the spec: This value controls how frequently the audioprocess\n    // event is dispatched and how many sample-frames need to be processed\n    // each call. Lower values for buffer size will result in a lower\n    // (better) latency. Higher values will be necessary to avoid audio\n    // breakup and glitches.\n    // Legal values are (256, 512, 1024, 2048, 4096, 8192, 16384).\n    const bufferSize = 8192;\n    const recorder = context.createScriptProcessor(bufferSize, 2, 2);\n    // Keep a reference to the scriptProcessor.\n    // This is a workaround for a bug in Chrome that would otherwise lead to\n    // the recorder being garbage collected before it even recorded anything.\n    // https://bugs.webkit.org/show_bug.cgi?id=112521\n    this._recorder = recorder;\n\n    const self = this;\n    recorder.onaudioprocess = function(e) {\n      if (!self.recording) {\n        return;\n      }\n      const left = e.inputBuffer.getChannelData(0);\n      const right = e.inputBuffer.getChannelData(1);\n      // These returned channel buffers are pointers to the current samples\n      // coming in. Make a snapshot (clone). The webworkers can't serialize\n      // the pointers. Well, Chrome and FF could, but Edge can't.\n      const leftClone = new Float32Array(left);\n      const rightClone = new Float32Array(right);\n      self.packer.record(leftClone, rightClone);\n      if (streamingCallback) {\n        self.packer.recordStreaming(leftClone, rightClone, streamingCallback);\n      }\n    };\n\n    source.connect(recorder);\n    // If the script node is not connected to an output the \"onaudioprocess\"\n    // event is not triggered in chrome.\n    recorder.connect(context.destination);\n  }\n\n  /**\n   * Get the recorded audio specifications.\n   *\n   * @returns {Object} Containing metadata on the audio format.\n   */\n  getAudioSpecs() {\n    return {\n      audioFormat: 'audio/wave',\n      audioParameters: {\n        channels: this.channels,\n        sampleWidth: 16,\n        frameRate: this.sampleRate,\n        sampleRate: this.sampleRate\n      }\n    };\n  }\n\n  /**\n   * Start recording audio.\n   *\n   */\n  record() {\n    this.packer.clear();\n    this.recording = true;\n  }\n\n  /**\n   * Request encoded audio to be returned through callback.\n   *\n   * @param {Function} callback - The callback to use when returning the audio as a blob in Wave format.\n   */\n  getEncodedAudio(callback) {\n    this.packer.exportWAV(callback);\n  }\n\n  /**\n   * Is audio recording in progress.\n   *\n   * @returns {boolean} True when recording. False otherwise.\n   */\n  isRecording() {\n    return this.recording;\n  }\n\n  /**\n   * Stop recording audio.\n   */\n  stop() {\n    this.recording = false;\n  }\n}\n"]} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/audio/web-audio-recorder.js"],"names":["WebAudioRecorder","source","streamingCallback","packer","recording","context","recordedSampleRate","sampleRate","channels","console","log","getAudioSpecs","audioParameters","init","bufferSize","recorder","createScriptProcessor","_recorder","self","onaudioprocess","e","left","inputBuffer","getChannelData","right","leftClone","Float32Array","rightClone","record","recordStreaming","connect","destination","audioFormat","sampleWidth","frameRate","clear","getEncodedAudio","callback","exportWAV","isRecording","stop"],"mappings":";;;;;;AAAA;;;;IAIqBA,gB;AACnB;;;;;;;;;;AAUA,4BAAYC,MAAZ,EAAoBC,iBAApB,EAAuCC,MAAvC,EAA+C;AAAA;;AAC7C,SAAKC,SAAL,GAAiB,KAAjB;;AAEA,QAAMC,UAAUJ,OAAOI,OAAvB;AACA;AACA,SAAKC,kBAAL,GAA0BD,QAAQE,UAAlC;AACA;AACA;AACA;AACA,SAAKA,UAAL,GAAkB,KAAKD,kBAAL,GAA0B,CAA5C;AACA;AACA,SAAKC,UAAL,GAAkBL,oBAAoB,KAAKI,kBAAzB,GAChB,KAAKC,UADP;;AAGA;AACA,SAAKC,QAAL,GAAgB,CAAhB;AACAC,YAAQC,GAAR,CAAY,mBACE,KAAKC,aAAL,GAAqBC,eAArB,CAAqCL,UADnD;;AAGA,SAAKJ,MAAL,GAAcA,MAAd;AACA,SAAKA,MAAL,CAAYU,IAAZ,CAAiB,KAAKP,kBAAtB,EAA0C,KAAKC,UAA/C,EAA2D,KAAKC,QAAhE;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,QAAMM,aAAa,IAAnB;AACA,QAAMC,WAAWV,QAAQW,qBAAR,CAA8BF,UAA9B,EAA0C,CAA1C,EAA6C,CAA7C,CAAjB;AACA;AACA;AACA;AACA;AACA,SAAKG,SAAL,GAAiBF,QAAjB;;AAEA,QAAMG,OAAO,IAAb;AACAH,aAASI,cAAT,GAA0B,UAASC,CAAT,EAAY;AACpC,UAAI,CAACF,KAAKd,SAAV,EAAqB;AACnB;AACD;AACD,UAAMiB,OAAOD,EAAEE,WAAF,CAAcC,cAAd,CAA6B,CAA7B,CAAb;AACA,UAAMC,QAAQJ,EAAEE,WAAF,CAAcC,cAAd,CAA6B,CAA7B,CAAd;AACA;AACA;AACA;AACA,UAAME,YAAY,IAAIC,YAAJ,CAAiBL,IAAjB,CAAlB;AACA,UAAMM,aAAa,IAAID,YAAJ,CAAiBF,KAAjB,CAAnB;AACAN,WAAKf,MAAL,CAAYyB,MAAZ,CAAmBH,SAAnB,EAA8BE,UAA9B;AACA,UAAIzB,iBAAJ,EAAuB;AACrBgB,aAAKf,MAAL,CAAY0B,eAAZ,CAA4BJ,SAA5B,EAAuCE,UAAvC,EAAmDzB,iBAAnD;AACD;AACF,KAfD;;AAiBAD,WAAO6B,OAAP,CAAef,QAAf;AACA;AACA;AACAA,aAASe,OAAT,CAAiBzB,QAAQ0B,WAAzB;AACD;;AAED;;;;;;;6BAKApB,a,4BAAgB;AACd,WAAO;AACLqB,mBAAa,YADR;AAELpB,uBAAiB;AACfJ,kBAAU,KAAKA,QADA;AAEfyB,qBAAa,EAFE;AAGfC,mBAAW,KAAK3B,UAHD;AAIfA,oBAAY,KAAKA;AAJF;AAFZ,KAAP;AASD,G;;AAED;;;;;;6BAIAqB,M,qBAAS;AACP,SAAKzB,MAAL,CAAYgC,KAAZ;AACA,SAAK/B,SAAL,GAAiB,IAAjB;AACD,G;;AAED;;;;;;;6BAKAgC,e,4BAAgBC,Q,EAAU;AACxB,SAAKlC,MAAL,CAAYmC,SAAZ,CAAsBD,QAAtB;AACD,G;;AAED;;;;;;;6BAKAE,W,0BAAc;AACZ,WAAO,KAAKnC,SAAZ;AACD,G;;AAED;;;;;6BAGAoC,I,mBAAO;AACL,SAAKpC,SAAL,GAAiB,KAAjB;AACD,G;;;;;kBAxHkBJ,gB","file":"web-audio-recorder.js","sourcesContent":["/**\n * WebAudioRecorder.\n * @private\n */\nexport default class WebAudioRecorder {\n  /**\n   * Use 'low level' processing tooling to record audio and get a Wave\n   * (audio/wav) encoded recording.\n   *\n   * Currently supported in all modern HTML5/WebAudio browsers.\n   *\n   * @param {GainNode} source - The source to record.\n   * @param {?Function} streamingCallback - The callback to deliver audio chunks to.\n   * @param {WavePacker} packer - Packer to use.\n   */\n  constructor(source, streamingCallback, packer) {\n    this.recording = false;\n\n    const context = source.context;\n    // For the best quality, use the samplerate in which audio is recorded.\n    this.recordedSampleRate = context.sampleRate;\n    // var sampleRate = recordedSampleRate;\n    // 48000hz -> 24000hz recording, 44100hz -> 22050hz recording.\n    // Sheffield determined the minimum to be 16000hz, so /4 is too low.\n    this.sampleRate = this.recordedSampleRate / 2;\n    // Streaming doesn't yet downsample: #1302.\n    this.sampleRate = streamingCallback ? this.recordedSampleRate :\n      this.sampleRate;\n\n    // Always record audio in mono.\n    this.channels = 1;\n    console.log('Recording at: ' +\n                  this.getAudioSpecs().audioParameters.sampleRate);\n\n    this.packer = packer;\n    this.packer.init(this.recordedSampleRate, this.sampleRate, this.channels);\n\n    // From the spec: This value controls how frequently the audioprocess\n    // event is dispatched and how many sample-frames need to be processed\n    // each call. Lower values for buffer size will result in a lower\n    // (better) latency. Higher values will be necessary to avoid audio\n    // breakup and glitches.\n    // Legal values are (256, 512, 1024, 2048, 4096, 8192, 16384).\n    const bufferSize = 8192;\n    const recorder = context.createScriptProcessor(bufferSize, 2, 2);\n    // Keep a reference to the scriptProcessor.\n    // This is a workaround for a bug in Chrome that would otherwise lead to\n    // the recorder being garbage collected before it even recorded anything.\n    // https://bugs.webkit.org/show_bug.cgi?id=112521\n    this._recorder = recorder;\n\n    const self = this;\n    recorder.onaudioprocess = function(e) {\n      if (!self.recording) {\n        return;\n      }\n      const left = e.inputBuffer.getChannelData(0);\n      const right = e.inputBuffer.getChannelData(1);\n      // These returned channel buffers are pointers to the current samples\n      // coming in. Make a snapshot (clone). The webworkers can't serialize\n      // the pointers. Well, Chrome and FF could, but Edge can't.\n      const leftClone = new Float32Array(left);\n      const rightClone = new Float32Array(right);\n      self.packer.record(leftClone, rightClone);\n      if (streamingCallback) {\n        self.packer.recordStreaming(leftClone, rightClone, streamingCallback);\n      }\n    };\n\n    source.connect(recorder);\n    // If the script node is not connected to an output the \"onaudioprocess\"\n    // event is not triggered in chrome.\n    recorder.connect(context.destination);\n  }\n\n  /**\n   * Get the recorded audio specifications.\n   *\n   * @returns {Object} Containing metadata on the audio format.\n   */\n  getAudioSpecs() {\n    return {\n      audioFormat: 'audio/wave',\n      audioParameters: {\n        channels: this.channels,\n        sampleWidth: 16,\n        frameRate: this.sampleRate,\n        sampleRate: this.sampleRate\n      }\n    };\n  }\n\n  /**\n   * Start recording audio.\n   *\n   */\n  record() {\n    this.packer.clear();\n    this.recording = true;\n  }\n\n  /**\n   * Request encoded audio to be returned through callback.\n   *\n   * @param {Function} callback - The callback to use when returning the audio as a blob in Wave format.\n   */\n  getEncodedAudio(callback) {\n    this.packer.exportWAV(callback);\n  }\n\n  /**\n   * Is audio recording in progress.\n   *\n   * @returns {boolean} True when recording. False otherwise.\n   */\n  isRecording() {\n    return this.recording;\n  }\n\n  /**\n   * Stop recording audio.\n   */\n  stop() {\n    this.recording = false;\n  }\n}\n"]} |
14
index.js
'use strict'; | ||
exports.__esModule = true; | ||
exports.VolumeMeter = exports.Stopwatch = exports.generateWaveSample = exports.AudioRecorder = exports.AudioPlayer = exports.User = exports.SpeechChallenge = exports.Role = exports.PronunciationChallenge = exports.Progress = exports.Profile = exports.Organisation = exports.Group = exports.EmailCredentials = exports.Connection = exports.ChoiceChallenge = exports.Category = exports.BasicAuth = exports.AdministrativeSDK = undefined; | ||
exports.Recorder = exports.Player = exports.VolumeMeter = exports.Stopwatch = exports.generateWaveSample = exports.AudioRecorder = exports.AudioPlayer = exports.User = exports.SpeechChallenge = exports.Role = exports.PronunciationChallenge = exports.Progress = exports.Profile = exports.Organisation = exports.Group = exports.EmailCredentials = exports.Connection = exports.ChoiceChallenge = exports.Category = exports.BasicAuth = exports.AdministrativeSDK = undefined; | ||
@@ -50,2 +50,6 @@ var _audioTools = require('./audio/audio-tools'); | ||
var _Player = require('./WebAudio/Player'); | ||
var _Player2 = _interopRequireDefault(_Player); | ||
var _profile = require('./administrative-sdk/profile/profile'); | ||
@@ -63,2 +67,6 @@ | ||
var _Recorder = require('./WebAudio/Recorder'); | ||
var _Recorder2 = _interopRequireDefault(_Recorder); | ||
var _role = require('./administrative-sdk/role/role'); | ||
@@ -101,2 +109,4 @@ | ||
exports.VolumeMeter = _audioTools2.default; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9pbmRleC5qcyJdLCJuYW1lcyI6WyJBZG1pbmlzdHJhdGl2ZVNESyIsIkJhc2ljQXV0aCIsIkNhdGVnb3J5IiwiQ2hvaWNlQ2hhbGxlbmdlIiwiQ29ubmVjdGlvbiIsIkVtYWlsQ3JlZGVudGlhbHMiLCJHcm91cCIsIk9yZ2FuaXNhdGlvbiIsIlByb2ZpbGUiLCJQcm9ncmVzcyIsIlByb251bmNpYXRpb25DaGFsbGVuZ2UiLCJSb2xlIiwiU3BlZWNoQ2hhbGxlbmdlIiwiVXNlciIsIkF1ZGlvUGxheWVyIiwiQXVkaW9SZWNvcmRlciIsImdlbmVyYXRlV2F2ZVNhbXBsZSIsIlN0b3B3YXRjaCIsIlZvbHVtZU1ldGVyIl0sIm1hcHBpbmdzIjoiOzs7OztBQUFBOzs7O0FBQ0E7Ozs7QUFDQTs7OztBQUNBOzs7O0FBQ0E7Ozs7QUFDQTs7OztBQUNBOzs7O0FBQ0E7Ozs7QUFDQTs7OztBQUNBOzs7O0FBQ0E7Ozs7QUFDQTs7OztBQUNBOzs7O0FBQ0E7Ozs7QUFDQTs7OztBQUNBOzs7O0FBQ0E7Ozs7QUFDQTs7Ozs7O1FBR0VBLGlCO1FBQ0FDLFM7UUFDQUMsUTtRQUNBQyxlO1FBQ0FDLFU7UUFDQUMsZ0I7UUFDQUMsSztRQUNBQyxZO1FBQ0FDLE87UUFDQUMsUTtRQUNBQyxzQjtRQUNBQyxJO1FBQ0FDLGU7UUFDQUMsSTtRQUVBQyxXO1FBQ0FDLGE7UUFFQUMsa0I7UUFFQUMsUztRQUVBQyxXIiwiZmlsZSI6ImluZGV4LmpzIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtkZWZhdWx0IGFzIFZvbHVtZU1ldGVyLCBnZW5lcmF0ZVdhdmVTYW1wbGV9IGZyb20gJy4vYXVkaW8vYXVkaW8tdG9vbHMnO1xuaW1wb3J0IEFkbWluaXN0cmF0aXZlU0RLIGZyb20gJy4vYWRtaW5pc3RyYXRpdmUtc2RrL2FkbWluaXN0cmF0aXZlLXNkayc7XG5pbXBvcnQgQXVkaW9QbGF5ZXIgZnJvbSAnLi9hdWRpby9hdWRpby1wbGF5ZXInO1xuaW1wb3J0IEF1ZGlvUmVjb3JkZXIgZnJvbSAnLi9hdWRpby9hdWRpby1yZWNvcmRlcic7XG5pbXBvcnQgQmFzaWNBdXRoIGZyb20gJy4vYWRtaW5pc3RyYXRpdmUtc2RrL2Jhc2ljLWF1dGgvYmFzaWMtYXV0aCc7XG5pbXBvcnQgQ2F0ZWdvcnkgZnJvbSAnLi9hZG1pbmlzdHJhdGl2ZS1zZGsvY2F0ZWdvcnkvY2F0ZWdvcnknO1xuaW1wb3J0IENob2ljZUNoYWxsZW5nZSBmcm9tICcuL2FkbWluaXN0cmF0aXZlLXNkay9jaG9pY2UtY2hhbGxlbmdlL2Nob2ljZS1jaGFsbGVuZ2UnO1xuaW1wb3J0IENvbm5lY3Rpb24gZnJvbSAnLi9hZG1pbmlzdHJhdGl2ZS1zZGsvY29ubmVjdGlvbi9jb25uZWN0aW9uLWNvbnRyb2xsZXInO1xuaW1wb3J0IEVtYWlsQ3JlZGVudGlhbHMgZnJvbSAnLi9hZG1pbmlzdHJhdGl2ZS1zZGsvZW1haWwtY3JlZGVudGlhbHMvZW1haWwtY3JlZGVudGlhbHMnO1xuaW1wb3J0IEdyb3VwIGZyb20gJy4vYWRtaW5pc3RyYXRpdmUtc2RrL2dyb3VwL2dyb3VwJztcbmltcG9ydCBPcmdhbmlzYXRpb24gZnJvbSAnLi9hZG1pbmlzdHJhdGl2ZS1zZGsvb3JnYW5pc2F0aW9uL29yZ2FuaXNhdGlvbic7XG5pbXBvcnQgUHJvZmlsZSBmcm9tICcuL2FkbWluaXN0cmF0aXZlLXNkay9wcm9maWxlL3Byb2ZpbGUnO1xuaW1wb3J0IFByb2dyZXNzIGZyb20gJy4vYWRtaW5pc3RyYXRpdmUtc2RrL3Byb2dyZXNzL3Byb2dyZXNzJztcbmltcG9ydCBQcm9udW5jaWF0aW9uQ2hhbGxlbmdlIGZyb20gJy4vYWRtaW5pc3RyYXRpdmUtc2RrL3Byb251bmNpYXRpb24tY2hhbGxlbmdlL3Byb251bmNpYXRpb24tY2hhbGxlbmdlJztcbmltcG9ydCBSb2xlIGZyb20gJy4vYWRtaW5pc3RyYXRpdmUtc2RrL3JvbGUvcm9sZSc7XG5pbXBvcnQgU3BlZWNoQ2hhbGxlbmdlIGZyb20gJy4vYWRtaW5pc3RyYXRpdmUtc2RrL3NwZWVjaC1jaGFsbGVuZ2Uvc3BlZWNoLWNoYWxsZW5nZSc7XG5pbXBvcnQgU3RvcHdhdGNoIGZyb20gJy4vYXVkaW8vdG9vbHMnO1xuaW1wb3J0IFVzZXIgZnJvbSAnLi9hZG1pbmlzdHJhdGl2ZS1zZGsvdXNlci91c2VyJztcblxuZXhwb3J0IHtcbiAgQWRtaW5pc3RyYXRpdmVTREssXG4gIEJhc2ljQXV0aCxcbiAgQ2F0ZWdvcnksXG4gIENob2ljZUNoYWxsZW5nZSxcbiAgQ29ubmVjdGlvbixcbiAgRW1haWxDcmVkZW50aWFscyxcbiAgR3JvdXAsXG4gIE9yZ2FuaXNhdGlvbixcbiAgUHJvZmlsZSxcbiAgUHJvZ3Jlc3MsXG4gIFByb251bmNpYXRpb25DaGFsbGVuZ2UsXG4gIFJvbGUsXG4gIFNwZWVjaENoYWxsZW5nZSxcbiAgVXNlcixcblxuICBBdWRpb1BsYXllcixcbiAgQXVkaW9SZWNvcmRlcixcblxuICBnZW5lcmF0ZVdhdmVTYW1wbGUsXG5cbiAgU3RvcHdhdGNoLFxuXG4gIFZvbHVtZU1ldGVyXG59O1xuIl19 | ||
exports.Player = _Player2.default; | ||
exports.Recorder = _Recorder2.default; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9pbmRleC5qcyJdLCJuYW1lcyI6WyJBZG1pbmlzdHJhdGl2ZVNESyIsIkJhc2ljQXV0aCIsIkNhdGVnb3J5IiwiQ2hvaWNlQ2hhbGxlbmdlIiwiQ29ubmVjdGlvbiIsIkVtYWlsQ3JlZGVudGlhbHMiLCJHcm91cCIsIk9yZ2FuaXNhdGlvbiIsIlByb2ZpbGUiLCJQcm9ncmVzcyIsIlByb251bmNpYXRpb25DaGFsbGVuZ2UiLCJSb2xlIiwiU3BlZWNoQ2hhbGxlbmdlIiwiVXNlciIsIkF1ZGlvUGxheWVyIiwiQXVkaW9SZWNvcmRlciIsImdlbmVyYXRlV2F2ZVNhbXBsZSIsIlN0b3B3YXRjaCIsIlZvbHVtZU1ldGVyIiwiUGxheWVyIiwiUmVjb3JkZXIiXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUE7Ozs7QUFDQTs7OztBQUNBOzs7O0FBQ0E7Ozs7QUFDQTs7OztBQUNBOzs7O0FBQ0E7Ozs7QUFDQTs7OztBQUNBOzs7O0FBQ0E7Ozs7QUFDQTs7OztBQUNBOzs7O0FBQ0E7Ozs7QUFDQTs7OztBQUNBOzs7O0FBQ0E7Ozs7QUFDQTs7OztBQUNBOzs7O0FBQ0E7Ozs7QUFDQTs7Ozs7O1FBR0VBLGlCO1FBQ0FDLFM7UUFDQUMsUTtRQUNBQyxlO1FBQ0FDLFU7UUFDQUMsZ0I7UUFDQUMsSztRQUNBQyxZO1FBQ0FDLE87UUFDQUMsUTtRQUNBQyxzQjtRQUNBQyxJO1FBQ0FDLGU7UUFDQUMsSTtRQUVBQyxXO1FBQ0FDLGE7UUFFQUMsa0I7UUFFQUMsUztRQUVBQyxXO1FBRUFDLE07UUFDQUMsUSIsImZpbGUiOiJpbmRleC5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7ZGVmYXVsdCBhcyBWb2x1bWVNZXRlciwgZ2VuZXJhdGVXYXZlU2FtcGxlfSBmcm9tICcuL2F1ZGlvL2F1ZGlvLXRvb2xzJztcbmltcG9ydCBBZG1pbmlzdHJhdGl2ZVNESyBmcm9tICcuL2FkbWluaXN0cmF0aXZlLXNkay9hZG1pbmlzdHJhdGl2ZS1zZGsnO1xuaW1wb3J0IEF1ZGlvUGxheWVyIGZyb20gJy4vYXVkaW8vYXVkaW8tcGxheWVyJztcbmltcG9ydCBBdWRpb1JlY29yZGVyIGZyb20gJy4vYXVkaW8vYXVkaW8tcmVjb3JkZXInO1xuaW1wb3J0IEJhc2ljQXV0aCBmcm9tICcuL2FkbWluaXN0cmF0aXZlLXNkay9iYXNpYy1hdXRoL2Jhc2ljLWF1dGgnO1xuaW1wb3J0IENhdGVnb3J5IGZyb20gJy4vYWRtaW5pc3RyYXRpdmUtc2RrL2NhdGVnb3J5L2NhdGVnb3J5JztcbmltcG9ydCBDaG9pY2VDaGFsbGVuZ2UgZnJvbSAnLi9hZG1pbmlzdHJhdGl2ZS1zZGsvY2hvaWNlLWNoYWxsZW5nZS9jaG9pY2UtY2hhbGxlbmdlJztcbmltcG9ydCBDb25uZWN0aW9uIGZyb20gJy4vYWRtaW5pc3RyYXRpdmUtc2RrL2Nvbm5lY3Rpb24vY29ubmVjdGlvbi1jb250cm9sbGVyJztcbmltcG9ydCBFbWFpbENyZWRlbnRpYWxzIGZyb20gJy4vYWRtaW5pc3RyYXRpdmUtc2RrL2VtYWlsLWNyZWRlbnRpYWxzL2VtYWlsLWNyZWRlbnRpYWxzJztcbmltcG9ydCBHcm91cCBmcm9tICcuL2FkbWluaXN0cmF0aXZlLXNkay9ncm91cC9ncm91cCc7XG5pbXBvcnQgT3JnYW5pc2F0aW9uIGZyb20gJy4vYWRtaW5pc3RyYXRpdmUtc2RrL29yZ2FuaXNhdGlvbi9vcmdhbmlzYXRpb24nO1xuaW1wb3J0IFBsYXllciBmcm9tICcuL1dlYkF1ZGlvL1BsYXllcic7XG5pbXBvcnQgUHJvZmlsZSBmcm9tICcuL2FkbWluaXN0cmF0aXZlLXNkay9wcm9maWxlL3Byb2ZpbGUnO1xuaW1wb3J0IFByb2dyZXNzIGZyb20gJy4vYWRtaW5pc3RyYXRpdmUtc2RrL3Byb2dyZXNzL3Byb2dyZXNzJztcbmltcG9ydCBQcm9udW5jaWF0aW9uQ2hhbGxlbmdlIGZyb20gJy4vYWRtaW5pc3RyYXRpdmUtc2RrL3Byb251bmNpYXRpb24tY2hhbGxlbmdlL3Byb251bmNpYXRpb24tY2hhbGxlbmdlJztcbmltcG9ydCBSZWNvcmRlciBmcm9tICcuL1dlYkF1ZGlvL1JlY29yZGVyJztcbmltcG9ydCBSb2xlIGZyb20gJy4vYWRtaW5pc3RyYXRpdmUtc2RrL3JvbGUvcm9sZSc7XG5pbXBvcnQgU3BlZWNoQ2hhbGxlbmdlIGZyb20gJy4vYWRtaW5pc3RyYXRpdmUtc2RrL3NwZWVjaC1jaGFsbGVuZ2Uvc3BlZWNoLWNoYWxsZW5nZSc7XG5pbXBvcnQgU3RvcHdhdGNoIGZyb20gJy4vYXVkaW8vdG9vbHMnO1xuaW1wb3J0IFVzZXIgZnJvbSAnLi9hZG1pbmlzdHJhdGl2ZS1zZGsvdXNlci91c2VyJztcblxuZXhwb3J0IHtcbiAgQWRtaW5pc3RyYXRpdmVTREssXG4gIEJhc2ljQXV0aCxcbiAgQ2F0ZWdvcnksXG4gIENob2ljZUNoYWxsZW5nZSxcbiAgQ29ubmVjdGlvbixcbiAgRW1haWxDcmVkZW50aWFscyxcbiAgR3JvdXAsXG4gIE9yZ2FuaXNhdGlvbixcbiAgUHJvZmlsZSxcbiAgUHJvZ3Jlc3MsXG4gIFByb251bmNpYXRpb25DaGFsbGVuZ2UsXG4gIFJvbGUsXG4gIFNwZWVjaENoYWxsZW5nZSxcbiAgVXNlcixcblxuICBBdWRpb1BsYXllcixcbiAgQXVkaW9SZWNvcmRlcixcblxuICBnZW5lcmF0ZVdhdmVTYW1wbGUsXG5cbiAgU3RvcHdhdGNoLFxuXG4gIFZvbHVtZU1ldGVyLFxuXG4gIFBsYXllcixcbiAgUmVjb3JkZXJcbn07XG4iXX0= |
{ | ||
"name": "itslanguage", | ||
"version": "3.0.0-beta-3", | ||
"version": "3.0.0-beta-4", | ||
"description": "The core JavaScript SDK for ITSLanguage.", | ||
"main": "index.js", | ||
"scripts": { | ||
"test": "concurrently 'eslint .' 'karma start --single-run --color'", | ||
"esdoc": "esdoc -c esdoc.json", | ||
"babelify": "babel src --out-dir build --source-maps inline && cp package.json README.md build/", | ||
"clearbuild": "rm -rf build/" | ||
"engines": { | ||
"npm": ">=5", | ||
"node": ">=8" | ||
}, | ||
@@ -28,32 +26,9 @@ "author": "ITSLanguage (https://www.itslanguage.nl) <support@d-centralize.nl>", | ||
"dependencies": { | ||
"autobahn": "0.11.2", | ||
"client-oauth2": "3.4.1", | ||
"event-emitter": "^0.3.4", | ||
"pcmjs": "^0.0.2", | ||
"uuid": "^3.0.1", | ||
"when": "^3.7.7" | ||
}, | ||
"devDependencies": { | ||
"babel-cli": "^6.22.2", | ||
"babel-preset-es2015": "^6.22.0", | ||
"babelify": "^7.3.0", | ||
"browserify": "^14.0.0", | ||
"browserify-istanbul": "^2.0.0", | ||
"bufferutil": "^2.0.1", | ||
"concurrently": "^3.3.0", | ||
"esdoc": "^0.5.2", | ||
"eslint": "^3.15.0", | ||
"eslint-config-d-centralize": "^0.4.1", | ||
"eslint-plugin-jsdoc": "^2.4.0", | ||
"isparta": "^4.0.0", | ||
"jasmine-ajax": "^3.3.1", | ||
"jasmine-core": "^2.5.2", | ||
"karma": "^1.4.1", | ||
"karma-browserify": "^5.1.1", | ||
"karma-chrome-launcher": "^2.0.0", | ||
"karma-cli": "^1.0.1", | ||
"karma-coverage": "^1.1.1", | ||
"karma-jasmine": "^1.1.0", | ||
"watchify": "^3.9.0" | ||
"autobahn": "17.5.2", | ||
"client-oauth2": "4.1.0", | ||
"event-emitter": "0.3.5", | ||
"pcmjs": "0.0.2", | ||
"uuid": "3.1.0", | ||
"when": "3.7.8" | ||
} | ||
} |
@@ -5,2 +5,4 @@ # ITSLanguage JavaScript SDK | ||
[](https://travis-ci.org/itslanguage/itslanguage-js) | ||
## Getting started | ||
@@ -13,52 +15,1 @@ | ||
``` | ||
### Dependencies | ||
Any JavaScript project can use this SDK to build applications for the | ||
ITSLanguage platform. There are, however, a few things to keep in mind; mostly | ||
the dependencies. ITSLanguage JavaScript SDK uses: | ||
1. [The `fetch` API][MDN fetch] | ||
1. [URLSearchParams][MDN URLSearchParams] | ||
1. [FormData][MDN FormData] | ||
1. [WebSocket][MDN WebSocket] | ||
It is expected that these are accessible through their `global` accessors (i.e. | ||
by simply calling `new FormData()`, `fetch(...)`, etc.). | ||
Modern browsers support these (at least to the capacity we use it). Older | ||
browsers as well as `Node` don't necessarily support these because the are, as | ||
of writing this, still seen as experimental (browser) features. They are living | ||
standards and therefore expected to be implemented in the future. | ||
In the mean time; you might want to look at a few libraries which will add | ||
these APIs to your environment. Here are a few we found useful. | ||
#### Browsers | ||
1. [whatwg-fetch][NPM whatwg-fetch] | ||
1. [url-search-params-polyfill][NPM url-search-params-polyfill] | ||
#### Node | ||
1. [node-fetch][NPM node-fetch] | ||
1. [url-search-params][NPM url-search-params] | ||
1. [form-data][NPM form-data] | ||
#### Both | ||
1. [isomorphic-fetch][NPM isomorphic-fetch] | ||
> We appreciate any contribution to extend/update these lists. | ||
[MDN fetch]: https://developer.mozilla.org/en/docs/Web/API/Fetch_API | ||
[MDN URLSearchParams]: https://developer.mozilla.org/en-US/docs/Web/API/URLSearchParams | ||
[MDN FormData]: https://developer.mozilla.org/en-US/docs/Web/API/FormData | ||
[MDN WebSocket]: https://developer.mozilla.org/en-US/docs/Web/API/WebSocket | ||
[NPM isomorphic-fetch]: https://www.npmjs.com/package/isomorphic-fetch | ||
[NPM form-data]: https://www.npmjs.com/package/form-data | ||
[NPM node-fetch]: https://www.npmjs.com/package/node-fetch | ||
[NPM url-search-params]: https://www.npmjs.com/package/url-search-params | ||
[NPM whatwg-fetch]: https://www.npmjs.com/package/whatwg-fetch | ||
[NPM url-search-params-polyfill]: https://www.npmjs.com/package/url-search-params-polyfill |
802331
0
67
6884
14
+ Addedautobahn@17.5.2(transitive)
+ Addedcbor@10.0.3(transitive)
+ Addedclient-oauth2@4.1.0(transitive)
+ Addednofilter@3.1.0(transitive)
+ Addedpopsicle@9.2.0(transitive)
+ Addeduuid@3.1.0(transitive)
- Removedany-promise@1.3.0(transitive)
- Removedarrify@1.0.1(transitive)
- Removedautobahn@0.11.2(transitive)
- Removedclient-oauth2@3.4.1(transitive)
- Removedpopsicle@8.2.0(transitive)
- Removedthrowback@1.1.1(transitive)
- Removeduuid@3.4.0(transitive)
- Removedxtend@4.0.2(transitive)
Updatedautobahn@17.5.2
Updatedclient-oauth2@4.1.0
Updatedevent-emitter@0.3.5
Updatedpcmjs@0.0.2
Updateduuid@3.1.0
Updatedwhen@3.7.8