itslanguage
Advanced tools
Comparing version 3.0.0-beta-4.7 to 3.0.0-beta-4.8
@@ -11,2 +11,3 @@ 'use strict'; | ||
exports.endStreamAudioForChoiceRecognition = endStreamAudioForChoiceRecognition; | ||
exports.prepareAudioStream = prepareAudioStream; | ||
@@ -60,9 +61,17 @@ var _audioOverSocket = require('../../utils/audio-over-socket'); | ||
* below is some code for the new streaming yet to come. | ||
* | ||
* export function prepareAudioStream(recognitionId, recorder) { | ||
* const rpc = 'choise.chuck'; | ||
* return registerAudioStream(recorder, rpc) | ||
* .then(() => makeWebsocketCall('choice.recognise', {args: [recognitionId, rpc]})); | ||
* } | ||
*/ | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9hcGkvY2hhbGxlbmdlcy9jaG9pY2UvcmVjb2duaXRpb24uanMiXSwibmFtZXMiOlsiZ2V0QWxsQ2hvaWNlUmVjb2duaXRpb25zIiwiZ2V0Q2hvaWNlUmVjb2duaXRpb25CeUlEIiwicHJlcGFyZUNob2ljZVJlY29nbml0aW9uIiwicHJlcGFyZUNob2ljZVJlY29nbml0aW9uQ2hhbGxlbmdlIiwicHJlcGFyZUF1ZGlvRm9yQ2hvaWNlUmVjb2duaXRpb24iLCJzdHJlYW1BdWRpb0ZvckNob2ljZVJlY29nbml0aW9uIiwiZW5kU3RyZWFtQXVkaW9Gb3JDaG9pY2VSZWNvZ25pdGlvbiIsInVybCIsImNoYWxsZW5nZSIsImlkIiwicmVjb2duaXRpb25JZCIsImNoYWxsZW5nZUlkIiwiYXJncyIsInJlY29yZGVyIiwicHJvZ3Jlc3NDYiJdLCJtYXBwaW5ncyI6Ijs7O1FBa0JnQkEsd0IsR0FBQUEsd0I7UUFJQUMsd0IsR0FBQUEsd0I7UUFJQUMsd0IsR0FBQUEsd0I7UUFJQUMsaUMsR0FBQUEsaUM7UUFJQUMsZ0MsR0FBQUEsZ0M7UUFJQUMsK0IsR0FBQUEsK0I7UUFJQUMsa0MsR0FBQUEsa0M7O0FBakNoQjs7QUFJQTs7QUFDQTs7QUFFQSxJQUFNQyxNQUFNLFNBQU5BLEdBQU07QUFBQSxpQ0FBbUNDLFNBQW5DO0FBQUEsQ0FBWixDLENBaEJBOzs7Ozs7Ozs7QUFrQk8sU0FBU1Isd0JBQVQsR0FBb0M7QUFDekMsU0FBTyxzQ0FBa0IsS0FBbEIsT0FBNEJPLEdBQTVCLENBQVA7QUFDRDs7QUFFTSxTQUFTTix3QkFBVCxDQUFrQ1EsRUFBbEMsRUFBc0M7QUFDM0MsU0FBTyxzQ0FBa0IsS0FBbEIsRUFBNEJGLEdBQTVCLFNBQW1DRSxFQUFuQyxDQUFQO0FBQ0Q7O0FBRU0sU0FBU1Asd0JBQVQsR0FBb0M7QUFDekMsU0FBTyxrQ0FBa0IseUJBQWxCLENBQVA7QUFDRDs7QUFFTSxTQUFTQyxpQ0FBVCxDQUEyQ08sYUFBM0MsRUFBMERDLFdBQTFELEVBQXVFO0FBQzVFLFNBQU8sa0NBQWtCLHVCQUFsQixFQUEyQyxFQUFDQyxNQUFNLENBQUNGLGFBQUQsRUFBZ0JDLFdBQWhCLENBQVAsRUFBM0MsQ0FBUDtBQUNEOztBQUVNLFNBQVNQLGdDQUFULENBQTBDTSxhQUExQyxFQUF5REcsUUFBekQsRUFBbUU7QUFDeEUsU0FBTyw0Q0FBc0JILGFBQXRCLEVBQXFDRyxRQUFyQyxFQUErQyxtQkFBL0MsQ0FBUDtBQUNEOztBQUVNLFNBQVNSLCtCQUFULENBQXlDSyxhQUF6QyxFQUF3REcsUUFBeEQsRUFBa0U7QUFDdkUsU0FBTyx3REFBa0NILGFBQWxDLEVBQWlERyxRQUFqRCxFQUEyRCxjQUEzRCxDQUFQO0FBQ0Q7O0FBRU0sU0FBU1Asa0NBQVQsQ0FBNENJLGFBQTVDLEVBQTJESSxVQUEzRCxFQUF1RTtBQUM1RSxTQUFPLGtDQUFrQixrQkFBbEIsRUFBc0MsRUFBQ0YsTUFBTSxDQUFDRixhQUFELENBQVAsRUFBd0JJLHNCQUF4QixFQUF0QyxDQUFQO0FBQ0Q7O0FBRUQiLCJmaWxlIjoicmVjb2duaXRpb24uanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFRoaXMgZmlsZSBjb250YWlucyB0aGUgcmVhZGlseSBhdmFpbGFibGUgZnVuY3Rpb25zIHdoaWNoIGludGVyYWN0IHdpdGggdGhlXG4gKiBJVFNMYW5ndWFnZSBjaG9pY2UgcmVjb2duaXRpb24gQVBJLlxuICpcbiAqIE5vdGUgdGhhdCB0aGlzIGlzIG9uZSBvZiB0aGUgXCJuZXN0ZWRcIiBvciBcImNvbXBvc2l0ZVwiIEFQSXM7IFlvdSBjYW4gb25seVxuICogb2J0YWluIHRoZSBkYXRhIGlmIHlvdSBwcm92aWRlIGEgcmVmZXJlbmNlIHRvIHRoZSBjaGFsbGVuZ2UgZm9yIHdoaWNoIHlvdVxuICogd2FudCBhIHJlY29yZGluZy5cbiAqL1xuXG5pbXBvcnQge1xuICBlbmNvZGVBbmRTZW5kQXVkaW9PbkRhdGFBdmFpbGlibGUsXG4gIHByZXBhcmVTZXJ2ZXJGb3JBdWRpb1xufSBmcm9tICcuLi8uLi91dGlscy9hdWRpby1vdmVyLXNvY2tldCc7XG5pbXBvcnQge2F1dGhvcmlzZWRSZXF1ZXN0fSBmcm9tICcuLi8uLi9jb21tdW5pY2F0aW9uJztcbmltcG9ydCB7bWFrZVdlYnNvY2tldENhbGx9IGZyb20gJy4uLy4uL2NvbW11bmljYXRpb24vd2Vic29ja2V0JztcblxuY29uc3QgdXJsID0gY2hhbGxlbmdlID0+IGAvY2hhbGxlbmdlcy9jaG9pY2UvJHtjaGFsbGVuZ2V9L3JlY29nbml0aW9uc2A7XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRBbGxDaG9pY2VSZWNvZ25pdGlvbnMoKSB7XG4gIHJldHVybiBhdXRob3Jpc2VkUmVxdWVzdCgnR0VUJywgYCR7dXJsfWApO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0Q2hvaWNlUmVjb2duaXRpb25CeUlEKGlkKSB7XG4gIHJldHVybiBhdXRob3Jpc2VkUmVxdWVzdCgnR0VUJywgYCR7dXJsfS8ke2lkfWApO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gcHJlcGFyZUNob2ljZVJlY29nbml0aW9uKCkge1xuICByZXR1cm4gbWFrZVdlYnNvY2tldENhbGwoJ2Nob2ljZS5pbml0X3JlY29nbml0aW9uJyk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBwcmVwYXJlQ2hvaWNlUmVjb2duaXRpb25DaGFsbGVuZ2UocmVjb2duaXRpb25JZCwgY2hhbGxlbmdlSWQpIHtcbiAgcmV0dXJuIG1ha2VXZWJzb2NrZXRDYWxsKCdjaG9pY2UuaW5pdF9jaGFsbGVuZ2UnLCB7YXJnczogW3JlY29nbml0aW9uSWQsIGNoYWxsZW5nZUlkXX0pO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gcHJlcGFyZUF1ZGlvRm9yQ2hvaWNlUmVjb2duaXRpb24ocmVjb2duaXRpb25JZCwgcmVjb3JkZXIpIHtcbiAgcmV0dXJuIHByZXBhcmVTZXJ2ZXJGb3JBdWRpbyhyZWNvZ25pdGlvbklkLCByZWNvcmRlciwgJ2Nob2ljZS5pbml0X2F1ZGlvJyk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBzdHJlYW1BdWRpb0ZvckNob2ljZVJlY29nbml0aW9uKHJlY29nbml0aW9uSWQsIHJlY29yZGVyKSB7XG4gIHJldHVybiBlbmNvZGVBbmRTZW5kQXVkaW9PbkRhdGFBdmFpbGlibGUocmVjb2duaXRpb25JZCwgcmVjb3JkZXIsICdjaG9pY2Uud3JpdGUnKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGVuZFN0cmVhbUF1ZGlvRm9yQ2hvaWNlUmVjb2duaXRpb24ocmVjb2duaXRpb25JZCwgcHJvZ3Jlc3NDYikge1xuICByZXR1cm4gbWFrZVdlYnNvY2tldENhbGwoJ2Nob2ljZS5yZWNvZ25pc2UnLCB7YXJnczogW3JlY29nbml0aW9uSWRdLCBwcm9ncmVzc0NifSk7XG59XG5cbi8qXG4gKiBiZWxvdyBpcyBzb21lIGNvZGUgZm9yIHRoZSBuZXcgc3RyZWFtaW5nIHlldCB0byBjb21lLlxuICpcbiAqIGV4cG9ydCBmdW5jdGlvbiBwcmVwYXJlQXVkaW9TdHJlYW0ocmVjb2duaXRpb25JZCwgcmVjb3JkZXIpIHtcbiAqICAgY29uc3QgcnBjID0gJ2Nob2lzZS5jaHVjayc7XG4gKiAgIHJldHVybiByZWdpc3RlckF1ZGlvU3RyZWFtKHJlY29yZGVyLCBycGMpXG4gKiAgICAgLnRoZW4oKCkgPT4gbWFrZVdlYnNvY2tldENhbGwoJ2Nob2ljZS5yZWNvZ25pc2UnLCB7YXJnczogW3JlY29nbml0aW9uSWQsIHJwY119KSk7XG4gKiB9XG4gKi9cbiJdfQ== | ||
function prepareAudioStream(recognitionId, recorder) { | ||
var _arguments = arguments; | ||
var rpc = 'choice.chuck'; | ||
return (0, _audioOverSocket.registerAudioStream)(recorder, rpc).then(function (_ref) { | ||
var procedure = _ref.procedure; | ||
if (procedure) { | ||
(0, _websocket.makeWebsocketCall)('choice.recognise', { args: [recognitionId, procedure] }); | ||
} else { | ||
console.log(_arguments); | ||
} | ||
}); | ||
} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9hcGkvY2hhbGxlbmdlcy9jaG9pY2UvcmVjb2duaXRpb24uanMiXSwibmFtZXMiOlsiZ2V0QWxsQ2hvaWNlUmVjb2duaXRpb25zIiwiZ2V0Q2hvaWNlUmVjb2duaXRpb25CeUlEIiwicHJlcGFyZUNob2ljZVJlY29nbml0aW9uIiwicHJlcGFyZUNob2ljZVJlY29nbml0aW9uQ2hhbGxlbmdlIiwicHJlcGFyZUF1ZGlvRm9yQ2hvaWNlUmVjb2duaXRpb24iLCJzdHJlYW1BdWRpb0ZvckNob2ljZVJlY29nbml0aW9uIiwiZW5kU3RyZWFtQXVkaW9Gb3JDaG9pY2VSZWNvZ25pdGlvbiIsInByZXBhcmVBdWRpb1N0cmVhbSIsInVybCIsImNoYWxsZW5nZSIsImlkIiwicmVjb2duaXRpb25JZCIsImNoYWxsZW5nZUlkIiwiYXJncyIsInJlY29yZGVyIiwicHJvZ3Jlc3NDYiIsInJwYyIsInRoZW4iLCJwcm9jZWR1cmUiLCJjb25zb2xlIiwibG9nIl0sIm1hcHBpbmdzIjoiOzs7UUFtQmdCQSx3QixHQUFBQSx3QjtRQUlBQyx3QixHQUFBQSx3QjtRQUlBQyx3QixHQUFBQSx3QjtRQUlBQyxpQyxHQUFBQSxpQztRQUlBQyxnQyxHQUFBQSxnQztRQUlBQywrQixHQUFBQSwrQjtRQUlBQyxrQyxHQUFBQSxrQztRQU9BQyxrQixHQUFBQSxrQjs7QUF6Q2hCOztBQUtBOztBQUNBOztBQUVBLElBQU1DLE1BQU0sU0FBTkEsR0FBTTtBQUFBLGlDQUFtQ0MsU0FBbkM7QUFBQSxDQUFaLEMsQ0FqQkE7Ozs7Ozs7OztBQW1CTyxTQUFTVCx3QkFBVCxHQUFvQztBQUN6QyxTQUFPLHNDQUFrQixLQUFsQixPQUE0QlEsR0FBNUIsQ0FBUDtBQUNEOztBQUVNLFNBQVNQLHdCQUFULENBQWtDUyxFQUFsQyxFQUFzQztBQUMzQyxTQUFPLHNDQUFrQixLQUFsQixFQUE0QkYsR0FBNUIsU0FBbUNFLEVBQW5DLENBQVA7QUFDRDs7QUFFTSxTQUFTUix3QkFBVCxHQUFvQztBQUN6QyxTQUFPLGtDQUFrQix5QkFBbEIsQ0FBUDtBQUNEOztBQUVNLFNBQVNDLGlDQUFULENBQTJDUSxhQUEzQyxFQUEwREMsV0FBMUQsRUFBdUU7QUFDNUUsU0FBTyxrQ0FBa0IsdUJBQWxCLEVBQTJDLEVBQUNDLE1BQU0sQ0FBQ0YsYUFBRCxFQUFnQkMsV0FBaEIsQ0FBUCxFQUEzQyxDQUFQO0FBQ0Q7O0FBRU0sU0FBU1IsZ0NBQVQsQ0FBMENPLGFBQTFDLEVBQXlERyxRQUF6RCxFQUFtRTtBQUN4RSxTQUFPLDRDQUFzQkgsYUFBdEIsRUFBcUNHLFFBQXJDLEVBQStDLG1CQUEvQyxDQUFQO0FBQ0Q7O0FBRU0sU0FBU1QsK0JBQVQsQ0FBeUNNLGFBQXpDLEVBQXdERyxRQUF4RCxFQUFrRTtBQUN2RSxTQUFPLHdEQUFrQ0gsYUFBbEMsRUFBaURHLFFBQWpELEVBQTJELGNBQTNELENBQVA7QUFDRDs7QUFFTSxTQUFTUixrQ0FBVCxDQUE0Q0ssYUFBNUMsRUFBMkRJLFVBQTNELEVBQXVFO0FBQzVFLFNBQU8sa0NBQWtCLGtCQUFsQixFQUFzQyxFQUFDRixNQUFNLENBQUNGLGFBQUQsQ0FBUCxFQUF3Qkksc0JBQXhCLEVBQXRDLENBQVA7QUFDRDs7QUFFRDs7O0FBR08sU0FBU1Isa0JBQVQsQ0FBNEJJLGFBQTVCLEVBQTJDRyxRQUEzQyxFQUFxRDtBQUFBOztBQUMxRCxNQUFNRSxNQUFNLGNBQVo7QUFDQSxTQUFPLDBDQUFvQkYsUUFBcEIsRUFBOEJFLEdBQTlCLEVBQ0pDLElBREksQ0FDQyxnQkFBaUI7QUFBQSxRQUFmQyxTQUFlLFFBQWZBLFNBQWU7O0FBQ3JCLFFBQUlBLFNBQUosRUFBZTtBQUNiLHdDQUFrQixrQkFBbEIsRUFBc0MsRUFBQ0wsTUFBTSxDQUFDRixhQUFELEVBQWdCTyxTQUFoQixDQUFQLEVBQXRDO0FBQ0QsS0FGRCxNQUVPO0FBQ0xDLGNBQVFDLEdBQVI7QUFDRDtBQUNGLEdBUEksQ0FBUDtBQVFEIiwiZmlsZSI6InJlY29nbml0aW9uLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBUaGlzIGZpbGUgY29udGFpbnMgdGhlIHJlYWRpbHkgYXZhaWxhYmxlIGZ1bmN0aW9ucyB3aGljaCBpbnRlcmFjdCB3aXRoIHRoZVxuICogSVRTTGFuZ3VhZ2UgY2hvaWNlIHJlY29nbml0aW9uIEFQSS5cbiAqXG4gKiBOb3RlIHRoYXQgdGhpcyBpcyBvbmUgb2YgdGhlIFwibmVzdGVkXCIgb3IgXCJjb21wb3NpdGVcIiBBUElzOyBZb3UgY2FuIG9ubHlcbiAqIG9idGFpbiB0aGUgZGF0YSBpZiB5b3UgcHJvdmlkZSBhIHJlZmVyZW5jZSB0byB0aGUgY2hhbGxlbmdlIGZvciB3aGljaCB5b3VcbiAqIHdhbnQgYSByZWNvcmRpbmcuXG4gKi9cblxuaW1wb3J0IHtcbiAgZW5jb2RlQW5kU2VuZEF1ZGlvT25EYXRhQXZhaWxpYmxlLFxuICBwcmVwYXJlU2VydmVyRm9yQXVkaW8sXG4gIHJlZ2lzdGVyQXVkaW9TdHJlYW0sXG59IGZyb20gJy4uLy4uL3V0aWxzL2F1ZGlvLW92ZXItc29ja2V0JztcbmltcG9ydCB7YXV0aG9yaXNlZFJlcXVlc3R9IGZyb20gJy4uLy4uL2NvbW11bmljYXRpb24nO1xuaW1wb3J0IHttYWtlV2Vic29ja2V0Q2FsbH0gZnJvbSAnLi4vLi4vY29tbXVuaWNhdGlvbi93ZWJzb2NrZXQnO1xuXG5jb25zdCB1cmwgPSBjaGFsbGVuZ2UgPT4gYC9jaGFsbGVuZ2VzL2Nob2ljZS8ke2NoYWxsZW5nZX0vcmVjb2duaXRpb25zYDtcblxuZXhwb3J0IGZ1bmN0aW9uIGdldEFsbENob2ljZVJlY29nbml0aW9ucygpIHtcbiAgcmV0dXJuIGF1dGhvcmlzZWRSZXF1ZXN0KCdHRVQnLCBgJHt1cmx9YCk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRDaG9pY2VSZWNvZ25pdGlvbkJ5SUQoaWQpIHtcbiAgcmV0dXJuIGF1dGhvcmlzZWRSZXF1ZXN0KCdHRVQnLCBgJHt1cmx9LyR7aWR9YCk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBwcmVwYXJlQ2hvaWNlUmVjb2duaXRpb24oKSB7XG4gIHJldHVybiBtYWtlV2Vic29ja2V0Q2FsbCgnY2hvaWNlLmluaXRfcmVjb2duaXRpb24nKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIHByZXBhcmVDaG9pY2VSZWNvZ25pdGlvbkNoYWxsZW5nZShyZWNvZ25pdGlvbklkLCBjaGFsbGVuZ2VJZCkge1xuICByZXR1cm4gbWFrZVdlYnNvY2tldENhbGwoJ2Nob2ljZS5pbml0X2NoYWxsZW5nZScsIHthcmdzOiBbcmVjb2duaXRpb25JZCwgY2hhbGxlbmdlSWRdfSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBwcmVwYXJlQXVkaW9Gb3JDaG9pY2VSZWNvZ25pdGlvbihyZWNvZ25pdGlvbklkLCByZWNvcmRlcikge1xuICByZXR1cm4gcHJlcGFyZVNlcnZlckZvckF1ZGlvKHJlY29nbml0aW9uSWQsIHJlY29yZGVyLCAnY2hvaWNlLmluaXRfYXVkaW8nKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIHN0cmVhbUF1ZGlvRm9yQ2hvaWNlUmVjb2duaXRpb24ocmVjb2duaXRpb25JZCwgcmVjb3JkZXIpIHtcbiAgcmV0dXJuIGVuY29kZUFuZFNlbmRBdWRpb09uRGF0YUF2YWlsaWJsZShyZWNvZ25pdGlvbklkLCByZWNvcmRlciwgJ2Nob2ljZS53cml0ZScpO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZW5kU3RyZWFtQXVkaW9Gb3JDaG9pY2VSZWNvZ25pdGlvbihyZWNvZ25pdGlvbklkLCBwcm9ncmVzc0NiKSB7XG4gIHJldHVybiBtYWtlV2Vic29ja2V0Q2FsbCgnY2hvaWNlLnJlY29nbmlzZScsIHthcmdzOiBbcmVjb2duaXRpb25JZF0sIHByb2dyZXNzQ2J9KTtcbn1cblxuLypcbiAqIGJlbG93IGlzIHNvbWUgY29kZSBmb3IgdGhlIG5ldyBzdHJlYW1pbmcgeWV0IHRvIGNvbWUuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBwcmVwYXJlQXVkaW9TdHJlYW0ocmVjb2duaXRpb25JZCwgcmVjb3JkZXIpIHtcbiAgY29uc3QgcnBjID0gJ2Nob2ljZS5jaHVjayc7XG4gIHJldHVybiByZWdpc3RlckF1ZGlvU3RyZWFtKHJlY29yZGVyLCBycGMpXG4gICAgLnRoZW4oKHtwcm9jZWR1cmV9KSA9PiB7XG4gICAgICBpZiAocHJvY2VkdXJlKSB7XG4gICAgICAgIG1ha2VXZWJzb2NrZXRDYWxsKCdjaG9pY2UucmVjb2duaXNlJywge2FyZ3M6IFtyZWNvZ25pdGlvbklkLCBwcm9jZWR1cmVdfSk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBjb25zb2xlLmxvZyhhcmd1bWVudHMpO1xuICAgICAgfVxuICAgIH0pO1xufVxuXG4iXX0= |
'use strict'; | ||
exports.__esModule = true; | ||
exports.settings = undefined; | ||
exports.updateSettings = updateSettings; | ||
exports.request = request; | ||
exports.authorisedRequest = authorisedRequest; | ||
/** | ||
* This file contains the settings and the communication mechanism for the | ||
* ITSLanguage REST API. | ||
*/ | ||
require('whatwg-fetch'); | ||
// Headers | ||
var CONTENT_TYPE = 'Content-Type'; | ||
var CONTENT_TYPE = 'Content-Type'; /** | ||
* This file contains the settings and the communication mechanism for the | ||
* ITSLanguage REST API. | ||
*/ | ||
var AUTHORIZATION = 'Authorization'; | ||
@@ -175,2 +178,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","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"]} | ||
//# 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":";;;;QAiCgBA,c,GAAAA,c;QAiEAC,O,GAAAA,O;QA6DAC,iB,GAAAA,iB;;AA1JhB;;AAEA;AACA,IAAMC,eAAe,cAArB,C,CARA;;;;;AASA,IAAMC,gBAAgB,eAAtB;;AAEA;AACA,IAAMC,mBAAmB,kBAAzB;;AAGA;;;AAGO,IAAMC,8BAAW;AACtBC,UAAQ,4BADc;AAEtBC,SAAO,IAFe;AAGtBC,sBAAoB;AAHE,CAAjB;;AAOP;;;;;;;;AAQO,SAAST,cAAT,CAAwBU,WAAxB,EAAqC;AAC1C,MAAI,CAACA,WAAD,IAAgB,EAAEA,uBAAuBC,MAAzB,CAApB,EAAsD;AACpD,UAAM,IAAIC,KAAJ,CAAU,2CAAV,CAAN;AACD;;AAEDD,SAAOE,MAAP,CAAcP,QAAd,EAAwBI,WAAxB;AACD;;AAGD;;;;;;;;;;;;;;;;;AAiBA,SAASI,cAAT,CAAwBC,QAAxB,EAAkC;AAChC,MAAMC,sBAAsBD,SAASE,OAAT,CAAiBC,GAAjB,CAAqBf,YAArB,CAA5B;;AAEA;AACA,MAAIa,uBAAuBA,oBAAoBG,QAApB,CAA6Bd,gBAA7B,CAA3B,EAA2E;AACzE,WAAOU,SAASK,IAAT,GAAgBC,IAAhB,CAAqB,gBAAQ;AAClC,UAAIN,SAASO,EAAb,EAAiB;AACf,eAAOF,IAAP;AACD;;AAED,aAAOG,QAAQC,MAAR,CAAeJ,IAAf,CAAP;AACD,KANM,CAAP;AAOD;;AAED,MAAI,CAACL,SAASO,EAAd,EAAkB;AAChB,WAAOC,QAAQC,MAAR,CAAkBT,SAASU,MAA3B,UAAsCV,SAASW,UAA/C,CAAP;AACD;;AAED,SAAOX,QAAP;AACD;;AAGD;;;;;;;;;;;;;;;;;AAiBO,SAASd,OAAT,CAAiB0B,MAAjB,EAAyBC,GAAzB,EAA8BC,IAA9B,EAAoCZ,OAApC,EAA6C;AAClD,MAAMa,iBAAiBb,WAAW,IAAIc,OAAJ,EAAlC;;AAEA,MAAIC,cAAcH,IAAlB;AACA,MAAI,EAAEA,gBAAgBI,eAAhB,IAAmCJ,gBAAgBK,QAArD,KAAkEL,gBAAgBlB,MAAtF,EAA8F;AAC5FmB,mBAAeK,GAAf,CAAmB,cAAnB,EAAmC,kBAAnC;AACAH,kBAAcI,KAAKC,SAAL,CAAeR,IAAf,CAAd;AACD;;AAED,MAAIS,aAAaV,GAAjB;AACA;AACA;AACA,MAAIA,OAAOA,IAAIW,UAAJ,CAAe,GAAf,CAAX,EAAgC;AAC9BD,sBAAgBhC,SAASC,MAAzB,GAAkCqB,GAAlC;AACD;;AAED,MAAMY,iBAAiB;AACrBb,kBADqB;AAErBV,aAASa,cAFY;AAGrBD,UAAMG;AAHe,GAAvB;;AAMA,SAAOS,MAAMH,UAAN,EAAkBE,cAAlB,EAAkCnB,IAAlC,CAAuCP,cAAvC,CAAP;AACD;;AAGD;;;;;;;AAOA,SAAS4B,cAAT,GAA0B;AACxB,MAAI,CAACpC,SAASG,kBAAd,EAAkC;AAChC,UAAM,IAAIG,KAAJ,CAAU,4BAAV,CAAN;AACD;;AAED,qBAAiBN,SAASG,kBAA1B;AACD;;AAGD;;;;;;;;;;;;;;;;;;;AAmBO,SAASP,iBAAT,CAA2ByB,MAA3B,EAAmCC,GAAnC,EAAwCC,IAAxC,EAA8CZ,OAA9C,EAAuD;AAC5D;AACA;AACA;AACA,MAAIW,OAAQ,CAACA,IAAIW,UAAJ,CAAe,GAAf,CAAD,IAAwB,CAACX,IAAIW,UAAJ,CAAejC,SAASC,MAAxB,CAArC,EAAuE;AACrE,WAAOgB,QAAQC,MAAR,CAAe,iDAAf,CAAP;AACD;;AAED,MAAI;AACF,QAAMM,iBAAiBb,WAAW,IAAIc,OAAJ,EAAlC;AACAD,mBAAeK,GAAf,CAAmB/B,aAAnB,EAAkCsC,gBAAlC;;AAEA,WAAOzC,QAAQ0B,MAAR,EAAgBC,GAAhB,EAAqBC,IAArB,EAA2BC,cAA3B,CAAP;AACD,GALD,CAKE,OAAOa,YAAP,EAAqB;AACrB,WAAOpB,QAAQC,MAAR,CAAemB,aAAaC,OAA5B,CAAP;AACD;AACF","file":"index.js","sourcesContent":["/**\n * This file contains the settings and the communication mechanism for the\n * ITSLanguage REST API.\n */\n\nimport 'whatwg-fetch';\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"]} |
@@ -198,4 +198,6 @@ 'use strict'; | ||
var _ref = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}, | ||
args = _ref.args, | ||
kwargs = _ref.kwargs, | ||
_ref$args = _ref.args, | ||
args = _ref$args === undefined ? [] : _ref$args, | ||
_ref$kwargs = _ref.kwargs, | ||
kwargs = _ref$kwargs === undefined ? {} : _ref$kwargs, | ||
options = _ref.options, | ||
@@ -229,2 +231,2 @@ progressCb = _ref.progressCb; | ||
} | ||
//# 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":"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","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"]} | ||
//# 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":"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","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"]} |
'use strict'; | ||
exports.__esModule = true; | ||
exports.registerAudioStream = undefined; | ||
/* | ||
* next to sending audo as encoded chunks, we can also prepare a rpc | ||
* that'll fetch the result for us. | ||
* Below is an example of how this could look like. | ||
* Remember to import getWebsocket and authobahn for this to work! | ||
*/ | ||
var registerAudioStream = exports.registerAudioStream = function () { | ||
var _ref = _asyncToGenerator( /*#__PURE__*/regeneratorRuntime.mark(function _callee(recorder, rpcName) { | ||
var rpc, rpcRegistration, asyncChucks; | ||
return regeneratorRuntime.wrap(function _callee$(_context) { | ||
while (1) { | ||
switch (_context.prev = _context.next) { | ||
case 0: | ||
asyncChucks = function asyncChucks(args, kwargs, details) { | ||
// eslint-disable-next-line new-cap | ||
var defer = new _autobahn2.default.when.defer(); | ||
if (details.progress) { | ||
recorder.addEventListener('dataavailable', function (chuck) { | ||
var dataToSend = Array.from(new Uint8Array(chuck)); | ||
details.progress([dataToSend]); | ||
}); | ||
recorder.addEventListener('recorded', function () { | ||
defer.resolve('Done with streaming you basterd'); | ||
if (rpcRegistration) { | ||
console.log('yeash!'); | ||
(0, _websocket.getWebsocketConnection)().then(function (connection) { | ||
return connection.session.unregister(rpcRegistration).then(function () { | ||
console.log('succesfully un-registered'); | ||
}); | ||
}); | ||
} | ||
}); | ||
} | ||
return defer.promise; | ||
}; | ||
rpc = 'nl.itslanguage.' + rpcName; | ||
rpcRegistration = null; | ||
return _context.abrupt('return', (0, _websocket.getWebsocketConnection)().then(function (connection) { | ||
return connection.session.register(rpc, asyncChucks).then(function (registration) { | ||
rpcRegistration = registration; | ||
return registration; | ||
}); | ||
})); | ||
case 4: | ||
case 'end': | ||
return _context.stop(); | ||
} | ||
} | ||
}, _callee, this); | ||
})); | ||
return function registerAudioStream(_x, _x2) { | ||
return _ref.apply(this, arguments); | ||
}; | ||
}(); | ||
/** | ||
* Send the recorder settings to the websocket server to initialize it. | ||
* | ||
* The reserved ID (passed in the parameters) is returned once the promise is | ||
* resolved. | ||
* | ||
* @param {string} id - The reserved ID for the audio. | ||
* @param {MediaRecorder|Recorder} recorder - The recorder which has been set up to | ||
* record. | ||
* @param {string} rpc - The RPC to use to initialize the websocket server. | ||
* | ||
* @emits {websocketserverreadyforaudio} - When the websocket server has been | ||
* prepared for and is ready to receive | ||
* the audio. | ||
* | ||
* @returns {Promise} - The promise which resolves when the websocket server | ||
* is ready for the audio. | ||
*/ | ||
exports.encodeAndSendAudioOnDataAvailible = encodeAndSendAudioOnDataAvailible; | ||
@@ -8,2 +91,6 @@ exports.prepareServerForAudio = prepareServerForAudio; | ||
var _autobahn = require('autobahn'); | ||
var _autobahn2 = _interopRequireDefault(_autobahn); | ||
var _broadcaster = require('../broadcaster'); | ||
@@ -19,2 +106,6 @@ | ||
function _asyncToGenerator(fn) { return function () { var gen = fn.apply(this, arguments); return new Promise(function (resolve, reject) { function step(key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { return Promise.resolve(value).then(function (value) { step("next", value); }, function (err) { step("throw", err); }); } } return step("next"); }); }; } /** | ||
* This file contains some re-usable parts for websocket audio communication. | ||
*/ | ||
/** | ||
@@ -38,57 +129,3 @@ * Encode the audio as base64 and send it to the websocket server. | ||
}); | ||
} | ||
/* | ||
* next to sending audo as encoded chunks, we can also prepare a rpc | ||
* that'll fetch the result for us. | ||
* Below is an example of how this could look like. | ||
* Remember to import getWebsocket and authobahn for this to work! | ||
* | ||
* export function registerAudioStream(recorder, rpcName) { | ||
* const rpc = `nl.itslanguage.${rpcName}`; | ||
* | ||
* const sendSomeChucks = () => { | ||
* const defer = new autobahn.when.defer(); | ||
* | ||
* recorder.addEventListener('dataavailable', chuck => { | ||
* console.log(chuck); | ||
* defer.notify(chuck); | ||
* }); | ||
* | ||
* recorder.addEventListener('recorded', () => { | ||
* defer.resolve('stream ended'); | ||
* }); | ||
* | ||
* return defer.promise; | ||
* }; | ||
* | ||
* return getWebsocketConnection().then(connection => | ||
* connection.session.register(rpc, sendSomeChucks) | ||
* ); | ||
* } | ||
*/ | ||
/** | ||
* Send the recorder settings to the websocket server to initialize it. | ||
* | ||
* The reserved ID (passed in the parameters) is returned once the promise is | ||
* resolved. | ||
* | ||
* @param {string} id - The reserved ID for the audio. | ||
* @param {MediaRecorder|Recorder} recorder - The recorder which has been set up to | ||
* record. | ||
* @param {string} rpc - The RPC to use to initialize the websocket server. | ||
* | ||
* @emits {websocketserverreadyforaudio} - When the websocket server has been | ||
* prepared for and is ready to receive | ||
* the audio. | ||
* | ||
* @returns {Promise} - The promise which resolves when the websocket server | ||
* is ready for the audio. | ||
*/ | ||
/** | ||
* This file contains some re-usable parts for websocket audio communication. | ||
*/ | ||
function prepareServerForAudio(id, recorder, rpc) { | ||
}function prepareServerForAudio(id, recorder, rpc) { | ||
var _recorder$getAudioSpe = recorder.getAudioSpecs(), | ||
@@ -131,2 +168,2 @@ audioFormat = _recorder$getAudioSpe.audioFormat, | ||
} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9hcGkvdXRpbHMvYXVkaW8tb3Zlci1zb2NrZXQuanMiXSwibmFtZXMiOlsiZW5jb2RlQW5kU2VuZEF1ZGlvT25EYXRhQXZhaWxpYmxlIiwicHJlcGFyZVNlcnZlckZvckF1ZGlvIiwid2FpdEZvclVzZXJNZWRpYUFwcHJvdmFsIiwiaWQiLCJyZWNvcmRlciIsInJwYyIsIlByb21pc2UiLCJyZXNvbHZlIiwicmVqZWN0IiwiYWRkRXZlbnRMaXN0ZW5lciIsImVuY29kZWQiLCJjaHVuayIsImFyZ3MiLCJ0aGVuIiwiZ2V0QXVkaW9TcGVjcyIsImF1ZGlvRm9ybWF0IiwiYXVkaW9QYXJhbWV0ZXJzIiwia3dhcmdzIiwiZW1pdCIsImhhc1VzZXJNZWRpYUFwcHJvdmFsIl0sIm1hcHBpbmdzIjoiOzs7UUFrQmdCQSxpQyxHQUFBQSxpQztRQTREQUMscUIsR0FBQUEscUI7UUF3QkFDLHdCLEdBQUFBLHdCOztBQWxHaEI7Ozs7QUFDQTs7QUFDQTs7OztBQUdBOzs7Ozs7Ozs7QUFTTyxTQUFTRixpQ0FBVCxDQUEyQ0csRUFBM0MsRUFBK0NDLFFBQS9DLEVBQXlEQyxHQUF6RCxFQUE4RDtBQUNuRSxTQUFPLElBQUlDLE9BQUosQ0FBWSxVQUFDQyxPQUFELEVBQVVDLE1BQVYsRUFBcUI7QUFDdEM7QUFDQTtBQUNBSixhQUFTSyxnQkFBVCxDQUEwQixlQUExQixFQUEyQyxpQkFBUztBQUNsRCxVQUFNQyxVQUFVLHlCQUFhQyxLQUFiLENBQWhCO0FBQ0Esd0NBQWtCTixHQUFsQixFQUF1QixFQUFDTyxNQUFNLENBQUNULEVBQUQsRUFBS08sT0FBTCxFQUFjLFFBQWQsQ0FBUCxFQUF2QixFQUNHRyxJQURILENBQ1FOLE9BRFIsRUFDaUJDLE1BRGpCO0FBRUQsS0FKRDtBQUtELEdBUk0sQ0FBUDtBQVNEOztBQUVEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUE4QkE7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQTVEQTs7OztBQThFTyxTQUFTUCxxQkFBVCxDQUErQkUsRUFBL0IsRUFBbUNDLFFBQW5DLEVBQTZDQyxHQUE3QyxFQUFrRDtBQUFBLDhCQUNoQkQsU0FBU1UsYUFBVCxFQURnQjtBQUFBLE1BQ2hEQyxXQURnRCx5QkFDaERBLFdBRGdEO0FBQUEsTUFDbkNDLGVBRG1DLHlCQUNuQ0EsZUFEbUM7O0FBRXZELFNBQU8sa0NBQWtCWCxHQUFsQixFQUF1QixFQUFDTyxNQUFNLENBQUNULEVBQUQsRUFBS1ksV0FBTCxDQUFQLEVBQTBCRSxRQUFRRCxlQUFsQyxFQUF2QixFQUNKSCxJQURJLENBQ0MsWUFBTTtBQUNWO0FBQ0E7QUFDQSwwQkFBWUssSUFBWixDQUFpQiw4QkFBakI7QUFDQSxXQUFPZixFQUFQO0FBQ0QsR0FOSSxDQUFQO0FBT0Q7O0FBR0Q7Ozs7Ozs7Ozs7OztBQVlPLFNBQVNELHdCQUFULENBQWtDQyxFQUFsQyxFQUFzQ0MsUUFBdEMsRUFBZ0Q7QUFDckQsU0FBTyxJQUFJRSxPQUFKLENBQVksbUJBQVc7QUFDNUI7QUFDQTtBQUNBLFFBQUlGLFNBQVNlLG9CQUFULEVBQUosRUFBcUM7QUFDbkNaO0FBQ0QsS0FGRCxNQUVPO0FBQ0xILGVBQVNLLGdCQUFULENBQTBCLE9BQTFCLEVBQW1DRixPQUFuQztBQUNEO0FBQ0YsR0FSTSxFQVFKTSxJQVJJLENBUUM7QUFBQSxXQUFNVixFQUFOO0FBQUEsR0FSRCxDQUFQO0FBU0QiLCJmaWxlIjoiYXVkaW8tb3Zlci1zb2NrZXQuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFRoaXMgZmlsZSBjb250YWlucyBzb21lIHJlLXVzYWJsZSBwYXJ0cyBmb3Igd2Vic29ja2V0IGF1ZGlvIGNvbW11bmljYXRpb24uXG4gKi9cblxuaW1wb3J0IGJyb2FkY2FzdGVyIGZyb20gJy4uL2Jyb2FkY2FzdGVyJztcbmltcG9ydCB7ZGF0YVRvQmFzZTY0fSBmcm9tICcuL2luZGV4JztcbmltcG9ydCB7bWFrZVdlYnNvY2tldENhbGx9IGZyb20gJy4uL2NvbW11bmljYXRpb24vd2Vic29ja2V0JztcblxuXG4vKipcbiAqIEVuY29kZSB0aGUgYXVkaW8gYXMgYmFzZTY0IGFuZCBzZW5kIGl0IHRvIHRoZSB3ZWJzb2NrZXQgc2VydmVyLlxuICpcbiAqIEBwYXJhbSB7c3RyaW5nfSBpZCAtIFRoZSByZXNlcnZlZCBJRCBmb3IgdGhlIGF1ZGlvLlxuICogQHBhcmFtIHtNZWRpYVJlY29yZGVyfFJlY29yZGVyfSByZWNvcmRlciAtIFRoZSByZWNvcmRlciB0byB1c2UgdG8gZ2V0IHRoZSByZWNvcmRpbmcuXG4gKiBAcGFyYW0ge3N0cmluZ30gcnBjIC0gVGhlIFJQQyB0byB1c2UgdG8gc3RvcmUgdGhlIGRhdGEuXG4gKlxuICogQHJldHVybnMge1Byb21pc2UuPCo+fSAtIFRoZSByZXNwb25zZSBvZiB0aGUgZ2l2ZW4gUlBDLlxuICovXG5leHBvcnQgZnVuY3Rpb24gZW5jb2RlQW5kU2VuZEF1ZGlvT25EYXRhQXZhaWxpYmxlKGlkLCByZWNvcmRlciwgcnBjKSB7XG4gIHJldHVybiBuZXcgUHJvbWlzZSgocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XG4gICAgLy8gV2hlbiB0aGUgYXVkaW8gaXMgZG9uZSByZWNvcmRpbmc6IGVuY29kZSB0aGUgZGF0YSwgc2VuZCBpdCB0byB0aGVcbiAgICAvLyB3ZWJzb2NrZXQgc2VydmVyIGFuZCBjb250aW51ZSB3aXRoIHRoZSBjaGFpbi5cbiAgICByZWNvcmRlci5hZGRFdmVudExpc3RlbmVyKCdkYXRhYXZhaWxhYmxlJywgY2h1bmsgPT4ge1xuICAgICAgY29uc3QgZW5jb2RlZCA9IGRhdGFUb0Jhc2U2NChjaHVuayk7XG4gICAgICBtYWtlV2Vic29ja2V0Q2FsbChycGMsIHthcmdzOiBbaWQsIGVuY29kZWQsICdiYXNlNjQnXX0pXG4gICAgICAgIC50aGVuKHJlc29sdmUsIHJlamVjdCk7XG4gICAgfSk7XG4gIH0pO1xufVxuXG4vKlxuICogbmV4dCB0byBzZW5kaW5nIGF1ZG8gYXMgZW5jb2RlZCBjaHVua3MsIHdlIGNhbiBhbHNvIHByZXBhcmUgYSBycGNcbiAqIHRoYXQnbGwgZmV0Y2ggdGhlIHJlc3VsdCBmb3IgdXMuXG4gKiBCZWxvdyBpcyBhbiBleGFtcGxlIG9mIGhvdyB0aGlzIGNvdWxkIGxvb2sgbGlrZS5cbiAqIFJlbWVtYmVyIHRvIGltcG9ydCBnZXRXZWJzb2NrZXQgYW5kIGF1dGhvYmFobiBmb3IgdGhpcyB0byB3b3JrIVxuICpcbiAqIGV4cG9ydCBmdW5jdGlvbiByZWdpc3RlckF1ZGlvU3RyZWFtKHJlY29yZGVyLCBycGNOYW1lKSB7XG4gKiAgIGNvbnN0IHJwYyA9IGBubC5pdHNsYW5ndWFnZS4ke3JwY05hbWV9YDtcbiAqXG4gKiAgIGNvbnN0IHNlbmRTb21lQ2h1Y2tzID0gKCkgPT4ge1xuICogICAgIGNvbnN0IGRlZmVyID0gbmV3IGF1dG9iYWhuLndoZW4uZGVmZXIoKTtcbiAqXG4gKiAgICAgcmVjb3JkZXIuYWRkRXZlbnRMaXN0ZW5lcignZGF0YWF2YWlsYWJsZScsIGNodWNrID0+IHtcbiAqICAgICAgIGNvbnNvbGUubG9nKGNodWNrKTtcbiAqICAgICAgIGRlZmVyLm5vdGlmeShjaHVjayk7XG4gKiAgICAgfSk7XG4gKlxuICogICAgIHJlY29yZGVyLmFkZEV2ZW50TGlzdGVuZXIoJ3JlY29yZGVkJywgKCkgPT4ge1xuICogICAgICAgZGVmZXIucmVzb2x2ZSgnc3RyZWFtIGVuZGVkJyk7XG4gKiAgICAgfSk7XG4gKlxuICogICAgIHJldHVybiBkZWZlci5wcm9taXNlO1xuICogICB9O1xuICpcbiAqICAgcmV0dXJuIGdldFdlYnNvY2tldENvbm5lY3Rpb24oKS50aGVuKGNvbm5lY3Rpb24gPT5cbiAqICAgICBjb25uZWN0aW9uLnNlc3Npb24ucmVnaXN0ZXIocnBjLCBzZW5kU29tZUNodWNrcylcbiAqICAgKTtcbiAqIH1cbiAqL1xuXG4vKipcbiAqIFNlbmQgdGhlIHJlY29yZGVyIHNldHRpbmdzIHRvIHRoZSB3ZWJzb2NrZXQgc2VydmVyIHRvIGluaXRpYWxpemUgaXQuXG4gKlxuICogVGhlIHJlc2VydmVkIElEIChwYXNzZWQgaW4gdGhlIHBhcmFtZXRlcnMpIGlzIHJldHVybmVkIG9uY2UgdGhlIHByb21pc2UgaXNcbiAqIHJlc29sdmVkLlxuICpcbiAqIEBwYXJhbSB7c3RyaW5nfSBpZCAtIFRoZSByZXNlcnZlZCBJRCBmb3IgdGhlIGF1ZGlvLlxuICogQHBhcmFtIHtNZWRpYVJlY29yZGVyfFJlY29yZGVyfSByZWNvcmRlciAtIFRoZSByZWNvcmRlciB3aGljaCBoYXMgYmVlbiBzZXQgdXAgdG9cbiAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZWNvcmQuXG4gKiBAcGFyYW0ge3N0cmluZ30gcnBjIC0gVGhlIFJQQyB0byB1c2UgdG8gaW5pdGlhbGl6ZSB0aGUgd2Vic29ja2V0IHNlcnZlci5cbiAqXG4gKiBAZW1pdHMge3dlYnNvY2tldHNlcnZlcnJlYWR5Zm9yYXVkaW99IC0gV2hlbiB0aGUgd2Vic29ja2V0IHNlcnZlciBoYXMgYmVlblxuICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHByZXBhcmVkIGZvciBhbmQgaXMgcmVhZHkgdG8gcmVjZWl2ZVxuICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoZSBhdWRpby5cbiAqXG4gKiBAcmV0dXJucyB7UHJvbWlzZX0gLSBUaGUgcHJvbWlzZSB3aGljaCByZXNvbHZlcyB3aGVuIHRoZSB3ZWJzb2NrZXQgc2VydmVyXG4gKiAgICAgICAgICAgICAgICAgICAgICBpcyByZWFkeSBmb3IgdGhlIGF1ZGlvLlxuICovXG5leHBvcnQgZnVuY3Rpb24gcHJlcGFyZVNlcnZlckZvckF1ZGlvKGlkLCByZWNvcmRlciwgcnBjKSB7XG4gIGNvbnN0IHthdWRpb0Zvcm1hdCwgYXVkaW9QYXJhbWV0ZXJzfSA9IHJlY29yZGVyLmdldEF1ZGlvU3BlY3MoKTtcbiAgcmV0dXJuIG1ha2VXZWJzb2NrZXRDYWxsKHJwYywge2FyZ3M6IFtpZCwgYXVkaW9Gb3JtYXRdLCBrd2FyZ3M6IGF1ZGlvUGFyYW1ldGVyc30pXG4gICAgLnRoZW4oKCkgPT4ge1xuICAgICAgLy8gV2UndmUgcHJlcGVkIHRoZSB3ZWJzb2NrZXQgc2VydmVyLCBub3cgaXQgY2FuIHJlY2VpdmUgYXVkaW8uIEJyb2FkY2FzdFxuICAgICAgLy8gdGhhdCBpdCBpcyBhbGxvd2VkIHRvIHJlY29yZC5cbiAgICAgIGJyb2FkY2FzdGVyLmVtaXQoJ3dlYnNvY2tldHNlcnZlcnJlYWR5Zm9yYXVkaW8nKTtcbiAgICAgIHJldHVybiBpZDtcbiAgICB9KTtcbn1cblxuXG4vKipcbiAqIFdhaXQgZm9yIHRoZSByZWNvcmRlciB0byBnZXQgdGhlIHBlcm1pc3Npb24gZm9yIHVzZXIgbWVkaWEuXG4gKlxuICogVGhlIHJlc2VydmVkIElEIChwYXNzZWQgaW4gdGhlIHBhcmFtZXRlcnMpIGlzIHJldHVybmVkIG9uY2UgdGhlIHByb21pc2UgaXNcbiAqIHJlc29sdmVkLlxuICpcbiAqIEBwYXJhbSB7c3RyaW5nfSBpZCAtIFRoZSByZXNlcnZlZCBJRCBmb3IgdGhlIGF1ZGlvLlxuICogQHBhcmFtIHtNZWRpYVJlY29yZGVyfSByZWNvcmRlciAtIFRoZSByZWNvcmRlciBmb3Igd2hpY2ggdG8gd2FpdC5cbiAqXG4gKiBAcmV0dXJucyB7UHJvbWlzZX0gLSBUaGUgcHJvbWlzZSB3aGljaCByZXNvbHZlcyBpZiB0aGUgdXNlciBoYXMgYWxsb3dlZCB1c1xuICogICAgICAgICAgICAgICAgICAgICAgdG8gcmVjb3JkIHRoZW0uXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiB3YWl0Rm9yVXNlck1lZGlhQXBwcm92YWwoaWQsIHJlY29yZGVyKSB7XG4gIHJldHVybiBuZXcgUHJvbWlzZShyZXNvbHZlID0+IHtcbiAgICAvLyBXZSBuZWVkIHRoZSB1c2VyJ3MgcGVybWlzc2lvbiBpbiBvcmRlciB0byByZWNvcmQgdGhlIGF1ZGlvLiBXYWl0IGZvclxuICAgIC8vIGl0IGlmIHdlIGRvbid0IGhhdmUgaXQgYWxyZWFkeS5cbiAgICBpZiAocmVjb3JkZXIuaGFzVXNlck1lZGlhQXBwcm92YWwoKSkge1xuICAgICAgcmVzb2x2ZSgpO1xuICAgIH0gZWxzZSB7XG4gICAgICByZWNvcmRlci5hZGRFdmVudExpc3RlbmVyKCdyZWFkeScsIHJlc29sdmUpO1xuICAgIH1cbiAgfSkudGhlbigoKSA9PiBpZCk7XG59XG4iXX0= | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/api/utils/audio-over-socket.js"],"names":["recorder","rpcName","asyncChucks","args","kwargs","details","defer","when","progress","addEventListener","dataToSend","Array","from","Uint8Array","chuck","resolve","rpcRegistration","console","log","then","connection","session","unregister","promise","rpc","register","registration","registerAudioStream","encodeAndSendAudioOnDataAvailible","prepareServerForAudio","waitForUserMediaApproval","id","Promise","reject","encoded","chunk","getAudioSpecs","audioFormat","audioParameters","emit","hasUserMediaApproval"],"mappings":"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","file":"audio-over-socket.js","sourcesContent":["/**\n * This file contains some re-usable parts for websocket audio communication.\n */\n\nimport autobahn from 'autobahn';\nimport broadcaster from '../broadcaster';\nimport {dataToBase64} from './index';\nimport {getWebsocketConnection, makeWebsocketCall} from '../communication/websocket';\n\n\n/**\n * Encode the audio as base64 and send it to the websocket server.\n *\n * @param {string} id - The reserved ID for the audio.\n * @param {MediaRecorder|Recorder} recorder - The recorder to use to get the recording.\n * @param {string} rpc - The RPC to use to store the data.\n *\n * @returns {Promise.<*>} - The response of the given RPC.\n */\nexport function encodeAndSendAudioOnDataAvailible(id, recorder, rpc) {\n  return new Promise((resolve, reject) => {\n    // When the audio is done recording: encode the data, send it to the\n    // websocket server and continue with the chain.\n    recorder.addEventListener('dataavailable', chunk => {\n      const encoded = dataToBase64(chunk);\n      makeWebsocketCall(rpc, {args: [id, encoded, 'base64']})\n        .then(resolve, reject);\n    });\n  });\n}\n\n/*\n * next to sending audo as encoded chunks, we can also prepare a rpc\n * that'll fetch the result for us.\n * Below is an example of how this could look like.\n * Remember to import getWebsocket and authobahn for this to work!\n */\nexport async function registerAudioStream(recorder, rpcName) {\n  const rpc = `nl.itslanguage.${rpcName}`;\n  let rpcRegistration = null;\n\n  function asyncChucks(args, kwargs, details) {\n    // eslint-disable-next-line new-cap\n    const defer = new autobahn.when.defer();\n\n    if (details.progress) {\n      recorder.addEventListener('dataavailable', chuck => {\n        const dataToSend = Array.from(new Uint8Array(chuck));\n        details.progress([dataToSend]);\n      });\n\n      recorder.addEventListener('recorded', () => {\n        defer.resolve('Done with streaming you basterd');\n        if (rpcRegistration) {\n          console.log('yeash!');\n          getWebsocketConnection().then(connection =>\n            connection.session.unregister(rpcRegistration).then(\n              () => {\n                console.log('succesfully un-registered');\n              }\n            )\n          );\n        }\n      });\n    }\n\n    return defer.promise;\n  }\n\n  return getWebsocketConnection().then(connection =>\n    connection.session.register(rpc, asyncChucks).then(registration => {\n      rpcRegistration = registration;\n      return registration;\n    })\n  );\n}\n\n/**\n * Send the recorder settings to the websocket server to initialize it.\n *\n * The reserved ID (passed in the parameters) is returned once the promise is\n * resolved.\n *\n * @param {string} id - The reserved ID for the audio.\n * @param {MediaRecorder|Recorder} recorder - The recorder which has been set up to\n *                                   record.\n * @param {string} rpc - The RPC to use to initialize the websocket server.\n *\n * @emits {websocketserverreadyforaudio} - When the websocket server has been\n *                                         prepared for and is ready to receive\n *                                         the audio.\n *\n * @returns {Promise} - The promise which resolves when the websocket server\n *                      is ready for the audio.\n */\nexport function prepareServerForAudio(id, recorder, rpc) {\n  const {audioFormat, audioParameters} = recorder.getAudioSpecs();\n  return makeWebsocketCall(rpc, {args: [id, audioFormat], kwargs: audioParameters})\n    .then(() => {\n      // We've preped the websocket server, now it can receive audio. Broadcast\n      // that it is allowed to record.\n      broadcaster.emit('websocketserverreadyforaudio');\n      return id;\n    });\n}\n\n\n/**\n * Wait for the recorder to get the permission for user media.\n *\n * The reserved ID (passed in the parameters) is returned once the promise is\n * resolved.\n *\n * @param {string} id - The reserved ID for the audio.\n * @param {MediaRecorder} recorder - The recorder for which to wait.\n *\n * @returns {Promise} - The promise which resolves if the user has allowed us\n *                      to record them.\n */\nexport function waitForUserMediaApproval(id, recorder) {\n  return new Promise(resolve => {\n    // We need the user's permission in order to record the audio. Wait for\n    // it if we don't have it already.\n    if (recorder.hasUserMediaApproval()) {\n      resolve();\n    } else {\n      recorder.addEventListener('ready', resolve);\n    }\n  }).then(() => id);\n}\n"]} |
@@ -5,6 +5,2 @@ 'use strict'; | ||
var _mediaRecorder = require('./media-recorder'); | ||
var _mediaRecorder2 = _interopRequireDefault(_mediaRecorder); | ||
var _tools = require('./tools'); | ||
@@ -67,5 +63,26 @@ | ||
this._stopwatch = null; | ||
if (options.audioContext) { | ||
this.audioContext = options.audioContext; | ||
} else { | ||
this.audioContext = this.createAudioContext(); | ||
} | ||
} | ||
/** | ||
* Get the audio context or create one. | ||
* | ||
* @return {AudioContext} The AudioContext created will be returned | ||
*/ | ||
AudioRecorder.prototype.createAudioContext = function createAudioContext() { | ||
if (!window.ItslAudioContext) { | ||
window.AudioContext = window.AudioContext || window.webkitAudioContext; | ||
window.ItslAudioContext = new window.AudioContext(); | ||
} | ||
return window.ItslAudioContext; | ||
}; | ||
/** | ||
* Turn off all event listeners for this recorder. | ||
@@ -140,12 +157,2 @@ */ | ||
AudioRecorder.prototype._recordingCompatibility /* istanbul ignore next */ = function _recordingCompatibility() { | ||
// Detect audio recording capabilities. | ||
// http://caniuse.com/#feat=stream | ||
// 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); | ||
// https://developer.mozilla.org/en-US/docs/Web/API/MediaDevices/mediaDevices.getUserMedia | ||
@@ -159,16 +166,2 @@ this.canMediaDevicesGetUserMedia = false; | ||
// Detect MediaStream Recording | ||
// It allows recording audio using the MediaStream from the above | ||
// getUserMedia directly with a native codec better than Wave. | ||
// http://www.w3.org/TR/mediastream-recording/ | ||
// this.canUseMediaRecorder = Boolean(window.MediaRecorder); | ||
// console.log('Native MediaRecorder recording capability:', this.canUseMediaRecorder); | ||
// Web Audio API | ||
// High-level JavaScript API for processing and synthesizing audio | ||
// http://caniuse.com/#feat=audio-api | ||
window.AudioContext = window.AudioContext || window.webkitAudioContext || window.mozAudioContext; | ||
var canCreateAudioContext = Boolean(window.AudioContext); | ||
console.log('Native Web Audio API (AudioContext) processing capability:', canCreateAudioContext); | ||
if (!this.canGetUserMedia && !this.canMediaDevicesGetUserMedia) { | ||
@@ -197,30 +190,18 @@ throw new Error('Some form of audio recording capability is required'); | ||
AudioRecorder.prototype.requestUserMedia = function requestUserMedia() { | ||
var self = this; | ||
function success(stream) { | ||
console.log('Got getUserMedia stream'); | ||
var _this = this; | ||
// checking audio presence | ||
if (self.canMediaDevicesGetUserMedia) { | ||
if (stream.getAudioTracks().length) { | ||
console.log('Got audio tracks:', stream.getAudioTracks().length); | ||
} | ||
} | ||
var readyForStream = function readyForStream(stream) { | ||
// Modify state of userMediaApproval now access is granted. | ||
self.userMediaApproval = true; | ||
_this.userMediaApproval = true; | ||
var micInputGain = self._startUserMedia(stream); | ||
self.fireEvent('ready', [self.audioContext, micInputGain]); | ||
} | ||
function failure(e) { | ||
console.log(e); | ||
var micInputGain = _this._startUserMedia(stream); | ||
_this.fireEvent('ready', [_this.audioContext, micInputGain]); | ||
}; | ||
var userCanceled = function userCanceled(error) { | ||
console.error(error); | ||
throw new Error('No live audio input available or permitted'); | ||
} | ||
}; | ||
if (this.canMediaDevicesGetUserMedia) { | ||
// Use of promises is required. | ||
navigator.mediaDevices.getUserMedia({ audio: true }).then(success).catch(failure); | ||
} else if (this.canGetUserMedia) { | ||
navigator.getUserMedia({ audio: true }, success, failure); | ||
} | ||
navigator.mediaDevices.getUserMedia({ audio: true }).then(readyForStream).catch(userCanceled); | ||
}; | ||
@@ -237,12 +218,2 @@ | ||
AudioRecorder.prototype._startUserMedia = function _startUserMedia(stream) { | ||
if (!this.audioContext) { | ||
// Initialize the context once, and only when getUserMedia was | ||
// successful. | ||
this.audioContext = new window.AudioContext(); | ||
} | ||
if (!this.audioContext.createMediaStreamSource) { | ||
throw new Error('AudioContext has no property createMediaStreamSource'); | ||
} | ||
// Creates an audio node from the microphone incoming stream. | ||
@@ -280,19 +251,7 @@ var micInput = this.audioContext.createMediaStreamSource(stream); | ||
AudioRecorder.prototype._getBestRecorder = function _getBestRecorder(micInputGain) { | ||
var recorder = null; | ||
// 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. | ||
var self = this; | ||
recorder = new _webAudioRecorder2.default(micInputGain, function (data) { | ||
self.streamCallback(data); | ||
}, new _wavePacker2.default()); | ||
} else { | ||
throw new Error('Unable to find a proper recorder.'); | ||
} | ||
console.log('Recorder initialised.'); | ||
return recorder; | ||
var _this2 = this; | ||
return new _webAudioRecorder2.default(micInputGain, this.audioContext, function (data) { | ||
_this2.streamCallback(data); | ||
}, new _wavePacker2.default(), false); | ||
}; | ||
@@ -361,2 +320,4 @@ | ||
this.audioContext.resume(); | ||
this._recorder.record(); | ||
@@ -459,2 +420,2 @@ if (this._stopwatch) { | ||
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","_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":"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","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"]} | ||
//# 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","audioContext","createAudioContext","window","ItslAudioContext","AudioContext","webkitAudioContext","removeAllEventListeners","addEventListener","name","handler","on","removeEventListener","off","fireEvent","args","emit","hasUserMediaApproval","canMediaDevicesGetUserMedia","navigator","mediaDevices","getUserMedia","webkitGetUserMedia","mozGetUserMedia","Boolean","console","log","canGetUserMedia","Error","URL","webkitURL","hasWindowURL","requestUserMedia","readyForStream","micInputGain","_startUserMedia","stream","userCanceled","error","audio","then","catch","micInput","createMediaStreamSource","createGain","connect","_getBestRecorder","streamCallback","data","chunk","_requireGetUserMedia","startRecordingSession","id","uuid_","undefined","v4","activeRecordingId","record","cb","isRecording","resume","_value","start","stop","forced","self","getEncodedAudio","blob","type","toggleRecording","getAudioSpecs","bindStopwatch","tickCb"],"mappings":"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","file":"audio-recorder.js","sourcesContent":["import 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    if (options.audioContext) {\n      this.audioContext = options.audioContext;\n    } else {\n      this.audioContext = this.createAudioContext();\n    }\n  }\n\n\n  /**\n   * Get the audio context or create one.\n   *\n   * @return {AudioContext} The AudioContext created will be returned\n   */\n  createAudioContext() {\n    if (!window.ItslAudioContext) {\n      window.AudioContext =\n        window.AudioContext || window.webkitAudioContext;\n      window.ItslAudioContext = new window.AudioContext();\n    }\n    return window.ItslAudioContext;\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    // 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    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 readyForStream = stream => {\n      // Modify state of userMediaApproval now access is granted.\n      this.userMediaApproval = true;\n\n      const micInputGain = this._startUserMedia(stream);\n      this.fireEvent('ready', [this.audioContext, micInputGain]);\n    };\n\n    const userCanceled = error => {\n      console.error(error);\n      throw new Error('No live audio input available or permitted');\n    };\n\n    navigator.mediaDevices.getUserMedia({audio: true})\n      .then(readyForStream)\n      .catch(userCanceled);\n  }\n\n  /**\n   * Audio access was granted, start analysing.\n   *\n   * @param {MediaStream} stream - Media Stream.\n   * @private\n   */\n  _startUserMedia(stream) {\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    return new WebAudioRecorder(micInputGain, this.audioContext, data => {\n      this.streamCallback(data);\n    }, new WavePacker(), false);\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.audioContext.resume();\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"]} |
@@ -8,2 +8,35 @@ 'use strict'; | ||
/** | ||
* Util function to concat two ArrayBuffers. | ||
* | ||
* @param {[ArrayBuffer]} buffers - Array with buffers to concat. | ||
* | ||
* @return {ArrayBuffer} - Concatenated ArrayBuffer | ||
*/ | ||
function concatArrayBuffers() { | ||
// Determine the length | ||
var length = 0; | ||
for (var _len = arguments.length, buffers = Array(_len), _key = 0; _key < _len; _key++) { | ||
buffers[_key] = arguments[_key]; | ||
} | ||
buffers.forEach(function (buffer) { | ||
length += buffer.byteLength; | ||
}); | ||
// Prepare the concatenated Array | ||
var result = new Uint8Array(length); | ||
// Loop through all items and put them in the result | ||
var offset = 0; | ||
buffers.forEach(function (buffer) { | ||
result.set(buffer, offset); | ||
offset += buffer.byteLength; | ||
}); | ||
// Return the ArrayBuffer | ||
return result.buffer; | ||
} | ||
/** | ||
* Packer class for audio packing | ||
@@ -13,5 +46,14 @@ * | ||
*/ | ||
var WavePacker = function () { | ||
function WavePacker() { | ||
_classCallCheck(this, WavePacker); | ||
this.recordingSampleRate = null; | ||
this.sampleRate = null; | ||
this.channels = null; | ||
this.recording = false; | ||
this.recLength = 0; | ||
this.recBuffersL = []; | ||
this.recBuffersR = []; | ||
} | ||
@@ -62,2 +104,3 @@ | ||
} | ||
// Both the left and right channel's data is a view (Float32Array) | ||
@@ -68,2 +111,4 @@ // on top of the buffer (ArrayBuffer). Each buffer's element should | ||
// a WAVE file at the server. Therefore convert from float here. | ||
// We choose only one channel (left) because the backend can only | ||
// process mono sound. Left is just a random choice then... | ||
var converted = convertFloat32ToInt16(left); | ||
@@ -216,2 +261,2 @@ callback(converted); | ||
exports.default = WavePacker; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/audio/wave-packer.js"],"names":["WavePacker","init","recordingSampleRate","sampleRate","channels","indexOf","Error","recording","clear","recLength","recBuffersL","recBuffersR","record","left","right","push","length","recordStreaming","callback","convertFloat32ToInt16","buffer","l","buf","Int16Array","Math","min","converted","exportWAV","bufferL","mergeBuffers","bufferR","interleaved","interleave","dataview","encodeWAV","audioBlob","Blob","type","exportMonoWAV","ArrayBuffer","view","DataView","writeUTFBytes","setUint32","setUint16","lng","index","volume","i","setInt16","blob","console","log","size","leftChannel","rightChannel","result","inputIndex","Float32Array","reduceBy","resampledResult","value","j","channelBuffer","recordingLength","offset","set","string","setUint8","charCodeAt"],"mappings":"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","file":"wave-packer.js","sourcesContent":["/**\n * Packer class for audio packing\n *\n * @private\n */\nexport default class WavePacker {\n  /**\n   * Stop recording audio.\n   *\n   * @param {number} recordingSampleRate - Sample rate of recording. Must be either 48000 or 44100.\n   * @param {number} sampleRate - Sample rate. Must be half or a quarter of the recording sample rate.\n   * @param {number} channels - Amount of audio channels. 1 or 2.\n   */\n  init(recordingSampleRate, sampleRate, channels) {\n    this.recordingSampleRate = recordingSampleRate;\n    if ([48000, 44100].indexOf(this.recordingSampleRate) === -1) {\n      throw new Error(\n        '48000 or 44100 are the only supported recordingSampleRates');\n    }\n\n    this.sampleRate = sampleRate;\n    if ([\n      this.recordingSampleRate,\n      this.recordingSampleRate / 2,\n      this.recordingSampleRate / 4\n    ].indexOf(this.sampleRate) === -1) {\n      throw new Error(\n        'sampleRate must be equal, half or a quarter of the ' +\n        'recording sample rate');\n    }\n\n    this.channels = channels;\n    this.recording = false;\n  }\n\n  clear() {\n    this.recLength = 0;\n    this.recBuffersL = [];\n    this.recBuffersR = [];\n  }\n\n  record(left, right) {\n    this.recBuffersL.push(left);\n    this.recBuffersR.push(right);\n    this.recLength += left.length;\n  }\n\n  recordStreaming(left, right, callback) {\n    function convertFloat32ToInt16(buffer) {\n      let l = buffer.length;\n      const buf = new Int16Array(l);\n      while (l--) {\n        buf[l] = Math.min(1, buffer[l]) * 0x7FFF;\n      }\n      return buf.buffer;\n    }\n    // Both the left and right channel's data is a view (Float32Array)\n    // on top of the buffer (ArrayBuffer). Each buffer's element should\n    // have _value between -1 and 1.\n    // The audio to export are 16 bit PCM samples that are wrapped in\n    // a WAVE file at the server. Therefore convert from float here.\n    const converted = convertFloat32ToInt16(left);\n    callback(converted);\n  }\n\n  exportWAV(callback) {\n    const bufferL = WavePacker.mergeBuffers(this.recBuffersL, this.recLength);\n    const bufferR = WavePacker.mergeBuffers(this.recBuffersR, this.recLength);\n    const interleaved = this.interleave(bufferL, bufferR);\n    const dataview = this.encodeWAV(interleaved);\n    const audioBlob = new Blob([dataview], {\n      type: 'audio/wav'\n    });\n    callback(audioBlob);\n  }\n\n\n  exportMonoWAV(callback) {\n    const bufferL = WavePacker.mergeBuffers(this.recBuffersL, this.recLength);\n    const dataview = this.encodeWAV(bufferL, true);\n    const audioBlob = new Blob([dataview], {\n      type: 'audio/wav'\n    });\n    callback(audioBlob);\n  }\n\n  /**\n   * Wrap the raw audio in a header to make it a WAVE format.\n   *\n   * Specs: {@link https://ccrma.stanford.edu/courses/422/projects/WaveFormat/}.\n   *\n   * @param {[]} interleaved - Array of interleaved audio.\n   */\n  encodeWAV(interleaved) {\n    const buffer = new ArrayBuffer(44 + interleaved.length * 2);\n    const view = new DataView(buffer);\n\n    // RIFF chunk descriptor\n    WavePacker.writeUTFBytes(view, 0, 'RIFF');\n    // file length\n    view.setUint32(4, 44 + interleaved.length * 2, true);\n    // RIFF type\n    WavePacker.writeUTFBytes(view, 8, 'WAVE');\n    // FMT sub-chunk\n    WavePacker.writeUTFBytes(view, 12, 'fmt ');\n    // format chunk length\n    view.setUint32(16, 16, true);\n    // sample format (raw)\n    view.setUint16(20, 1, true);\n    // channel count. mono=1, stereo=2\n    view.setUint16(22, this.channels, true);\n    // sample rate\n    view.setUint32(24, this.sampleRate, true);\n    // byte rate (sample rate * block align)\n    view.setUint32(28, this.sampleRate * 2 * this.channels, true);\n    // block align (channel count * bytes per sample)\n    view.setUint16(32, this.channels * 2, true);\n    // bits per sample\n    view.setUint16(34, 16, true);\n    // data sub-chunk\n    WavePacker.writeUTFBytes(view, 36, 'data');\n    view.setUint32(40, interleaved.length * 2, true);\n\n    // write the PCM samples\n    const lng = interleaved.length;\n    let index = 44;\n    const volume = 1;\n    for (let i = 0; i < lng; i++) {\n      view.setInt16(index, interleaved[i] * (0x7FFF * volume), true);\n      index += 2;\n    }\n\n    // Wrap in HTML5 Blob for transport\n    const blob = new Blob([view], {\n      type: 'audio/wav'\n    });\n    console.log('Recorded audio/wav Blob size: ' + blob.size);\n    return blob;\n  }\n\n  interleave(leftChannel, rightChannel) {\n    let result = null;\n    let length = null;\n    let i = null;\n    let inputIndex = null;\n    if (this.channels === 1) {\n      // Keep both right and left input channels, but \"pan\" them both\n      // in the center (to the single mono channel)\n      length = leftChannel.length;\n      result = new Float32Array(length);\n      for (i = 0; i < leftChannel.length; ++i) {\n        result[i] = 0.5 * (leftChannel[i] + rightChannel[i]);\n      }\n    } else {\n      length = leftChannel.length + rightChannel.length;\n      result = new Float32Array(length);\n\n      inputIndex = 0;\n      for (i = 0; i < length;) {\n        result[i++] = leftChannel[inputIndex];\n        result[i++] = rightChannel[inputIndex];\n        inputIndex++;\n      }\n    }\n\n    // Also downsample if needed.\n    if (this.recordingSampleRate !== this.sampleRate) {\n      // E.g. 44100/11025 = 4\n      const reduceBy = this.recordingSampleRate / this.sampleRate;\n      const resampledResult = new Float32Array(length / reduceBy);\n\n      inputIndex = 0;\n      for (i = 0; i < length;) {\n        let value = 0;\n        for (let j = 0; j < reduceBy; j++) {\n          value += result[inputIndex++];\n        }\n        resampledResult[i++] = 1 / reduceBy * value;\n      }\n      return resampledResult;\n    }\n    return result;\n  }\n\n  static mergeBuffers(channelBuffer, recordingLength) {\n    const result = new Float32Array(recordingLength);\n    let offset = 0;\n    const lng = channelBuffer.length;\n    for (let i = 0; i < lng; i++) {\n      const buffer = channelBuffer[i];\n      result.set(buffer, offset);\n      offset += buffer.length;\n    }\n    return result;\n  }\n\n  static writeUTFBytes(view, offset, string) {\n    const lng = string.length;\n    for (let i = 0; i < lng; i++) {\n      view.setUint8(offset + i, string.charCodeAt(i));\n    }\n  }\n}\n"]} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/audio/wave-packer.js"],"names":["concatArrayBuffers","length","buffers","forEach","buffer","byteLength","result","Uint8Array","offset","set","WavePacker","recordingSampleRate","sampleRate","channels","recording","recLength","recBuffersL","recBuffersR","init","indexOf","Error","clear","record","left","right","push","recordStreaming","callback","convertFloat32ToInt16","l","buf","Int16Array","Math","min","converted","exportWAV","bufferL","mergeBuffers","bufferR","interleaved","interleave","dataview","encodeWAV","audioBlob","Blob","type","exportMonoWAV","ArrayBuffer","view","DataView","writeUTFBytes","setUint32","setUint16","lng","index","volume","i","setInt16","blob","console","log","size","leftChannel","rightChannel","inputIndex","Float32Array","reduceBy","resampledResult","value","j","channelBuffer","recordingLength","string","setUint8","charCodeAt"],"mappings":"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","file":"wave-packer.js","sourcesContent":["/**\n * Util function to concat two ArrayBuffers.\n *\n * @param {[ArrayBuffer]} buffers - Array with buffers to concat.\n *\n * @return {ArrayBuffer} - Concatenated ArrayBuffer\n */\nfunction concatArrayBuffers(...buffers) {\n  // Determine the length\n  let length = 0;\n  buffers.forEach(buffer => {\n    length += buffer.byteLength;\n  });\n\n  // Prepare the concatenated Array\n  const result = new Uint8Array(length);\n\n  // Loop through all items and put them in the result\n  let offset = 0;\n  buffers.forEach(buffer => {\n    result.set(buffer, offset);\n    offset += buffer.byteLength;\n  });\n\n  // Return the ArrayBuffer\n  return result.buffer;\n}\n\n/**\n * Packer class for audio packing\n *\n * @private\n */\nexport default class WavePacker {\n  recordingSampleRate = null;\n  sampleRate = null;\n  channels = null;\n  recording = false;\n  recLength = 0;\n  recBuffersL = [];\n  recBuffersR = [];\n\n  /**\n   * Stop recording audio.\n   *\n   * @param {number} recordingSampleRate - Sample rate of recording. Must be either 48000 or 44100.\n   * @param {number} sampleRate - Sample rate. Must be half or a quarter of the recording sample rate.\n   * @param {number} channels - Amount of audio channels. 1 or 2.\n   */\n  init(recordingSampleRate, sampleRate, channels) {\n    this.recordingSampleRate = recordingSampleRate;\n    if ([48000, 44100].indexOf(this.recordingSampleRate) === -1) {\n      throw new Error(\n        '48000 or 44100 are the only supported recordingSampleRates');\n    }\n\n    this.sampleRate = sampleRate;\n    if ([\n      this.recordingSampleRate,\n      this.recordingSampleRate / 2,\n      this.recordingSampleRate / 4\n    ].indexOf(this.sampleRate) === -1) {\n      throw new Error(\n        'sampleRate must be equal, half or a quarter of the ' +\n        'recording sample rate');\n    }\n\n    this.channels = channels;\n    this.recording = false;\n  }\n\n  clear() {\n    this.recLength = 0;\n    this.recBuffersL = [];\n    this.recBuffersR = [];\n  }\n\n  record(left, right) {\n    this.recBuffersL.push(left);\n    this.recBuffersR.push(right);\n    this.recLength += left.length;\n  }\n\n  recordStreaming(left, right, callback) {\n    function convertFloat32ToInt16(buffer) {\n      let l = buffer.length;\n      const buf = new Int16Array(l);\n      while (l--) {\n        buf[l] = Math.min(1, buffer[l]) * 0x7FFF;\n      }\n      return buf.buffer;\n    }\n\n    // Both the left and right channel's data is a view (Float32Array)\n    // on top of the buffer (ArrayBuffer). Each buffer's element should\n    // have _value between -1 and 1.\n    // The audio to export are 16 bit PCM samples that are wrapped in\n    // a WAVE file at the server. Therefore convert from float here.\n    // We choose only one channel (left) because the backend can only\n    // process mono sound. Left is just a random choice then...\n    const converted = convertFloat32ToInt16(left);\n    callback(converted);\n  }\n\n  exportWAV(callback) {\n    const bufferL = WavePacker.mergeBuffers(this.recBuffersL, this.recLength);\n    const bufferR = WavePacker.mergeBuffers(this.recBuffersR, this.recLength);\n    const interleaved = this.interleave(bufferL, bufferR);\n    const dataview = this.encodeWAV(interleaved);\n    const audioBlob = new Blob([dataview], {\n      type: 'audio/wav'\n    });\n    callback(audioBlob);\n  }\n\n\n  exportMonoWAV(callback) {\n    const bufferL = WavePacker.mergeBuffers(this.recBuffersL, this.recLength);\n    const dataview = this.encodeWAV(bufferL, true);\n    const audioBlob = new Blob([dataview], {\n      type: 'audio/wav'\n    });\n    callback(audioBlob);\n  }\n\n  /**\n   * Wrap the raw audio in a header to make it a WAVE format.\n   *\n   * Specs: {@link https://ccrma.stanford.edu/courses/422/projects/WaveFormat/}.\n   *\n   * @param {[]} interleaved - Array of interleaved audio.\n   */\n  encodeWAV(interleaved) {\n    const buffer = new ArrayBuffer(44 + interleaved.length * 2);\n    const view = new DataView(buffer);\n\n    // RIFF chunk descriptor\n    WavePacker.writeUTFBytes(view, 0, 'RIFF');\n    // file length\n    view.setUint32(4, 44 + interleaved.length * 2, true);\n    // RIFF type\n    WavePacker.writeUTFBytes(view, 8, 'WAVE');\n    // FMT sub-chunk\n    WavePacker.writeUTFBytes(view, 12, 'fmt ');\n    // format chunk length\n    view.setUint32(16, 16, true);\n    // sample format (raw)\n    view.setUint16(20, 1, true);\n    // channel count. mono=1, stereo=2\n    view.setUint16(22, this.channels, true);\n    // sample rate\n    view.setUint32(24, this.sampleRate, true);\n    // byte rate (sample rate * block align)\n    view.setUint32(28, this.sampleRate * 2 * this.channels, true);\n    // block align (channel count * bytes per sample)\n    view.setUint16(32, this.channels * 2, true);\n    // bits per sample\n    view.setUint16(34, 16, true);\n    // data sub-chunk\n    WavePacker.writeUTFBytes(view, 36, 'data');\n    view.setUint32(40, interleaved.length * 2, true);\n\n    // write the PCM samples\n    const lng = interleaved.length;\n    let index = 44;\n    const volume = 1;\n    for (let i = 0; i < lng; i++) {\n      view.setInt16(index, interleaved[i] * (0x7FFF * volume), true);\n      index += 2;\n    }\n\n    // Wrap in HTML5 Blob for transport\n    const blob = new Blob([view], {\n      type: 'audio/wav'\n    });\n    console.log('Recorded audio/wav Blob size: ' + blob.size);\n    return blob;\n  }\n\n  interleave(leftChannel, rightChannel) {\n    let result = null;\n    let length = null;\n    let i = null;\n    let inputIndex = null;\n    if (this.channels === 1) {\n      // Keep both right and left input channels, but \"pan\" them both\n      // in the center (to the single mono channel)\n      length = leftChannel.length;\n      result = new Float32Array(length);\n      for (i = 0; i < leftChannel.length; ++i) {\n        result[i] = 0.5 * (leftChannel[i] + rightChannel[i]);\n      }\n    } else {\n      length = leftChannel.length + rightChannel.length;\n      result = new Float32Array(length);\n\n      inputIndex = 0;\n      for (i = 0; i < length;) {\n        result[i++] = leftChannel[inputIndex];\n        result[i++] = rightChannel[inputIndex];\n        inputIndex++;\n      }\n    }\n\n    // Also downsample if needed.\n    if (this.recordingSampleRate !== this.sampleRate) {\n      // E.g. 44100/11025 = 4\n      const reduceBy = this.recordingSampleRate / this.sampleRate;\n      const resampledResult = new Float32Array(length / reduceBy);\n\n      inputIndex = 0;\n      for (i = 0; i < length;) {\n        let value = 0;\n        for (let j = 0; j < reduceBy; j++) {\n          value += result[inputIndex++];\n        }\n        resampledResult[i++] = 1 / reduceBy * value;\n      }\n      return resampledResult;\n    }\n    return result;\n  }\n\n  static mergeBuffers(channelBuffer, recordingLength) {\n    const result = new Float32Array(recordingLength);\n    let offset = 0;\n    const lng = channelBuffer.length;\n    for (let i = 0; i < lng; i++) {\n      const buffer = channelBuffer[i];\n      result.set(buffer, offset);\n      offset += buffer.length;\n    }\n    return result;\n  }\n\n  static writeUTFBytes(view, offset, string) {\n    const lng = string.length;\n    for (let i = 0; i < lng; i++) {\n      view.setUint8(offset + i, string.charCodeAt(i));\n    }\n  }\n}\n"]} |
@@ -19,6 +19,7 @@ 'use strict'; | ||
* @param {GainNode} source - The source to record. | ||
* @param {AudioContext} ctx - The AudioContext to use. | ||
* @param {?Function} streamingCallback - The callback to deliver audio chunks to. | ||
* @param {WavePacker} packer - Packer to use. | ||
*/ | ||
function WebAudioRecorder(source, streamingCallback, packer) { | ||
function WebAudioRecorder(source, ctx, streamingCallback, packer) { | ||
_classCallCheck(this, WebAudioRecorder); | ||
@@ -28,3 +29,3 @@ | ||
var context = source.context; | ||
var context = ctx; | ||
// For the best quality, use the samplerate in which audio is recorded. | ||
@@ -61,8 +62,8 @@ this.recordedSampleRate = context.sampleRate; | ||
var self = this; | ||
recorder.addEventListener('audioprocess', function (e) { | ||
recorder.onaudioprocess = function (audioProcessing) { | ||
if (!self.recording) { | ||
return; | ||
} | ||
var left = e.inputBuffer.getChannelData(0); | ||
var right = e.inputBuffer.getChannelData(1); | ||
var left = audioProcessing.inputBuffer.getChannelData(0); | ||
var right = audioProcessing.inputBuffer.getChannelData(1); | ||
// These returned channel buffers are pointers to the current samples | ||
@@ -77,19 +78,3 @@ // coming in. Make a snapshot (clone). The webworkers can't serialize | ||
} | ||
}); | ||
// recorder.onaudioprocess = function(e) { | ||
// if (!self.recording) { | ||
// return; | ||
// } | ||
// const left = e.inputBuffer.getChannelData(0); | ||
// const right = e.inputBuffer.getChannelData(1); | ||
// // These returned channel buffers are pointers to the current samples | ||
// // coming in. Make a snapshot (clone). The webworkers can't serialize | ||
// // the pointers. Well, Chrome and FF could, but Edge can't. | ||
// const leftClone = new Float32Array(left); | ||
// const rightClone = new Float32Array(right); | ||
// self.packer.record(leftClone, rightClone); | ||
// if (streamingCallback) { | ||
// self.packer.recordStreaming(leftClone, rightClone, streamingCallback); | ||
// } | ||
// }; | ||
}; | ||
@@ -167,2 +152,2 @@ source.connect(recorder); | ||
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","addEventListener","left","e","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,gBAAT,CAA0B,cAA1B,EAA0C,aAAK;AAC7C,UAAI,CAACD,KAAKd,SAAV,EAAqB;AACnB;AACD;AACD,UAAMgB,OAAOC,EAAEC,WAAF,CAAcC,cAAd,CAA6B,CAA7B,CAAb;AACA,UAAMC,QAAQH,EAAEC,WAAF,CAAcC,cAAd,CAA6B,CAA7B,CAAd;AACA;AACA;AACA;AACA,UAAME,YAAY,IAAIC,YAAJ,CAAiBN,IAAjB,CAAlB;AACA,UAAMO,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;AAgBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEAD,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;;;;;kBAxIkBJ,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.addEventListener('audioprocess', 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    // 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","ctx","streamingCallback","packer","recording","context","recordedSampleRate","sampleRate","channels","console","log","getAudioSpecs","audioParameters","init","bufferSize","recorder","createScriptProcessor","_recorder","self","onaudioprocess","audioProcessing","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;;;;;;;;;;;AAWA,4BAAYC,MAAZ,EAAoBC,GAApB,EAAyBC,iBAAzB,EAA4CC,MAA5C,EAAoD;AAAA;;AAClD,SAAKC,SAAL,GAAiB,KAAjB;;AAEA,QAAMC,UAAUJ,GAAhB;AACA;AACA,SAAKK,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,eAAT,EAA0B;AAClD,UAAI,CAACF,KAAKd,SAAV,EAAqB;AACnB;AACD;AACD,UAAMiB,OAAOD,gBAAgBE,WAAhB,CAA4BC,cAA5B,CAA2C,CAA3C,CAAb;AACA,UAAMC,QAAQJ,gBAAgBE,WAAhB,CAA4BC,cAA5B,CAA2C,CAA3C,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;;AAiBAF,WAAO8B,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;;;;;kBAzHkBL,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 {AudioContext} ctx - The AudioContext to use.\n   * @param {?Function} streamingCallback - The callback to deliver audio chunks to.\n   * @param {WavePacker} packer - Packer to use.\n   */\n  constructor(source, ctx, streamingCallback, packer) {\n    this.recording = false;\n\n    const context = ctx;\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(audioProcessing) {\n      if (!self.recording) {\n        return;\n      }\n      const left = audioProcessing.inputBuffer.getChannelData(0);\n      const right = audioProcessing.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"]} |
{ | ||
"name": "itslanguage", | ||
"version": "3.0.0-beta-4.7", | ||
"version": "3.0.0-beta-4.8", | ||
"description": "The core JavaScript SDK for ITSLanguage.", | ||
@@ -5,0 +5,0 @@ "main": "index.js", |
@@ -126,3 +126,3 @@ 'use strict'; | ||
}); | ||
_context.next = 17; | ||
_context.next = 19; | ||
break; | ||
@@ -134,5 +134,7 @@ | ||
console.error(_context.t0); | ||
console.log('catchie catch'); | ||
this.error(_context.t0.name + ': ' + _context.t0.message); | ||
case 17: | ||
case 19: | ||
case 'end': | ||
@@ -218,2 +220,2 @@ return _context.stop(); | ||
exports.default = Player; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9XZWJBdWRpby9QbGF5ZXIuanMiXSwibmFtZXMiOlsiUGxheWVyIiwiYXVkaW9CdWZmZXIiLCJhdWRpb1NvdXJjZSIsImNyZWF0ZUJ1ZmZlclNvdXJjZSIsImRpc2Nvbm5lY3RCdWZmZXJTb3VyY2UiLCJhdWRpb0NvbnRleHQiLCJidWZmZXIiLCJjb25uZWN0IiwiZGVzdGluYXRpb24iLCJhZGRFdmVudExpc3RlbmVyIiwic3VzcGVuZEF1ZGlvQ29udGV4dCIsImF1ZGlvU291cmNlRXhpc3RzIiwiZGlzY29ubmVjdCIsInJlbW92ZUV2ZW50TGlzdGVuZXIiLCJsb2FkIiwidXJsIiwid2l0aEl0c2xUb2tlbiIsInJlcXVlc3RNZXRob2QiLCJyZXNwb25zZSIsImFycmF5QnVmZmVyIiwiYXVkaW9EYXRhIiwiZGVjb2RlQXVkaW9EYXRhIiwiZGVjb2RlZEF1ZGlvIiwiZmlyZUV2ZW50IiwiZXJyb3IiLCJuYW1lIiwibWVzc2FnZSIsImF1ZGlvQnVmZmVyRXhpc3RzIiwiQm9vbGVhbiIsInBsYXkiLCJzdGF0ZSIsInJlc3VtZSIsInN0YXJ0Iiwic3RvcCIsInBhdXNlIl0sIm1hcHBpbmdzIjoiOzs7O0FBQUE7O0FBQ0E7Ozs7Ozs7Ozs7Ozs7O0FBRUE7Ozs7SUFJcUJBLE07Ozs7Ozs7Ozs7OztrSkFLbkJDLFcsR0FBYyxJLFFBTWRDLFcsR0FBYyxJOztBQVZkOzs7Ozs7QUFNQTs7Ozs7O21CQU1BQyxrQixpQ0FBcUI7QUFDbkIsU0FBS0Msc0JBQUw7O0FBRUE7QUFDQSxTQUFLRixXQUFMLEdBQW1CLEtBQUtHLFlBQUwsQ0FBa0JGLGtCQUFsQixFQUFuQjs7QUFFQTtBQUNBLFNBQUtELFdBQUwsQ0FBaUJJLE1BQWpCLEdBQTBCLEtBQUtMLFdBQS9COztBQUVBO0FBQ0EsU0FBS0MsV0FBTCxDQUFpQkssT0FBakIsQ0FBeUIsS0FBS0YsWUFBTCxDQUFrQkcsV0FBM0M7O0FBRUE7QUFDQSxTQUFLTixXQUFMLENBQWlCTyxnQkFBakIsQ0FBa0MsT0FBbEMsRUFBMkMsS0FBS0MsbUJBQWhEO0FBQ0QsRzs7bUJBRUROLHNCLHFDQUF5QjtBQUN2QixRQUFJLEtBQUtPLGlCQUFMLEVBQUosRUFBOEI7QUFDNUIsV0FBS1QsV0FBTCxDQUFpQlUsVUFBakI7QUFDQSxXQUFLVixXQUFMLENBQWlCVyxtQkFBakIsQ0FBcUMsT0FBckMsRUFBOEMsS0FBS0gsbUJBQW5EO0FBQ0Q7QUFDRixHOztBQUVEOzs7Ozs7Ozs7Ozs7OzttQkFZTUksSTt3RkFBS0MsRzs7O1VBQUtDLGEsdUVBQWdCLEk7Ozs7OztrQkFDekJELEc7Ozs7Ozs7OztBQUlMO0FBQ01FLDJCLEdBQWdCRCx5RTtBQUNmWCwwQixHQUFnQixJLENBQWhCQSxZOzs7cUJBR2tCWSxjQUFjLEtBQWQsRUFBcUJGLEdBQXJCLEM7OztBQUFqQkcsc0I7O3FCQUNrQkEsU0FBU0MsV0FBVCxFOzs7QUFBbEJDLHVCOztBQUNOZiwyQkFBYWdCLGVBQWIsQ0FBNkJELFNBQTdCLEVBQXdDLHdCQUFnQjtBQUN0RCx1QkFBS25CLFdBQUwsR0FBbUJxQixZQUFuQjtBQUNBLHVCQUFLQyxTQUFMLENBQWUsUUFBZjtBQUNELGVBSEQ7Ozs7Ozs7O0FBS0EsbUJBQUtDLEtBQUwsQ0FBYyxZQUFNQyxJQUFwQixVQUE2QixZQUFNQyxPQUFuQzs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFJSjs7Ozs7O21CQUtBQyxpQixnQ0FBb0I7QUFDbEIsV0FBT0MsUUFBUSxLQUFLM0IsV0FBYixDQUFQO0FBQ0QsRzs7QUFFRDs7Ozs7OzttQkFLQVUsaUIsZ0NBQW9CO0FBQ2xCLFdBQU9pQixRQUFRLEtBQUsxQixXQUFiLENBQVA7QUFDRCxHOztBQUVEOzs7OzttQkFHQTJCLEksbUJBQU87QUFDTCxTQUFLMUIsa0JBQUw7O0FBRUEsUUFBSSxLQUFLRSxZQUFMLENBQWtCeUIsS0FBbEIsS0FBNEIsV0FBaEMsRUFBNkM7QUFDM0MsV0FBS3pCLFlBQUwsQ0FBa0IwQixNQUFsQjtBQUNEOztBQUVEO0FBQ0EsU0FBSzdCLFdBQUwsQ0FBaUI4QixLQUFqQjtBQUNBLFNBQUtULFNBQUwsQ0FBZSxTQUFmO0FBQ0QsRzs7QUFFRDs7Ozs7O21CQUlBVSxJLG1CQUFPO0FBQ0wsUUFBSSxDQUFDLEtBQUtOLGlCQUFMLEVBQUQsSUFBNkIsQ0FBQyxLQUFLaEIsaUJBQUwsRUFBbEMsRUFBNEQ7QUFDMUQ7QUFDRDs7QUFFRCxTQUFLVCxXQUFMLENBQWlCK0IsSUFBakI7QUFDQSxTQUFLVixTQUFMLENBQWUsU0FBZjtBQUNELEc7O21CQUVEVyxLLG9CQUFRO0FBQ04sUUFBSSxDQUFDLEtBQUtQLGlCQUFMLEVBQUQsSUFBNkIsQ0FBQyxLQUFLaEIsaUJBQUwsRUFBbEMsRUFBNEQ7QUFDMUQ7QUFDRDs7QUFFRCxTQUFLVCxXQUFMLENBQWlCK0IsSUFBakI7QUFDQSxTQUFLVixTQUFMLENBQWUsT0FBZjtBQUNELEc7Ozs7O2tCQTFIa0J2QixNIiwiZmlsZSI6IlBsYXllci5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7YXV0aG9yaXNlZFJlcXVlc3QsIHJlcXVlc3R9IGZyb20gJy4uL2FwaS9jb21tdW5pY2F0aW9uJztcbmltcG9ydCBBdWRpb0NvbnRleHQgZnJvbSAnLi9BdWRpb0NvbnRleHQnO1xuXG4vKipcbiAqIFBsYXllclxuICogU2ltcGxlIEF1ZGlvIFBsYXllciBiYXNlZCBvbiBXZWIgQXVkaW8gQVBJIHRlY2hub2xvZ3lcbiAqL1xuZXhwb3J0IGRlZmF1bHQgY2xhc3MgUGxheWVyIGV4dGVuZHMgQXVkaW9Db250ZXh0IHtcbiAgLyoqXG4gICAqIFByaXZhdGUgb2JqZWN0IHRvIGhvbGQgQXVkaW9CdWZmZXIgbm9kZS5cbiAgICogQHByaXZhdGVcbiAgICovXG4gIGF1ZGlvQnVmZmVyID0gbnVsbDtcblxuICAvKipcbiAgICogUHJpdmF0ZSBvYmplY3QgdG8gaG9sZCBBdWRpb0J1ZmZlclNvdXJjZU5vZGUgbm9kZS5cbiAgICogQHByaXZhdGVcbiAgICovXG4gIGF1ZGlvU291cmNlID0gbnVsbDtcblxuICBjcmVhdGVCdWZmZXJTb3VyY2UoKSB7XG4gICAgdGhpcy5kaXNjb25uZWN0QnVmZmVyU291cmNlKCk7XG5cbiAgICAvLyBDcmVhdGUgYSBzb3VuZCBzb3VyY2VcbiAgICB0aGlzLmF1ZGlvU291cmNlID0gdGhpcy5hdWRpb0NvbnRleHQuY3JlYXRlQnVmZmVyU291cmNlKCk7XG5cbiAgICAvLyBTZWxlY3Qgd2hhdCB0byBwbGF5XG4gICAgdGhpcy5hdWRpb1NvdXJjZS5idWZmZXIgPSB0aGlzLmF1ZGlvQnVmZmVyO1xuXG4gICAgLy8gQ29ubmVjdCB0byB0aGUgc3BlYWtlcnMhXG4gICAgdGhpcy5hdWRpb1NvdXJjZS5jb25uZWN0KHRoaXMuYXVkaW9Db250ZXh0LmRlc3RpbmF0aW9uKTtcblxuICAgIC8vIEFkZCBzb21lIGV2ZW50IGhhbmRsZXJzO1xuICAgIHRoaXMuYXVkaW9Tb3VyY2UuYWRkRXZlbnRMaXN0ZW5lcignZW5kZWQnLCB0aGlzLnN1c3BlbmRBdWRpb0NvbnRleHQpO1xuICB9XG5cbiAgZGlzY29ubmVjdEJ1ZmZlclNvdXJjZSgpIHtcbiAgICBpZiAodGhpcy5hdWRpb1NvdXJjZUV4aXN0cygpKSB7XG4gICAgICB0aGlzLmF1ZGlvU291cmNlLmRpc2Nvbm5lY3QoKTtcbiAgICAgIHRoaXMuYXVkaW9Tb3VyY2UucmVtb3ZlRXZlbnRMaXN0ZW5lcignZW5kZWQnLCB0aGlzLnN1c3BlbmRBdWRpb0NvbnRleHQpO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBHZXQgYW4gYXVkaW8gc3RyZWFtIGZyb20gYW4gVVJMLlxuICAgKlxuICAgKiBJZiB0aGUgd2l0aEl0c2xUb2tlbiBpcyBwcm92aWRlZCAoYW5kIHNldCB0byB0cnVlKSB1c2UgdGhlIGF1dGhvcml6ZWRSZXF1ZXN0IG1ldGhvZCB0byBsb2FkXG4gICAqIHRoZSBhdWRpby4gVGhpcyBpbiBlZmZlY3Qgd2lsbCBzZXQgdGhlIElUU0xhbmd1YWdlIGJlYXJlciB0b2tlbiAoaWYgYXZhaWxhYmxlKSB0byB0aGUgcmVxdWVzdC5cbiAgICpcbiAgICogRm9yIGJvdGggcmVxdWVzdCBtZXRob2RzIGdvZXM6IHRoZXJlJ3Mgbm8gY2hlY2sgd2hldGhlciB5b3UncmUgdHJ5aW5nIHRvIGxvYWQgZnJvbSBJVFNMYW5ndWFnZVxuICAgKiBiYWNrZW5kIHN5c3RlbSBvciBub3QuXG4gICAqXG4gICAqIEBwYXJhbSB7c3RyaW5nfSB1cmwgLSBVcmwgdG8gbG9hZC5cbiAgICogQHBhcmFtIHtib29sZWFufSB3aXRoSXRzbFRva2VuIC0gTWFrZSB1c2Ugb2YgYXV0aG9yaXplZFJlcXVlc3Qgb3IganVzdCByZXF1ZXN0IGlmIHNldCB0byBmYWxzZS5cbiAgICovXG4gIGFzeW5jIGxvYWQodXJsLCB3aXRoSXRzbFRva2VuID0gdHJ1ZSkge1xuICAgIGlmICghdXJsKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgLy8gRGV0ZXJtaW5lIHdoZXRoZXIgdG8gYXNrIGF1dGhvcml6ZWQsIG9yIG5vdC5cbiAgICBjb25zdCByZXF1ZXN0TWV0aG9kID0gd2l0aEl0c2xUb2tlbiA/IGF1dGhvcmlzZWRSZXF1ZXN0IDogcmVxdWVzdDtcbiAgICBjb25zdCB7YXVkaW9Db250ZXh0fSA9IHRoaXM7XG5cbiAgICB0cnkge1xuICAgICAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCByZXF1ZXN0TWV0aG9kKCdHRVQnLCB1cmwpO1xuICAgICAgY29uc3QgYXVkaW9EYXRhID0gYXdhaXQgcmVzcG9uc2UuYXJyYXlCdWZmZXIoKTtcbiAgICAgIGF1ZGlvQ29udGV4dC5kZWNvZGVBdWRpb0RhdGEoYXVkaW9EYXRhLCBkZWNvZGVkQXVkaW8gPT4ge1xuICAgICAgICB0aGlzLmF1ZGlvQnVmZmVyID0gZGVjb2RlZEF1ZGlvO1xuICAgICAgICB0aGlzLmZpcmVFdmVudCgnbG9hZGVkJyk7XG4gICAgICB9KTtcbiAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgdGhpcy5lcnJvcihgJHtlcnJvci5uYW1lfTogJHtlcnJvci5tZXNzYWdlfWApO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBSZXR1cm5zIGlmIHRoZSBhdWRpb0J1ZmZlciBoYXMgYmVlbiBjcmVhdGVkIGFuZCBsb2FkZWQsIG9yIG5vdC5cblxuICAgKiBAcmV0dXJucyB7Ym9vbGVhbn0gVGhlIGF1ZGlvQnVmZmVyIGNyZWF0ZWQuXG4gICAqL1xuICBhdWRpb0J1ZmZlckV4aXN0cygpIHtcbiAgICByZXR1cm4gQm9vbGVhbih0aGlzLmF1ZGlvQnVmZmVyKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZXR1cm4gaWYgYXVkaW9Tb3VyY2UgaGFzIGJlZW4gY3JlYXRlZCwgb3Igbm90LlxuICAgKlxuICAgKiBAcmV0dXJucyB7Ym9vbGVhbn0gVGhlIGF1ZGlvU291cmNlIGNyZWF0ZWQuXG4gICAqL1xuICBhdWRpb1NvdXJjZUV4aXN0cygpIHtcbiAgICByZXR1cm4gQm9vbGVhbih0aGlzLmF1ZGlvU291cmNlKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBTdGFydCBhdWRpbyBwbGF5YmFjayBvZiB0aGF0IHdoYXQgaXMgaW4gdGhlIGJ1ZmZlci5cbiAgICovXG4gIHBsYXkoKSB7XG4gICAgdGhpcy5jcmVhdGVCdWZmZXJTb3VyY2UoKTtcblxuICAgIGlmICh0aGlzLmF1ZGlvQ29udGV4dC5zdGF0ZSA9PT0gJ3N1c3BlbmRlZCcpIHtcbiAgICAgIHRoaXMuYXVkaW9Db250ZXh0LnJlc3VtZSgpO1xuICAgIH1cblxuICAgIC8vIHBsYXkgdGhlIHNvdXJjZSBub3dcbiAgICB0aGlzLmF1ZGlvU291cmNlLnN0YXJ0KCk7XG4gICAgdGhpcy5maXJlRXZlbnQoJ3BsYXlpbmcnKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBTdG9wIHBsYXliYWNrIG9mIGF1ZGlvLlxuICAgKiBDaGVjayBmb3IgYnVmZmVyIGFuZCBzb3VyY2UgdG8gZXhpc3QsIGlmIG5vdCwgZXhpdC5cbiAgICovXG4gIHN0b3AoKSB7XG4gICAgaWYgKCF0aGlzLmF1ZGlvQnVmZmVyRXhpc3RzKCkgJiYgIXRoaXMuYXVkaW9Tb3VyY2VFeGlzdHMoKSkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIHRoaXMuYXVkaW9Tb3VyY2Uuc3RvcCgpO1xuICAgIHRoaXMuZmlyZUV2ZW50KCdzdG9wcGVkJyk7XG4gIH1cblxuICBwYXVzZSgpIHtcbiAgICBpZiAoIXRoaXMuYXVkaW9CdWZmZXJFeGlzdHMoKSAmJiAhdGhpcy5hdWRpb1NvdXJjZUV4aXN0cygpKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgdGhpcy5hdWRpb1NvdXJjZS5zdG9wKCk7XG4gICAgdGhpcy5maXJlRXZlbnQoJ3BhdXNlJyk7XG4gIH1cbn1cbiJdfQ== | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9XZWJBdWRpby9QbGF5ZXIuanMiXSwibmFtZXMiOlsiUGxheWVyIiwiYXVkaW9CdWZmZXIiLCJhdWRpb1NvdXJjZSIsImNyZWF0ZUJ1ZmZlclNvdXJjZSIsImRpc2Nvbm5lY3RCdWZmZXJTb3VyY2UiLCJhdWRpb0NvbnRleHQiLCJidWZmZXIiLCJjb25uZWN0IiwiZGVzdGluYXRpb24iLCJhZGRFdmVudExpc3RlbmVyIiwic3VzcGVuZEF1ZGlvQ29udGV4dCIsImF1ZGlvU291cmNlRXhpc3RzIiwiZGlzY29ubmVjdCIsInJlbW92ZUV2ZW50TGlzdGVuZXIiLCJsb2FkIiwidXJsIiwid2l0aEl0c2xUb2tlbiIsInJlcXVlc3RNZXRob2QiLCJyZXNwb25zZSIsImFycmF5QnVmZmVyIiwiYXVkaW9EYXRhIiwiZGVjb2RlQXVkaW9EYXRhIiwiZGVjb2RlZEF1ZGlvIiwiZmlyZUV2ZW50IiwiY29uc29sZSIsImVycm9yIiwibG9nIiwibmFtZSIsIm1lc3NhZ2UiLCJhdWRpb0J1ZmZlckV4aXN0cyIsIkJvb2xlYW4iLCJwbGF5Iiwic3RhdGUiLCJyZXN1bWUiLCJzdGFydCIsInN0b3AiLCJwYXVzZSJdLCJtYXBwaW5ncyI6Ijs7OztBQUFBOztBQUNBOzs7Ozs7Ozs7Ozs7OztBQUVBOzs7O0lBSXFCQSxNOzs7Ozs7Ozs7Ozs7a0pBS25CQyxXLEdBQWMsSSxRQU1kQyxXLEdBQWMsSTs7QUFWZDs7Ozs7O0FBTUE7Ozs7OzttQkFNQUMsa0IsaUNBQXFCO0FBQ25CLFNBQUtDLHNCQUFMOztBQUVBO0FBQ0EsU0FBS0YsV0FBTCxHQUFtQixLQUFLRyxZQUFMLENBQWtCRixrQkFBbEIsRUFBbkI7O0FBRUE7QUFDQSxTQUFLRCxXQUFMLENBQWlCSSxNQUFqQixHQUEwQixLQUFLTCxXQUEvQjs7QUFFQTtBQUNBLFNBQUtDLFdBQUwsQ0FBaUJLLE9BQWpCLENBQXlCLEtBQUtGLFlBQUwsQ0FBa0JHLFdBQTNDOztBQUVBO0FBQ0EsU0FBS04sV0FBTCxDQUFpQk8sZ0JBQWpCLENBQWtDLE9BQWxDLEVBQTJDLEtBQUtDLG1CQUFoRDtBQUNELEc7O21CQUVETixzQixxQ0FBeUI7QUFDdkIsUUFBSSxLQUFLTyxpQkFBTCxFQUFKLEVBQThCO0FBQzVCLFdBQUtULFdBQUwsQ0FBaUJVLFVBQWpCO0FBQ0EsV0FBS1YsV0FBTCxDQUFpQlcsbUJBQWpCLENBQXFDLE9BQXJDLEVBQThDLEtBQUtILG1CQUFuRDtBQUNEO0FBQ0YsRzs7QUFFRDs7Ozs7Ozs7Ozs7Ozs7bUJBWU1JLEk7d0ZBQUtDLEc7OztVQUFLQyxhLHVFQUFnQixJOzs7Ozs7a0JBQ3pCRCxHOzs7Ozs7Ozs7QUFJTDtBQUNNRSwyQixHQUFnQkQseUU7QUFDZlgsMEIsR0FBZ0IsSSxDQUFoQkEsWTs7O3FCQUdrQlksY0FBYyxLQUFkLEVBQXFCRixHQUFyQixDOzs7QUFBakJHLHNCOztxQkFDa0JBLFNBQVNDLFdBQVQsRTs7O0FBQWxCQyx1Qjs7QUFDTmYsMkJBQWFnQixlQUFiLENBQTZCRCxTQUE3QixFQUF3Qyx3QkFBZ0I7QUFDdEQsdUJBQUtuQixXQUFMLEdBQW1CcUIsWUFBbkI7QUFDQSx1QkFBS0MsU0FBTCxDQUFlLFFBQWY7QUFDRCxlQUhEOzs7Ozs7OztBQUtBQyxzQkFBUUMsS0FBUjtBQUNBRCxzQkFBUUUsR0FBUixDQUFZLGVBQVo7QUFDQSxtQkFBS0QsS0FBTCxDQUFjLFlBQU1FLElBQXBCLFVBQTZCLFlBQU1DLE9BQW5DOzs7Ozs7Ozs7Ozs7Ozs7OztBQUlKOzs7Ozs7bUJBS0FDLGlCLGdDQUFvQjtBQUNsQixXQUFPQyxRQUFRLEtBQUs3QixXQUFiLENBQVA7QUFDRCxHOztBQUVEOzs7Ozs7O21CQUtBVSxpQixnQ0FBb0I7QUFDbEIsV0FBT21CLFFBQVEsS0FBSzVCLFdBQWIsQ0FBUDtBQUNELEc7O0FBRUQ7Ozs7O21CQUdBNkIsSSxtQkFBTztBQUNMLFNBQUs1QixrQkFBTDs7QUFFQSxRQUFJLEtBQUtFLFlBQUwsQ0FBa0IyQixLQUFsQixLQUE0QixXQUFoQyxFQUE2QztBQUMzQyxXQUFLM0IsWUFBTCxDQUFrQjRCLE1BQWxCO0FBQ0Q7O0FBRUQ7QUFDQSxTQUFLL0IsV0FBTCxDQUFpQmdDLEtBQWpCO0FBQ0EsU0FBS1gsU0FBTCxDQUFlLFNBQWY7QUFDRCxHOztBQUVEOzs7Ozs7bUJBSUFZLEksbUJBQU87QUFDTCxRQUFJLENBQUMsS0FBS04saUJBQUwsRUFBRCxJQUE2QixDQUFDLEtBQUtsQixpQkFBTCxFQUFsQyxFQUE0RDtBQUMxRDtBQUNEOztBQUVELFNBQUtULFdBQUwsQ0FBaUJpQyxJQUFqQjtBQUNBLFNBQUtaLFNBQUwsQ0FBZSxTQUFmO0FBQ0QsRzs7bUJBRURhLEssb0JBQVE7QUFDTixRQUFJLENBQUMsS0FBS1AsaUJBQUwsRUFBRCxJQUE2QixDQUFDLEtBQUtsQixpQkFBTCxFQUFsQyxFQUE0RDtBQUMxRDtBQUNEOztBQUVELFNBQUtULFdBQUwsQ0FBaUJpQyxJQUFqQjtBQUNBLFNBQUtaLFNBQUwsQ0FBZSxPQUFmO0FBQ0QsRzs7Ozs7a0JBNUhrQnZCLE0iLCJmaWxlIjoiUGxheWVyLmpzIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHthdXRob3Jpc2VkUmVxdWVzdCwgcmVxdWVzdH0gZnJvbSAnLi4vYXBpL2NvbW11bmljYXRpb24nO1xuaW1wb3J0IEF1ZGlvQ29udGV4dCBmcm9tICcuL0F1ZGlvQ29udGV4dCc7XG5cbi8qKlxuICogUGxheWVyXG4gKiBTaW1wbGUgQXVkaW8gUGxheWVyIGJhc2VkIG9uIFdlYiBBdWRpbyBBUEkgdGVjaG5vbG9neVxuICovXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBQbGF5ZXIgZXh0ZW5kcyBBdWRpb0NvbnRleHQge1xuICAvKipcbiAgICogUHJpdmF0ZSBvYmplY3QgdG8gaG9sZCBBdWRpb0J1ZmZlciBub2RlLlxuICAgKiBAcHJpdmF0ZVxuICAgKi9cbiAgYXVkaW9CdWZmZXIgPSBudWxsO1xuXG4gIC8qKlxuICAgKiBQcml2YXRlIG9iamVjdCB0byBob2xkIEF1ZGlvQnVmZmVyU291cmNlTm9kZSBub2RlLlxuICAgKiBAcHJpdmF0ZVxuICAgKi9cbiAgYXVkaW9Tb3VyY2UgPSBudWxsO1xuXG4gIGNyZWF0ZUJ1ZmZlclNvdXJjZSgpIHtcbiAgICB0aGlzLmRpc2Nvbm5lY3RCdWZmZXJTb3VyY2UoKTtcblxuICAgIC8vIENyZWF0ZSBhIHNvdW5kIHNvdXJjZVxuICAgIHRoaXMuYXVkaW9Tb3VyY2UgPSB0aGlzLmF1ZGlvQ29udGV4dC5jcmVhdGVCdWZmZXJTb3VyY2UoKTtcblxuICAgIC8vIFNlbGVjdCB3aGF0IHRvIHBsYXlcbiAgICB0aGlzLmF1ZGlvU291cmNlLmJ1ZmZlciA9IHRoaXMuYXVkaW9CdWZmZXI7XG5cbiAgICAvLyBDb25uZWN0IHRvIHRoZSBzcGVha2VycyFcbiAgICB0aGlzLmF1ZGlvU291cmNlLmNvbm5lY3QodGhpcy5hdWRpb0NvbnRleHQuZGVzdGluYXRpb24pO1xuXG4gICAgLy8gQWRkIHNvbWUgZXZlbnQgaGFuZGxlcnM7XG4gICAgdGhpcy5hdWRpb1NvdXJjZS5hZGRFdmVudExpc3RlbmVyKCdlbmRlZCcsIHRoaXMuc3VzcGVuZEF1ZGlvQ29udGV4dCk7XG4gIH1cblxuICBkaXNjb25uZWN0QnVmZmVyU291cmNlKCkge1xuICAgIGlmICh0aGlzLmF1ZGlvU291cmNlRXhpc3RzKCkpIHtcbiAgICAgIHRoaXMuYXVkaW9Tb3VyY2UuZGlzY29ubmVjdCgpO1xuICAgICAgdGhpcy5hdWRpb1NvdXJjZS5yZW1vdmVFdmVudExpc3RlbmVyKCdlbmRlZCcsIHRoaXMuc3VzcGVuZEF1ZGlvQ29udGV4dCk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIEdldCBhbiBhdWRpbyBzdHJlYW0gZnJvbSBhbiBVUkwuXG4gICAqXG4gICAqIElmIHRoZSB3aXRoSXRzbFRva2VuIGlzIHByb3ZpZGVkIChhbmQgc2V0IHRvIHRydWUpIHVzZSB0aGUgYXV0aG9yaXplZFJlcXVlc3QgbWV0aG9kIHRvIGxvYWRcbiAgICogdGhlIGF1ZGlvLiBUaGlzIGluIGVmZmVjdCB3aWxsIHNldCB0aGUgSVRTTGFuZ3VhZ2UgYmVhcmVyIHRva2VuIChpZiBhdmFpbGFibGUpIHRvIHRoZSByZXF1ZXN0LlxuICAgKlxuICAgKiBGb3IgYm90aCByZXF1ZXN0IG1ldGhvZHMgZ29lczogdGhlcmUncyBubyBjaGVjayB3aGV0aGVyIHlvdSdyZSB0cnlpbmcgdG8gbG9hZCBmcm9tIElUU0xhbmd1YWdlXG4gICAqIGJhY2tlbmQgc3lzdGVtIG9yIG5vdC5cbiAgICpcbiAgICogQHBhcmFtIHtzdHJpbmd9IHVybCAtIFVybCB0byBsb2FkLlxuICAgKiBAcGFyYW0ge2Jvb2xlYW59IHdpdGhJdHNsVG9rZW4gLSBNYWtlIHVzZSBvZiBhdXRob3JpemVkUmVxdWVzdCBvciBqdXN0IHJlcXVlc3QgaWYgc2V0IHRvIGZhbHNlLlxuICAgKi9cbiAgYXN5bmMgbG9hZCh1cmwsIHdpdGhJdHNsVG9rZW4gPSB0cnVlKSB7XG4gICAgaWYgKCF1cmwpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICAvLyBEZXRlcm1pbmUgd2hldGhlciB0byBhc2sgYXV0aG9yaXplZCwgb3Igbm90LlxuICAgIGNvbnN0IHJlcXVlc3RNZXRob2QgPSB3aXRoSXRzbFRva2VuID8gYXV0aG9yaXNlZFJlcXVlc3QgOiByZXF1ZXN0O1xuICAgIGNvbnN0IHthdWRpb0NvbnRleHR9ID0gdGhpcztcblxuICAgIHRyeSB7XG4gICAgICBjb25zdCByZXNwb25zZSA9IGF3YWl0IHJlcXVlc3RNZXRob2QoJ0dFVCcsIHVybCk7XG4gICAgICBjb25zdCBhdWRpb0RhdGEgPSBhd2FpdCByZXNwb25zZS5hcnJheUJ1ZmZlcigpO1xuICAgICAgYXVkaW9Db250ZXh0LmRlY29kZUF1ZGlvRGF0YShhdWRpb0RhdGEsIGRlY29kZWRBdWRpbyA9PiB7XG4gICAgICAgIHRoaXMuYXVkaW9CdWZmZXIgPSBkZWNvZGVkQXVkaW87XG4gICAgICAgIHRoaXMuZmlyZUV2ZW50KCdsb2FkZWQnKTtcbiAgICAgIH0pO1xuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICBjb25zb2xlLmVycm9yKGVycm9yKTtcbiAgICAgIGNvbnNvbGUubG9nKCdjYXRjaGllIGNhdGNoJyk7XG4gICAgICB0aGlzLmVycm9yKGAke2Vycm9yLm5hbWV9OiAke2Vycm9yLm1lc3NhZ2V9YCk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFJldHVybnMgaWYgdGhlIGF1ZGlvQnVmZmVyIGhhcyBiZWVuIGNyZWF0ZWQgYW5kIGxvYWRlZCwgb3Igbm90LlxuXG4gICAqIEByZXR1cm5zIHtib29sZWFufSBUaGUgYXVkaW9CdWZmZXIgY3JlYXRlZC5cbiAgICovXG4gIGF1ZGlvQnVmZmVyRXhpc3RzKCkge1xuICAgIHJldHVybiBCb29sZWFuKHRoaXMuYXVkaW9CdWZmZXIpO1xuICB9XG5cbiAgLyoqXG4gICAqIFJldHVybiBpZiBhdWRpb1NvdXJjZSBoYXMgYmVlbiBjcmVhdGVkLCBvciBub3QuXG4gICAqXG4gICAqIEByZXR1cm5zIHtib29sZWFufSBUaGUgYXVkaW9Tb3VyY2UgY3JlYXRlZC5cbiAgICovXG4gIGF1ZGlvU291cmNlRXhpc3RzKCkge1xuICAgIHJldHVybiBCb29sZWFuKHRoaXMuYXVkaW9Tb3VyY2UpO1xuICB9XG5cbiAgLyoqXG4gICAqIFN0YXJ0IGF1ZGlvIHBsYXliYWNrIG9mIHRoYXQgd2hhdCBpcyBpbiB0aGUgYnVmZmVyLlxuICAgKi9cbiAgcGxheSgpIHtcbiAgICB0aGlzLmNyZWF0ZUJ1ZmZlclNvdXJjZSgpO1xuXG4gICAgaWYgKHRoaXMuYXVkaW9Db250ZXh0LnN0YXRlID09PSAnc3VzcGVuZGVkJykge1xuICAgICAgdGhpcy5hdWRpb0NvbnRleHQucmVzdW1lKCk7XG4gICAgfVxuXG4gICAgLy8gcGxheSB0aGUgc291cmNlIG5vd1xuICAgIHRoaXMuYXVkaW9Tb3VyY2Uuc3RhcnQoKTtcbiAgICB0aGlzLmZpcmVFdmVudCgncGxheWluZycpO1xuICB9XG5cbiAgLyoqXG4gICAqIFN0b3AgcGxheWJhY2sgb2YgYXVkaW8uXG4gICAqIENoZWNrIGZvciBidWZmZXIgYW5kIHNvdXJjZSB0byBleGlzdCwgaWYgbm90LCBleGl0LlxuICAgKi9cbiAgc3RvcCgpIHtcbiAgICBpZiAoIXRoaXMuYXVkaW9CdWZmZXJFeGlzdHMoKSAmJiAhdGhpcy5hdWRpb1NvdXJjZUV4aXN0cygpKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgdGhpcy5hdWRpb1NvdXJjZS5zdG9wKCk7XG4gICAgdGhpcy5maXJlRXZlbnQoJ3N0b3BwZWQnKTtcbiAgfVxuXG4gIHBhdXNlKCkge1xuICAgIGlmICghdGhpcy5hdWRpb0J1ZmZlckV4aXN0cygpICYmICF0aGlzLmF1ZGlvU291cmNlRXhpc3RzKCkpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICB0aGlzLmF1ZGlvU291cmNlLnN0b3AoKTtcbiAgICB0aGlzLmZpcmVFdmVudCgncGF1c2UnKTtcbiAgfVxufVxuIl19 |
807328
6925