speaker-calibration 2.2.151 → 2.2.153

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/main.js CHANGED
@@ -818,7 +818,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var qrco
818
818
  /***/ (function(__unused_webpack_module, __webpack_exports__, __webpack_require__) {
819
819
 
820
820
  "use strict";
821
- eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var axios__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! axios */ \"./node_modules/axios/index.js\");\n/* harmony import */ var axios__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(axios__WEBPACK_IMPORTED_MODULE_0__);\n\n/**\n *\n */\nclass PythonServerAPI {\n static PYTHON_SERVER_URL = 'https://easyeyes-python-flask-server.herokuapp.com';\n\n static TEST_SERVER_URL = 'http://127.0.0.1:5000';\n // static PYTHON_SERVER_URL ='http://127.0.0.1:5000';\n\n /** @private */\n MAX_RETRY_COUNT = 3;\n /** @private */\n RETRY_DELAY_MS = 1000;\n /**\n * @param data- -\n * g = inverted impulse response, when convolved with the impulse\n * reponse, they cancel out.\n * @param data.payload\n * @param data.sampleRate\n * @param data.P\n * @param data-.payload\n * @param data-.sampleRate\n * @param data-.P\n * @returns\n * @example\n */\n getImpulseResponse = async ({\n mls, \n payload, \n sampleRate, \n P, \n numPeriods\n }) => {\n const task = 'impulse-response';\n let res = null;\n\n console.log({payload});\n\n const data = JSON.stringify({\n task,\n payload,\n 'sample-rate': sampleRate,\n mls,\n P,\n numPeriods,\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n return res.data[task];\n };\n\n getMLS = async ({length,amplitude}) => {\n const task = 'mls';\n let res = null;\n\n const data = JSON.stringify({\n task,\n length: length,\n amplitude: amplitude\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n\n return res.data[task];\n };\n getMLSWithRetry = async ({length,amplitude}) => {\n let retryCount = 0;\n let response = null;\n\n while (retryCount < this.MAX_RETRY_COUNT) {\n try {\n response = await this.getMLS({length,amplitude});\n // If the request is successful, break out of the loop\n break;\n } catch (error) {\n console.error(`Error occurred. Retrying... (${retryCount + 1}/${this.MAX_RETRY_COUNT})`);\n retryCount++;\n await new Promise(resolve => setTimeout(resolve, this.RETRY_DELAY_MS));\n }\n }\n\n if (response) {\n return response;\n } else {\n throw new Error(`Failed to get MLS after ${this.MAX_RETRY_COUNT} attempts.`);\n }\n };\n\n getPSD = async ({unconv_rec, conv_rec, sampleRate}) => {\n const task = 'psd';\n let res = null;\n\n const data = JSON.stringify({\n task,\n unconv_rec,\n conv_rec,\n sampleRate,\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n return res.data[task];\n };\n\n getBackgroundNoisePSD = async ({background_rec, sampleRate}) => {\n const task = 'background-psd';\n let res = null;\n\n const data = JSON.stringify({\n task,\n background_rec,\n sampleRate,\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n return res.data[task];\n };\n\n getBackgroundNoisePSDWithRetry = async ({background_rec, sampleRate}) => {\n let retryCount = 0;\n let response = null;\n\n while (retryCount < this.MAX_RETRY_COUNT) {\n try {\n response = await this.getBackgroundNoisePSD({background_rec, sampleRate});\n // If the request is successful, break out of the loop\n break;\n } catch (error) {\n console.error(`Error occurred. Retrying... (${retryCount + 1}/${this.MAX_RETRY_COUNT})`);\n retryCount++;\n await new Promise(resolve => setTimeout(resolve, this.RETRY_DELAY_MS));\n }\n }\n\n if (response) {\n return response;\n } else {\n throw new Error(`Failed to get PSD after ${this.MAX_RETRY_COUNT} attempts.`);\n }\n };\n\n getSubtractedPSD = async (rec, knownGains, knownFrequencies, sampleRate) => {\n const task = 'subtracted-psd';\n let res = null;\n\n const data = JSON.stringify({\n task,\n rec,\n knownGains,\n knownFrequencies,\n sampleRate,\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n return res.data[task];\n };\n\n getSubtractedPSDWithRetry = async (rec, knownGains, knownFrequencies, sampleRate) => {\n let retryCount = 0;\n let response = null;\n\n while (retryCount < this.MAX_RETRY_COUNT) {\n try {\n response = await this.getSubtractedPSD(rec, knownGains, knownFrequencies, sampleRate);\n // If the request is successful, break out of the loop\n break;\n } catch (error) {\n console.error(`Error occurred. Retrying... (${retryCount + 1}/${this.MAX_RETRY_COUNT})`);\n retryCount++;\n await new Promise(resolve => setTimeout(resolve, this.RETRY_DELAY_MS));\n }\n }\n\n if (response) {\n return response;\n } else {\n throw new Error(`Failed to get PSD after ${this.MAX_RETRY_COUNT} attempts.`);\n }\n };\n\n getPSDWithRetry = async ({unconv_rec, conv_rec, sampleRate}) => {\n let retryCount = 0;\n let response = null;\n\n while (retryCount < this.MAX_RETRY_COUNT) {\n try {\n response = await this.getPSD({unconv_rec, conv_rec, sampleRate});\n // If the request is successful, break out of the loop\n break;\n } catch (error) {\n console.error(`Error occurred. Retrying... (${retryCount + 1}/${this.MAX_RETRY_COUNT})`);\n retryCount++;\n await new Promise(resolve => setTimeout(resolve, this.RETRY_DELAY_MS));\n }\n }\n\n if (response) {\n return response;\n } else {\n throw new Error(`Failed to get PSD after ${this.MAX_RETRY_COUNT} attempts.`);\n }\n };\n\n getComponentInverseImpulseResponse = async ({\n payload,\n mls,\n lowHz,\n highHz,\n componentIRGains,\n iirLength,\n componentIRFreqs,\n num_periods,\n sampleRate,\n mlsAmplitude,\n irLength,\n calibrateSoundSmoothOctaves,\n calibrateSoundBurstFilteredExtraDb\n }) => {\n const task = 'component-inverse-impulse-response';\n let res = null;\n\n console.log({payload});\n\n const data = JSON.stringify({\n task,\n payload,\n mls,\n lowHz,\n highHz,\n iirLength,\n componentIRGains,\n componentIRFreqs,\n num_periods,\n sampleRate,\n mlsAmplitude,\n irLength,\n calibrateSoundSmoothOctaves,\n calibrateSoundBurstFilteredExtraDb\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n\n return res.data[task];\n };\n getSystemInverseImpulseResponse = async ({\n payload,\n mls,\n lowHz,\n highHz,\n iirLength,\n num_periods,\n sampleRate,\n mlsAmplitude,\n calibrateSoundBurstFilteredExtraDb\n }) => {\n const task = 'system-inverse-impulse-response';\n let res = null;\n\n console.log({payload});\n\n const data = JSON.stringify({\n task,\n payload,\n mls,\n lowHz,\n iirLength,\n highHz,\n num_periods,\n sampleRate,\n mlsAmplitude,\n calibrateSoundBurstFilteredExtraDb\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n\n return res.data[task];\n };\n\n getMLSPSD = async ({mls, sampleRate}) => {\n const task = 'mls-psd';\n let res = null;\n\n const data = JSON.stringify({\n task,\n mls,\n sampleRate,\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n\n return res.data[task];\n };\n\n getMLSPSDWithRetry = async ({mls, sampleRate}) => {\n let retryCount = 0;\n let response = null;\n\n while (retryCount < this.MAX_RETRY_COUNT) {\n try {\n response = await this.getMLSPSD({mls, sampleRate});\n // If the request is successful, break out of the loop\n break;\n } catch (error) {\n console.error(`Error occurred. Retrying... (${retryCount + 1}/${this.MAX_RETRY_COUNT})`);\n retryCount++;\n await new Promise(resolve => setTimeout(resolve, this.RETRY_DELAY_MS));\n }\n }\n\n if (response) {\n return response;\n } else {\n throw new Error(\n `Failed to get inverse impulse response after ${this.MAX_RETRY_COUNT} attempts.`\n );\n }\n };\n\n getComponentInverseImpulseResponseWithRetry = async ({\n payload,\n mls,\n lowHz,\n highHz,\n componentIRGains,\n iirLength,\n componentIRFreqs,\n num_periods,\n sampleRate,\n mlsAmplitude,\n irLength,\n calibrateSoundSmoothOctaves,\n calibrateSoundBurstFilteredExtraDb\n }) => {\n let retryCount = 0;\n let response = null;\n\n while (retryCount < this.MAX_RETRY_COUNT) {\n try {\n response = await this.getComponentInverseImpulseResponse({\n payload,\n mls,\n lowHz,\n highHz,\n componentIRGains,\n iirLength,\n componentIRFreqs,\n num_periods,\n sampleRate,\n mlsAmplitude,\n irLength,\n calibrateSoundSmoothOctaves,\n calibrateSoundBurstFilteredExtraDb\n });\n // If the request is successful, break out of the loop\n break;\n } catch (error) {\n console.error(`Error occurred. Retrying... (${retryCount + 1}/${this.MAX_RETRY_COUNT})`);\n retryCount++;\n await new Promise(resolve => setTimeout(resolve, this.RETRY_DELAY_MS));\n }\n }\n\n if (response) {\n return response;\n } else {\n throw new Error(\n `Failed to get inverse impulse response after ${this.MAX_RETRY_COUNT} attempts.`\n );\n }\n };\n\n getSystemInverseImpulseResponseWithRetry = async ({\n payload,\n mls,\n lowHz,\n highHz,\n iirLength,\n num_periods,\n sampleRate,\n mlsAmplitude,\n calibrateSoundBurstFilteredExtraDb\n }) => {\n let retryCount = 0;\n let response = null;\n\n while (retryCount < this.MAX_RETRY_COUNT) {\n try {\n response = await this.getSystemInverseImpulseResponse({\n payload,\n mls,\n lowHz,\n highHz,\n iirLength,\n num_periods,\n sampleRate,\n mlsAmplitude,\n calibrateSoundBurstFilteredExtraDb\n });\n // If the request is successful, break out of the loop\n break;\n } catch (error) {\n console.error(`Error occurred. Retrying... (${retryCount + 1}/${this.MAX_RETRY_COUNT})`);\n retryCount++;\n await new Promise(resolve => setTimeout(resolve, this.RETRY_DELAY_MS));\n }\n }\n\n if (response) {\n return response;\n } else {\n throw new Error(\n `Failed to get inverse impulse response after ${this.MAX_RETRY_COUNT} attempts.`\n );\n }\n };\n\n getVolumeCalibration = async ({payload, sampleRate, lCalib}) => {\n const task = 'volume';\n let res = null;\n\n console.log({payload});\n\n const data = JSON.stringify({\n task,\n payload,\n 'sample-rate': sampleRate,\n lCalib,\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n\n return res.data[task];\n };\n\n getVolumeCalibrationParameters = async ({\n inDBValues,\n outDBSPLValues,\n lCalib,\n componentGainDBSPL,\n }) => {\n const task = 'volume-parameters';\n let res = null;\n\n const data = JSON.stringify({\n task,\n inDBValues,\n outDBSPLValues,\n lCalib,\n componentGainDBSPL,\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL, //server\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n console.log(res.data[task]);\n })\n .catch(error => {\n throw error;\n });\n\n // console.log(res.data[task]);\n //below is an example of res.data[task]\n //{\n // R: 16.56981076554259,\n // RMSError: 1.9289162528535229\n // T: -47.79799120884434,\n // W: 61.0485247483732,\n // backgroundDBSPL: 43.88233142069752,\n // gainDBSPL: -128.24742161208985\n //}\n return res.data[task];\n };\n\n allHzPowerCheck = async ({\n payload, \n sampleRate, \n binDesiredSec,\n burstSec,\n repeats\n }) => {\n const task = 'all-hz-check';\n let res = null;\n\n const data = JSON.stringify({\n payload, \n sampleRate, \n binDesiredSec,\n burstSec,\n repeats\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL, //server\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n console.log(res.data[task]);\n })\n .catch(error => {\n throw error;\n });\n return res.data[task];\n };\n\n volumePowerCheck = async ({\n payload, \n sampleRate, \n preSec,\n Sec,\n binDesiredSec\n }) => {\n const task = 'volume-check';\n let res = null;\n\n const data = JSON.stringify({\n payload, \n sampleRate, \n preSec,\n Sec,\n binDesiredSec\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL, //server\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n console.log(res.data[task]);\n })\n .catch(error => {\n throw error;\n });\n return res.data[task];\n };\n}\n\n/* harmony default export */ __webpack_exports__[\"default\"] = (PythonServerAPI);\n\n\n//# sourceURL=webpack://speakerCalibrator/./src/server/PythonServerAPI.js?");
821
+ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var axios__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! axios */ \"./node_modules/axios/index.js\");\n/* harmony import */ var axios__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(axios__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../utils */ \"./src/utils.js\");\n\n\n/**\n *\n */\nclass PythonServerAPI {\n static PYTHON_SERVER_URL = 'https://easyeyes-python-flask-server.herokuapp.com';\n\n static TEST_SERVER_URL = 'http://127.0.0.1:5000';\n // static PYTHON_SERVER_URL ='http://127.0.0.1:5000';\n\n /** @private */\n MAX_RETRY_COUNT = 3;\n /** @private */\n RETRY_DELAY_MS = 1000;\n /**\n * @param data- -\n * g = inverted impulse response, when convolved with the impulse\n * reponse, they cancel out.\n * @param data.payload\n * @param data.sampleRate\n * @param data.P\n * @param data-.payload\n * @param data-.sampleRate\n * @param data-.P\n * @returns\n * @example\n */\n getImpulseResponse = async ({\n mls, \n sampleRate, \n numPeriods,\n sig,\n fs2,\n L_new_n,\n dL_n\n }) => {\n const task = 'impulse-response';\n let res = null;\n\n const data = JSON.stringify({\n task,\n 'sample-rate': sampleRate,\n mls,\n numPeriods,\n sig,\n fs2,\n L_new_n,\n dL_n\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n return res.data[task];\n };\n\n getAutocorrelation = async ({\n mls, \n payload, \n sampleRate, \n numPeriods\n }) => {\n const task = 'autocorrelation';\n let res = null;\n\n const data = JSON.stringify({\n task,\n payload,\n 'sample-rate': sampleRate,\n mls,\n numPeriods,\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n return res.data[task];\n };\n\n getConvolution = async ({\n mls, \n inverse_response, \n inverse_response_no_bandpass,\n attenuatorGain_dB,\n mls_amplitude\n }) => {\n const task = 'convolution';\n let res = null;\n\n const data = JSON.stringify({\n task,\n mls, \n inverse_response, \n inverse_response_no_bandpass,\n attenuatorGain_dB,\n mls_amplitude\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n return res.data[task];\n };\n\n getMLS = async ({length,amplitude}) => {\n const task = 'mls';\n let res = null;\n\n const data = JSON.stringify({\n task,\n length: length,\n amplitude: amplitude\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n\n return res.data[task];\n };\n\n getMemory = async () => {\n let res;\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/memory`,\n headers: {\n 'Content-Type': 'application/json',\n }\n })\n .then(response => {\n console.log(\"memory used:\", Math.round(response.data['memory']), \"mb\");\n res = response.data['memory'];\n })\n .catch(error => {\n throw error;\n });\n return res;\n };\n\n\n checkMemory = async () => {\n console.log(\"wait for memory under 500 mb to continue calibration\");\n await this.getMemory();\n // let memory = await this.getMemory();\n // while (memory >= 500) {\n // console.log(\"sleep 30s\");\n // await sleep(30);\n // memory = await this.getMemory();\n // }\n \n };\n\n\n getMLSWithRetry = async ({length,amplitude}) => {\n let retryCount = 0;\n let response = null;\n\n while (retryCount < this.MAX_RETRY_COUNT) {\n try {\n response = await this.getMLS({length,amplitude});\n // If the request is successful, break out of the loop\n break;\n } catch (error) {\n console.error(`Error occurred. Retrying... (${retryCount + 1}/${this.MAX_RETRY_COUNT})`);\n retryCount++;\n await new Promise(resolve => setTimeout(resolve, this.RETRY_DELAY_MS));\n }\n }\n\n if (response) {\n return response;\n } else {\n throw new Error(`Failed to get MLS after ${this.MAX_RETRY_COUNT} attempts.`);\n }\n };\n\n getPSD = async ({unconv_rec, conv_rec, sampleRate}) => {\n const task = 'psd';\n let res = null;\n\n const data = JSON.stringify({\n task,\n unconv_rec,\n conv_rec,\n sampleRate,\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n return res.data[task];\n };\n\n getBackgroundNoisePSD = async ({background_rec, sampleRate}) => {\n const task = 'background-psd';\n let res = null;\n\n const data = JSON.stringify({\n task,\n background_rec,\n sampleRate,\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n return res.data[task];\n };\n\n getBackgroundNoisePSDWithRetry = async ({background_rec, sampleRate}) => {\n let retryCount = 0;\n let response = null;\n\n while (retryCount < this.MAX_RETRY_COUNT) {\n try {\n response = await this.getBackgroundNoisePSD({background_rec, sampleRate});\n // If the request is successful, break out of the loop\n break;\n } catch (error) {\n console.error(`Error occurred. Retrying... (${retryCount + 1}/${this.MAX_RETRY_COUNT})`);\n retryCount++;\n await new Promise(resolve => setTimeout(resolve, this.RETRY_DELAY_MS));\n }\n }\n\n if (response) {\n return response;\n } else {\n throw new Error(`Failed to get PSD after ${this.MAX_RETRY_COUNT} attempts.`);\n }\n };\n\n getSubtractedPSD = async (rec, knownGains, knownFrequencies, sampleRate) => {\n const task = 'subtracted-psd';\n let res = null;\n\n const data = JSON.stringify({\n task,\n rec,\n knownGains,\n knownFrequencies,\n sampleRate,\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n return res.data[task];\n };\n\n getSubtractedPSDWithRetry = async (rec, knownGains, knownFrequencies, sampleRate) => {\n let retryCount = 0;\n let response = null;\n\n while (retryCount < this.MAX_RETRY_COUNT) {\n try {\n response = await this.getSubtractedPSD(rec, knownGains, knownFrequencies, sampleRate);\n // If the request is successful, break out of the loop\n break;\n } catch (error) {\n console.error(`Error occurred. Retrying... (${retryCount + 1}/${this.MAX_RETRY_COUNT})`);\n retryCount++;\n await new Promise(resolve => setTimeout(resolve, this.RETRY_DELAY_MS));\n }\n }\n\n if (response) {\n return response;\n } else {\n throw new Error(`Failed to get PSD after ${this.MAX_RETRY_COUNT} attempts.`);\n }\n };\n\n getPSDWithRetry = async ({unconv_rec, conv_rec, sampleRate}) => {\n let retryCount = 0;\n let response = null;\n\n while (retryCount < this.MAX_RETRY_COUNT) {\n try {\n response = await this.getPSD({unconv_rec, conv_rec, sampleRate});\n // If the request is successful, break out of the loop\n break;\n } catch (error) {\n console.error(`Error occurred. Retrying... (${retryCount + 1}/${this.MAX_RETRY_COUNT})`);\n retryCount++;\n await new Promise(resolve => setTimeout(resolve, this.RETRY_DELAY_MS));\n }\n }\n\n if (response) {\n return response;\n } else {\n throw new Error(`Failed to get PSD after ${this.MAX_RETRY_COUNT} attempts.`);\n }\n };\n\n getComponentInverseImpulseResponse = async ({\n payload,\n mls,\n lowHz,\n highHz,\n componentIRGains,\n iirLength,\n componentIRFreqs,\n sampleRate,\n mlsAmplitude,\n irLength,\n calibrateSoundSmoothOctaves,\n calibrateSoundBurstFilteredExtraDb\n }) => {\n const task = 'component-inverse-impulse-response';\n let res = null;\n\n console.log({payload});\n\n const data = JSON.stringify({\n task,\n payload,\n mls,\n lowHz,\n highHz,\n iirLength,\n componentIRGains,\n componentIRFreqs,\n sampleRate,\n mlsAmplitude,\n irLength,\n calibrateSoundSmoothOctaves,\n calibrateSoundBurstFilteredExtraDb\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n\n return res.data[task];\n };\n getSystemInverseImpulseResponse = async ({\n payload,\n mls,\n lowHz,\n highHz,\n iirLength,\n sampleRate,\n mlsAmplitude,\n calibrateSoundBurstFilteredExtraDb\n }) => {\n const task = 'system-inverse-impulse-response';\n let res = null;\n\n console.log({payload});\n\n const data = JSON.stringify({\n task,\n payload,\n mls,\n lowHz,\n iirLength,\n highHz,\n sampleRate,\n mlsAmplitude,\n calibrateSoundBurstFilteredExtraDb\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n\n return res.data[task];\n };\n\n getMLSPSD = async ({mls, sampleRate}) => {\n const task = 'mls-psd';\n let res = null;\n\n const data = JSON.stringify({\n task,\n mls,\n sampleRate,\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n\n return res.data[task];\n };\n\n getMLSPSDWithRetry = async ({mls, sampleRate}) => {\n let retryCount = 0;\n let response = null;\n\n while (retryCount < this.MAX_RETRY_COUNT) {\n try {\n response = await this.getMLSPSD({mls, sampleRate});\n // If the request is successful, break out of the loop\n break;\n } catch (error) {\n console.error(`Error occurred. Retrying... (${retryCount + 1}/${this.MAX_RETRY_COUNT})`);\n retryCount++;\n await new Promise(resolve => setTimeout(resolve, this.RETRY_DELAY_MS));\n }\n }\n\n if (response) {\n return response;\n } else {\n throw new Error(\n `Failed to get inverse impulse response after ${this.MAX_RETRY_COUNT} attempts.`\n );\n }\n };\n\n getComponentInverseImpulseResponseWithRetry = async ({\n payload,\n mls,\n lowHz,\n highHz,\n componentIRGains,\n iirLength,\n componentIRFreqs,\n sampleRate,\n mlsAmplitude,\n irLength,\n calibrateSoundSmoothOctaves,\n calibrateSoundBurstFilteredExtraDb\n }) => {\n let retryCount = 0;\n let response = null;\n\n while (retryCount < this.MAX_RETRY_COUNT) {\n try {\n response = await this.getComponentInverseImpulseResponse({\n payload,\n mls,\n lowHz,\n highHz,\n componentIRGains,\n iirLength,\n componentIRFreqs,\n sampleRate,\n mlsAmplitude,\n irLength,\n calibrateSoundSmoothOctaves,\n calibrateSoundBurstFilteredExtraDb\n });\n // If the request is successful, break out of the loop\n break;\n } catch (error) {\n console.error(`Error occurred. Retrying... (${retryCount + 1}/${this.MAX_RETRY_COUNT})`);\n retryCount++;\n await new Promise(resolve => setTimeout(resolve, this.RETRY_DELAY_MS));\n }\n }\n\n if (response) {\n return response;\n } else {\n throw new Error(\n `Failed to get inverse impulse response after ${this.MAX_RETRY_COUNT} attempts.`\n );\n }\n };\n\n getSystemInverseImpulseResponseWithRetry = async ({\n payload,\n mls,\n lowHz,\n highHz,\n iirLength,\n sampleRate,\n mlsAmplitude,\n calibrateSoundBurstFilteredExtraDb\n }) => {\n let retryCount = 0;\n let response = null;\n\n while (retryCount < this.MAX_RETRY_COUNT) {\n try {\n response = await this.getSystemInverseImpulseResponse({\n payload,\n mls,\n lowHz,\n highHz,\n iirLength,\n sampleRate,\n mlsAmplitude,\n calibrateSoundBurstFilteredExtraDb\n });\n // If the request is successful, break out of the loop\n break;\n } catch (error) {\n console.error(`Error occurred. Retrying... (${retryCount + 1}/${this.MAX_RETRY_COUNT})`);\n retryCount++;\n await new Promise(resolve => setTimeout(resolve, this.RETRY_DELAY_MS));\n }\n }\n\n if (response) {\n return response;\n } else {\n throw new Error(\n `Failed to get inverse impulse response after ${this.MAX_RETRY_COUNT} attempts.`\n );\n }\n };\n\n getVolumeCalibration = async ({payload, sampleRate, lCalib}) => {\n const task = 'volume';\n let res = null;\n\n console.log({payload});\n\n const data = JSON.stringify({\n task,\n payload,\n 'sample-rate': sampleRate,\n lCalib,\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL,\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n })\n .catch(error => {\n throw error;\n });\n\n return res.data[task];\n };\n\n getVolumeCalibrationParameters = async ({\n inDBValues,\n outDBSPLValues,\n lCalib,\n componentGainDBSPL,\n }) => {\n const task = 'volume-parameters';\n let res = null;\n\n const data = JSON.stringify({\n task,\n inDBValues,\n outDBSPLValues,\n lCalib,\n componentGainDBSPL,\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL, //server\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n console.log(res.data[task]);\n })\n .catch(error => {\n throw error;\n });\n\n // console.log(res.data[task]);\n //below is an example of res.data[task]\n //{\n // R: 16.56981076554259,\n // RMSError: 1.9289162528535229\n // T: -47.79799120884434,\n // W: 61.0485247483732,\n // backgroundDBSPL: 43.88233142069752,\n // gainDBSPL: -128.24742161208985\n //}\n return res.data[task];\n };\n\n allHzPowerCheck = async ({\n payload, \n sampleRate, \n binDesiredSec,\n burstSec,\n repeats\n }) => {\n const task = 'all-hz-check';\n let res = null;\n\n const data = JSON.stringify({\n payload, \n sampleRate, \n binDesiredSec,\n burstSec,\n repeats\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL, //server\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n console.log(res.data[task]);\n })\n .catch(error => {\n throw error;\n });\n return res.data[task];\n };\n\n volumePowerCheck = async ({\n payload, \n sampleRate, \n preSec,\n Sec,\n binDesiredSec\n }) => {\n const task = 'volume-check';\n let res = null;\n\n const data = JSON.stringify({\n payload, \n sampleRate, \n preSec,\n Sec,\n binDesiredSec\n });\n\n await axios__WEBPACK_IMPORTED_MODULE_0___default()({\n method: 'post',\n baseURL: PythonServerAPI.PYTHON_SERVER_URL, //server\n url: `/task/${task}`,\n headers: {\n 'Content-Type': 'application/json',\n },\n data,\n })\n .then(response => {\n res = response;\n console.log(res.data[task]);\n })\n .catch(error => {\n throw error;\n });\n return res.data[task];\n };\n}\n\n/* harmony default export */ __webpack_exports__[\"default\"] = (PythonServerAPI);\n\n\n//# sourceURL=webpack://speakerCalibrator/./src/server/PythonServerAPI.js?");
822
822
 
823
823
  /***/ }),
824
824
 
@@ -851,7 +851,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _myE
851
851
  /***/ (function(__unused_webpack_module, __webpack_exports__, __webpack_require__) {
852
852
 
853
853
  "use strict";
854
- eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _audioCalibrator__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../audioCalibrator */ \"./src/tasks/audioCalibrator.js\");\n/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../utils */ \"./src/utils.js\");\n/* harmony import */ var _config_firebase__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../../config/firebase */ \"./src/config/firebase.js\");\n/* harmony import */ var firebase_database__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! firebase/database */ \"./node_modules/firebase/database/dist/esm/index.esm.js\");\n/* harmony import */ var firebase_firestore__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! firebase/firestore */ \"./node_modules/firebase/firestore/dist/esm/index.esm.js\");\n\n\n\n\n\n\n\n/**\n *\n */\nclass Combination extends _audioCalibrator__WEBPACK_IMPORTED_MODULE_0__[\"default\"] {\n /**\n * Default constructor. Creates an instance with any number of paramters passed or the default parameters defined here.\n *\n * @param {Object<boolean, number, number, number>} calibratorParams - paramter object\n * @param {boolean} [calibratorParams.download = false] - boolean flag to download captures\n * @param {number} [calibratorParams.mlsOrder = 18] - order of the MLS to be generated\n * @param {number} [calibratorParams.numCaptures = 5] - number of captures to perform\n * @param {number} [calibratorParams.numMLSPerCapture = 2] - number of bursts of MLS per capture\n */\n constructor({\n download = false,\n mlsOrder = 18,\n numCaptures = 3,\n numMLSPerCapture = 2,\n lowHz = 20,\n highHz = 10000,\n }) {\n super(numCaptures, numMLSPerCapture);\n this.#mlsOrder = parseInt(mlsOrder, 10);\n this.#P = 2 ** mlsOrder - 1;\n this.#download = download;\n this.#mls = [];\n this.#lowHz = lowHz;\n this.#highHz = highHz;\n }\n\n /** @private */\n stepNum = 0;\n\n /** @private */\n totalSteps = 25;\n\n /** @private */\n #download;\n\n /** @private */\n #mlsGenInterface;\n\n /** @private */\n #mlsBufferView;\n\n /** @private */\n componentInvertedImpulseResponse = null;\n\n /** @private */\n systemInvertedImpulseResponse = null;\n\n //averaged and subtracted ir returned from calibration used to calculated iir\n /** @private */\n ir = null;\n\n /** @private */\n impulseResponses = [];\n\n /** @private */\n #mlsOrder;\n\n /** @private */\n #lowHz;\n\n /** @private */\n #highHz;\n\n /** @private */\n #mls;\n\n /** @private */\n #P;\n\n /** @private */\n #audioContext;\n\n /** @private */\n TAPER_SECS = 5;\n\n /** @private */\n offsetGainNode;\n\n /** @private */\n componentConvolution;\n\n /** @private */\n componentConvolutionNoBandpass;\n\n /** @private */\n componentIROrigin = {\n Freq: [],\n Gain: [],\n };\n\n /** @private */\n systemConvolution;\n\n /** @private */\n systemConvolutionNoBandpass;\n\n ////////////////////////volume\n /** @private */\n #CALIBRATION_TONE_FREQUENCY = 1000; // Hz\n\n /** @private */\n #CALIBRATION_TONE_TYPE = 'sine';\n\n CALIBRATION_TONE_DURATION = 5; // seconds\n calibrateSound1000HzPreSec = 3.5;\n calibrateSound1000HzSec = 1.0;\n calibrateSound1000HzPostSec = 0.5;\n\n /** @private */\n outDBSPL = null;\n THD = null;\n outDBSPL1000 = null;\n\n /** @private */\n TAPER_SECS = 0.01; // seconds\n\n /** @private */\n status_denominator = 8;\n\n /** @private */\n status_numerator = 0;\n\n /** @private */\n percent_complete = 0;\n\n /** @private */\n status = ``;\n\n /**@private */\n status_literal = `<div style=\"display: flex; justify-content: center;\"><div style=\"width: 200px; height: 20px; border: 2px solid #000; border-radius: 10px;\"><div style=\"width: ${this.percent_complete}%; height: 100%; background-color: #00aaff; border-radius: 8px;\"></div></div></div>`;\n\n /**@private */\n componentIR = null;\n\n /**@private */\n oldComponentIR = null;\n\n /**@private */\n systemIR = null;\n\n /**@private */\n _calibrateSoundCheck = '';\n\n deviceType = null;\n\n deviceName = null;\n\n deviceInfo = null;\n\n desired_time_per_mls = 0;\n\n num_mls_to_skip = 0;\n\n desired_sampling_rate = 0;\n\n #currentConvolution = [];\n\n mode = 'unfiltered';\n\n sourceNode;\n\n autocorrelations = [];\n\n iirLength = 0;\n\n irLength = 0;\n\n componentInvertedImpulseResponseNoBandpass = [];\n\n componentIRInTimeDomain = [];\n\n systemInvertedImpulseResponseNoBandpass = [];\n\n _calibrateSoundBackgroundSecs;\n\n _calibrateSoundSmoothOctaves;\n\n background_noise = {};\n\n numSuccessfulBackgroundCaptured;\n\n _calibrateSoundBurstDb;\n\n _calibrateSoundBurstFilteredExtraDb;\n\n _calibrateSoundBurstLevelReTBool;\n\n SDofFilteredRange = {\n mls: undefined,\n component: undefined,\n system: undefined,\n };\n\n transducerType = 'Loudspeaker';\n\n componentIRPhase = [];\n\n systemIRPhase = [];\n\n webAudioDeviceNames = {loudspeaker: '', microphone: '', loudspeakerText: '', microphoneText: ''};\n\n recordingChecks = {\n volume: {},\n unfiltered: [],\n system: [],\n component: [],\n };\n\n inDB;\n\n soundCheck = '';\n\n filteredMLSRange = {\n component: {\n Min: null,\n Max: null,\n },\n system: {\n Min: null,\n Max: null,\n },\n };\n\n /** @private */\n timeStamp = [];\n\n \n\n restartCalibration = false;\n\n calibrateSoundLimit = 1;\n\n filteredMLSAttenuation = {\n component: 1,\n system: 1,\n maxAbsSystem: 1,\n maxAbsComponent: 1,\n };\n\n //parameter result from volume calibration\n T = 0;\n //gainDBSPL result from volume calibration\n gainDBSPL = 0;\n //not always just using _calibrateSoundBurstDb for MLS so created a new parameter\n power_dB = 0;\n\n //system\n systemAttenuatorGainDB = 0;\n systemFMaxHz = 0;\n\n //component\n componentAttentuatorGainDB = 0;\n componentFMaxHz = 0;\n\n L_new_n;\n fs2;\n\n /**generate string template that gets reevaluated as variable increases */\n generateTemplate = () => {\n if (this.isCalibrating) {\n return '';\n }\n if (this.percent_complete > 100) {\n this.percent_complete = 100;\n }\n let MLSsd = '';\n let componentSD = '';\n let systemSD = '';\n let flags = '';\n const reportWebAudioNames = `<br>${this.webAudioDeviceNames.loudspeakerText} <br> ${this.webAudioDeviceNames.microphoneText}`;\n const reportParameters = `<br> Sampling: Loudspeaker ${this.sourceSamplingRate} Hz, Microphone ${this.sinkSamplingRate} Hz, ${this.sampleSize} bits`;\n if (this.flags) {\n flags = `<br> autoGainControl: ${this.flags.autoGainControl} \n <br>echoCancellation: ${this.flags.echoCancellation}\n <br>noiseSuppression: ${this.flags.noiseSuppression}`\n }\n if (this.SDofFilteredRange['mls']) {\n MLSsd = `<br> Recorded MLS power SD: ${this.SDofFilteredRange['mls']} dB`;\n }\n if (this.SDofFilteredRange['system']) {\n systemSD = `<br> Loudspeaker+Microphone correction SD: ${this.SDofFilteredRange['system']} dB`;\n }\n if (this.SDofFilteredRange['component']) {\n componentSD = `<br> ${this.transducerType} correction SD: ${this.SDofFilteredRange['component']} dB`;\n }\n const template = `<div style=\"display: flex; justify-content: center; margin-top:12px;\"><div style=\"width: 800px; height: 20px; border: 2px solid #000; border-radius: 10px;\"><div style=\"width: ${this.percent_complete}%; height: 100%; background-color: #00aaff; border-radius: 8px;\"></div></div></div>`;\n return reportWebAudioNames + reportParameters + flags + MLSsd + systemSD + componentSD + template;\n };\n\n /** increment numerator and percent for status bar */\n incrementStatusBar = () => {\n this.status_numerator += 1;\n this.percent_complete = (this.status_numerator / this.status_denominator) * 100;\n };\n\n setDeviceType = deviceType => {\n this.deviceType = deviceType;\n };\n\n setDeviceName = deviceName => {\n this.deviceName = deviceName;\n };\n\n setDeviceInfo = deviceInfo => {\n this.deviceInfo = deviceInfo;\n };\n\n /** .\n * .\n * .\n * Sends all the computed impulse responses to the backend server for processing\n *\n * @returns sets the resulting inverted impulse response to the class property\n * @example\n */\n sendSystemImpulseResponsesToServerForProcessing = async () => {\n this.addTimeStamp('Get system iir');\n const computedIRs = await Promise.all(this.impulseResponses);\n const filteredComputedIRs = computedIRs.filter(element => {\n return element != undefined;\n }); //log any errors that are found in this step\n const mls = this.#mls;\n const lowHz = this.#lowHz; //gain of 1 below cutoff, need gain of 0\n const highHz = this.#highHz; //check error for anything other than 10 kHz\n const iirLength = this.iirLength;\n const num_periods = this.numMLSPerCapture + this.num_mls_to_skip;\n this.stepNum += 1;\n console.log('send impulse responses to server: ' + this.stepNum);\n this.status =\n `All Hz Calibration: computing the IIR...`.toString() + this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return this.pyServerAPI\n .getSystemInverseImpulseResponseWithRetry({\n payload: filteredComputedIRs.slice(0, this.numCaptures),\n mls,\n lowHz,\n highHz,\n iirLength,\n num_periods,\n sampleRate: this.sourceSamplingRate || 96000,\n mlsAmplitude: Math.pow(10, this.power_dB / 20),\n calibrateSoundBurstFilteredExtraDb: this._calibrateSoundBurstFilteredExtraDb,\n })\n .then(res => {\n this.stepNum += 1;\n console.log('got impulse response ' + this.stepNum);\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the IIR...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n this.systemInvertedImpulseResponse = res['iir'];\n this.systemIR = res['ir'];\n this.systemConvolution = res['convolution'];\n this.systemInvertedImpulseResponseNoBandpass = res['iirNoBandpass'];\n this.systemAttenuatorGainDB = res['attenuatorGain_dB'];\n this.systemFMaxHz = res['fMaxHz'];\n this.systemConvolutionNoBandpass = res['convolutionNoBandpass'];\n\n // attenuate the system convolution if the amplitude is greater than this.calibrateSoundLimit\n // find max of absolute value of system convolution\n\n const max = Math.max(...this.systemConvolution.map(Math.abs));\n // if (max > this.calibrateSoundLimit) {\n // const gain = this.calibrateSoundLimit / max;\n // // apply gain to system convolution\n // this.systemConvolution = this.systemConvolution.map(value => value * gain);\n // this.filteredMLSAttenuation.system = gain;\n // }\n this.filteredMLSAttenuation.system = \n this.systemConvolution.reduce((a, b) => a + b**2, 0) / this.systemConvolution.length;\n this.filteredMLSAttenuation.maxAbsSystem = max;\n })\n .catch(err => {\n console.error(err);\n });\n };\n\n /** .\n * .\n * .\n * Sends all the computed impulse responses to the backend server for processing\n *\n * @returns sets the resulting inverted impulse response to the class property\n * @example\n */\n sendComponentImpulseResponsesToServerForProcessing = async () => {\n this.addTimeStamp('Get component iir');\n const computedIRs = await Promise.all(this.impulseResponses);\n const filteredComputedIRs = computedIRs.filter(element => {\n return element != undefined;\n });\n const componentIRGains = this.componentIR['Gain'];\n const componentIRFreqs = this.componentIR['Freq'];\n const mls = this.#mls;\n const lowHz = this.#lowHz;\n const iirLength = this.iirLength;\n const irLength = this.irLength;\n const num_periods = this.numMLSPerCapture + this.num_mls_to_skip;\n const highHz = this.#highHz;\n this.stepNum += 1;\n console.log('send impulse responses to server: ' + this.stepNum);\n this.status =\n `All Hz Calibration: computing the IIR...`.toString() + this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return this.pyServerAPI\n .getComponentInverseImpulseResponseWithRetry({\n payload: filteredComputedIRs.slice(0, this.numCaptures),\n mls,\n lowHz,\n highHz,\n iirLength,\n componentIRGains,\n componentIRFreqs,\n num_periods,\n sampleRate: this.sourceSamplingRate || 96000,\n mlsAmplitude: Math.pow(10, this.power_dB / 20),\n irLength,\n calibrateSoundSmoothOctaves: this._calibrateSoundSmoothOctaves,\n calibrateSoundBurstFilteredExtraDb: this._calibrateSoundBurstFilteredExtraDb,\n })\n .then(res => {\n this.stepNum += 1;\n console.log('got impulse response ' + this.stepNum);\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the IIR...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n this.componentInvertedImpulseResponse = res['iir'];\n this.componentIR['Gain'] = res['ir'];\n this.componentIR['Freq'] = res['frequencies'];\n this.componentIRPhase = res['component_angle'];\n this.systemIRPhase = res['system_angle'];\n this.componentIROrigin['Freq'] = res['frequencies'];\n this.componentIROrigin['Gain'] = res['irOrigin'];\n this.componentConvolution = res['convolution'];\n this.componentConvolutionNoBandpass = res['convolutionNoBandpass'];\n this.componentInvertedImpulseResponseNoBandpass = res['iirNoBandpass'];\n this.componentIRInTimeDomain = res['irTime'];\n this.componentAttenuatorGainDB = res['attenuatorGain_dB'];\n this.componentFMaxHz = res['fMaxHz'];\n\n // attenuate the component convolution if the amplitude is greater than this.calibrateSoundLimit\n // find max of absolute value of component convolution\n const max = Math.max(...this.componentConvolution.map(Math.abs));\n // if (max > this.calibrateSoundLimit) {\n // const gain = this.calibrateSoundLimit / max;\n // // apply gain to component convolution\n // this.componentConvolution = this.componentConvolution.map(value => value * gain);\n // this.filteredMLSAttenuation.component = gain;\n // }\n this.filteredMLSAttenuation.component = \n this.componentConvolution.reduce((a, b) => a + b**2, 0) / this.componentConvolution.length;\n this.filteredMLSAttenuation.maxAbsComponent = max;\n })\n .catch(err => {\n // this.emit('InvertedImpulseResponse', {res: false});\n console.error(err);\n });\n };\n\n sendBackgroundRecording = () => {\n const allSignals = this.getAllBackgroundRecordings();\n const numSignals = allSignals.length;\n const background_rec_whole = allSignals[numSignals - 1];\n const fraction = 0.5 / (this._calibrateSoundBackgroundSecs + 0.5);\n // Calculate the starting index for slicing the array\n const startIndex = Math.round(fraction * background_rec_whole.length);\n // Slice the array from the calculated start index to the end of the array\n const background_rec = background_rec_whole.slice(startIndex);\n console.log('Sending background recording to server for processing');\n this.addTimeStamp('Get background PSD');\n this.pyServerAPI\n .getBackgroundNoisePSDWithRetry({\n background_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n if (this.numSuccessfulBackgroundCaptured < 1) {\n this.numSuccessfulBackgroundCaptured += 1;\n //storing all background data in background_psd object\n this.background_noise['x_background'] = res['x_background'];\n this.background_noise['y_background'] = res['y_background'];\n this.background_noise['recording'] = background_rec;\n }\n })\n .catch(err => {\n console.error(err);\n });\n };\n\n /** .\n * .\n * .\n * Sends the recorded signal, or a given csv string of a signal, to the back end server for processing\n *\n * @param {<array>String} signalCsv - Optional csv string of a previously recorded signal, if given, this signal will be processed\n * @example\n */\n sendRecordingToServerForProcessing = async signalCsv => {\n const allSignals = this.getAllUnfilteredRecordedSignals();\n console.log(\n 'Obtaining last all hz unfiltered recording from #allHzUnfilteredRecordings to send to server for processing'\n );\n const numSignals = allSignals.length;\n const mls = this.#mlsBufferView;\n const payload =\n signalCsv && signalCsv.length > 0 ? (0,_utils__WEBPACK_IMPORTED_MODULE_1__.csvToArray)(signalCsv) : allSignals[numSignals - 1];\n console.log('sending rec');\n this.stepNum += 1;\n console.log('send rec ' + this.stepNum);\n this.status =\n `All Hz Calibration Step: computing the IR of the last recording...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n if (this.isCalibrating) return null;\n await this.pyServerAPI\n .allHzPowerCheck({\n payload,\n sampleRate: this.sourceSamplingRate || 96000,\n binDesiredSec: this._calibrateSoundPowerBinDesiredSec,\n burstSec: this.desired_time_per_mls,\n repeats: this.numMLSPerCapture,\n })\n .then(result => {\n if (result) {\n if (result['sd'] < this._calibrateSoundPowerDbSDToleratedDb) {\n this.recordingChecks['unfiltered'].push(result);\n this.impulseResponses.push(\n this.pyServerAPI\n .getImpulseResponse({\n sampleRate: this.sourceSamplingRate || 96000,\n payload,\n mls,\n P: this.#P, //get rid of this\n numPeriods: this.numMLSPerCapture,\n })\n .then(res => {\n if (this.numSuccessfulCaptured < this.numCaptures) {\n this.numSuccessfulCaptured += 1;\n console.log('num succ capt: ' + this.numSuccessfulCaptured);\n this.stepNum += 1;\n console.log('got impulse response ' + this.stepNum);\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: ${this.numSuccessfulCaptured}/${this.numCaptures} IRs computed...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {\n message: this.status,\n });\n this.autocorrelations.push(res['autocorrelation']);\n this.L_new_n = res['L_new_n'];\n this.fs2 = res['fs2'];\n return res['ir'];\n }\n })\n .catch(err => {\n console.error(err);\n })\n );\n } else if (result['sd'] > this._calibrateSoundPowerDbSDToleratedDb) {\n this.clearLastUnfilteredRecordedSignals();\n console.log('unfiltered rec', this.getAllUnfilteredRecordedSignals.length);\n }\n }\n })\n .catch(err => {\n console.error(err);\n });\n };\n\n /**\n * Passed to the calibration steps function, awaits the desired amount of seconds to capture the desired number\n * of MLS periods defined in the constructor.\n *\n * @example\n */\n #awaitDesiredMLSLength = async () => {\n // seconds per MLS = P / SR\n // await N * P / SR\n this.stepNum += 1;\n console.log('await desired length ' + this.stepNum);\n this.status =\n `All Hz Calibration: sampling the calibration signal...`.toString() +\n `\\niteration ${this.stepNum}` +\n this.generateTemplate();\n this.emit('update', {\n message: this.status,\n });\n let time_to_wait = 0;\n if (this.mode === 'unfiltered') {\n //unfiltered\n time_to_wait = (this.#mls.length / this.sourceSamplingRate) * this.numMLSPerCapture;\n time_to_wait = time_to_wait * 1.1;\n } else if (this.mode === 'filtered') {\n //filtered\n // time_to_wait =\n // (this.#currentConvolution.length / this.sourceSamplingRate) *\n // (this.numMLSPerCapture / (this.num_mls_to_skip + this.numMLSPerCapture));\n time_to_wait =\n (this.#currentConvolution.length / this.sourceSamplingRate) * this.numMLSPerCapture;\n time_to_wait = time_to_wait * 1.1;\n } else {\n throw new Error('Mode broke in awaitDesiredMLSLength');\n }\n\n await (0,_utils__WEBPACK_IMPORTED_MODULE_1__.sleep)(time_to_wait);\n };\n\n /**\n * Passed to the background noise recording function, awaits the desired amount of seconds to capture the desired number\n * of seconds of background noise\n *\n * @example\n */\n #awaitBackgroundNoiseRecording = async () => {\n console.log(\n 'Waiting ' + this._calibrateSoundBackgroundSecs + ' second(s) to record background noise'\n );\n let time_to_wait = this._calibrateSoundBackgroundSecs + 0.5;\n await (0,_utils__WEBPACK_IMPORTED_MODULE_1__.sleep)(time_to_wait);\n };\n\n /** .\n * .\n * .\n * Passed to the calibration steps function, awaits the onset of the signal to ensure a steady state\n *\n * @example\n */\n #awaitSignalOnset = async () => {\n this.stepNum += 1;\n console.log('await signal onset ' + this.stepNum);\n this.status =\n `All Hz Calibration: waiting for the signal to stabilize...`.toString() +\n this.generateTemplate();\n this.emit('update', {\n message: this.status,\n });\n let number_of_bursts_to_skip = this.num_mls_to_skip;\n let time_to_sleep = 0;\n if (this.mode === 'unfiltered') {\n time_to_sleep = (this.#mls.length / this.sourceSamplingRate) * number_of_bursts_to_skip;\n } else if (this.mode === 'filtered') {\n console.log(this.#currentConvolution.length);\n // time_to_sleep =\n // (this.#currentConvolution.length / this.sourceSamplingRate) *\n // (number_of_bursts_to_skip / (number_of_bursts_to_skip + this.numMLSPerCapture));\n time_to_sleep =\n (this.#currentConvolution.length / this.sourceSamplingRate) * number_of_bursts_to_skip;\n } else {\n throw new Error('Mode broke in awaitSignalOnset');\n }\n await (0,_utils__WEBPACK_IMPORTED_MODULE_1__.sleep)(time_to_sleep);\n };\n\n /**\n * Called immediately after a recording is captured. Used to process the resulting signal\n * whether by sending the result to a server or by computing a result locally.\n *\n * @example\n */\n #afterMLSRecord = async () => {\n console.log('after record');\n await this.sendRecordingToServerForProcessing();\n };\n\n #afterMLSwIIRRecord = async () => {\n await this.checkPowerVariation();\n };\n\n /** .\n * .\n * .\n * Created an S Curver Buffer to taper the signal onset\n *\n * @param {*} length\n * @param {*} phase\n * @returns\n * @example\n */\n static createSCurveBuffer = (length, phase) => {\n const curve = new Float32Array(length);\n let i;\n for (i = 0; i < length; i += 1) {\n // scale the curve to be between 0-1\n curve[i] = Math.sin((Math.PI * i) / length - phase) / 2 + 0.5;\n }\n return curve;\n };\n\n static createInverseSCurveBuffer = (length, phase) => {\n const curve = new Float32Array(length);\n let i;\n let j = length - 1;\n for (i = 0; i < length; i += 1) {\n // scale the curve to be between 0-1\n curve[i] = Math.sin((Math.PI * j) / length - phase) / 2 + 0.5;\n j -= 1;\n }\n return curve;\n };\n\n /**\n * Construct a Calibration Node with the calibration parameters.\n *\n * @param dataBuffer\n * @private\n * @example\n */\n #createCalibrationNodeFromBuffer = dataBuffer => {\n console.log('length databuffer');\n console.log(dataBuffer.length);\n if (!this.sourceAudioContext) {\n this.makeNewSourceAudioContext();\n }\n\n const buffer = this.sourceAudioContext.createBuffer(\n 1, // number of channels\n dataBuffer.length,\n this.sourceAudioContext.sampleRate // sample rate\n );\n\n const data = buffer.getChannelData(0); // get data\n\n // fill the buffer with our data\n try {\n for (let i = 0; i < dataBuffer.length; i += 1) {\n data[i] = dataBuffer[i];\n }\n } catch (error) {\n console.error(error);\n }\n\n this.sourceNode = this.sourceAudioContext.createBufferSource();\n\n this.sourceNode.buffer = buffer;\n\n if (this.mode === 'filtered') {\n //used to not loop filtered\n this.sourceNode.loop = true;\n } else {\n this.sourceNode.loop = true;\n }\n\n this.sourceNode.connect(this.sourceAudioContext.destination);\n\n this.addCalibrationNode(this.sourceNode);\n };\n\n /**\n * Given a data buffer, creates the required calibration node\n *\n * @param {*} dataBufferArray\n * @example\n */\n #setCalibrationNodesFromBuffer = (dataBufferArray = [this.#mlsBufferView]) => {\n if (dataBufferArray.length === 1) {\n this.#createCalibrationNodeFromBuffer(dataBufferArray[0]);\n } else {\n throw new Error('The length of the data buffer array must be 1');\n }\n };\n\n /**\n * Creates an audio context and plays it for a few seconds.\n *\n * @private\n * @returns - Resolves when the audio is done playing.\n * @example\n */\n #playCalibrationAudio = () => {\n this.addTimeStamp('Play unfiltered mls');\n this.calibrationNodes[0].start(0);\n this.status = ``;\n if (this.mode === 'unfiltered') {\n console.log('play calibration audio ' + this.stepNum);\n this.status =\n `All Hz Calibration: playing the calibration tone...`.toString() +\n this.generateTemplate().toString();\n } else if (this.mode === 'filtered') {\n console.log('play convolved audio ' + this.stepNum);\n this.status =\n `All Hz Calibration: playing the convolved calibration tone...`.toString() +\n this.generateTemplate().toString();\n } else {\n throw new Error('Mode is incorrect');\n }\n this.emit('update', {message: this.status});\n this.stepNum += 1;\n console.log('sink sampling rate');\n console.log(this.sinkSamplingRate);\n console.log('source sampling rate');\n console.log(this.sourceSamplingRate);\n console.log('sample size');\n console.log(this.sampleSize);\n };\n\n /** .\n * .\n * .\n * Stops the audio with tapered offset\n *\n * @example\n */\n stopCalibrationAudio = () => {\n if (this.calibrationNodes.length === 0) {\n return;\n }\n this.calibrationNodes[0].stop(0);\n this.calibrationNodes = [];\n if (this.sourceNode) this.sourceNode.disconnect();\n this.stepNum += 1;\n console.log('stop calibration audio ' + this.stepNum);\n this.status =\n `All Hz Calibration: stopping the calibration tone...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n };\n\n playMLSwithIIR = async (stream, convolution) => {\n let checkRec = false;\n this.mode = 'filtered';\n console.log('play mls with iir');\n //this.invertedImpulseResponse = iir\n\n await this.calibrationSteps(\n stream,\n this.#playCalibrationAudio, // play audio func (required)\n this.#createCalibrationNodeFromBuffer(convolution), // before play func\n this.#awaitSignalOnset, // before record\n () => this.numSuccessfulCaptured < 1,\n this.#awaitDesiredMLSLength, // during record\n this.#afterMLSwIIRRecord, // after record\n this.mode,\n checkRec\n );\n };\n\n bothSoundCheck = async stream => {\n let iir_ir_and_plots;\n this.#currentConvolution = this.componentConvolution;\n this.filteredMLSRange.component.Min = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMinValue)(this.#currentConvolution);\n this.filteredMLSRange.component.Max = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMaxValue)(this.#currentConvolution);\n this.addTimeStamp('Play MLS with component IIR');\n this.soundCheck = 'component';\n if (this.isCalibrating) return null;\n await this.playMLSwithIIR(stream, this.#currentConvolution);\n this.stopCalibrationAudio();\n let component_conv_recs = this.getAllFilteredRecordedSignals();\n\n if (this.componentAttentuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.componentAttentuatorGainDB/20);\n component_conv_recs = component_conv_recs.map(rec => {\n return rec.map(value => value / this.linearScaleAttenuation);\n });\n }\n\n let return_component_conv_rec = component_conv_recs[component_conv_recs.length - 1];\n this.clearAllFilteredRecordedSignals();\n // await this.checkPowerVariation(return_component_conv_rec);\n this.numSuccessfulCaptured = 0;\n this.#currentConvolution = this.systemConvolution;\n this.filteredMLSRange.system.Min = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMinValue)(this.#currentConvolution);\n this.filteredMLSRange.system.Max = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMaxValue)(this.#currentConvolution);\n this.soundCheck = 'system';\n this.addTimeStamp('Play MLS with system IIR');\n if (this.isCalibrating) return null;\n await this.playMLSwithIIR(stream, this.#currentConvolution);\n\n this.stopCalibrationAudio();\n\n let system_conv_recs = this.getAllFilteredRecordedSignals();\n\n if (this.systemAttenuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.systemAttenuatorGainDB/20);\n system_conv_recs = system_conv_recs.map(rec => {\n return rec.map(value => value / linearScaleAttenuation);\n });\n }\n\n let return_system_conv_rec = system_conv_recs[system_conv_recs.length - 1];\n // await this.checkPowerVariation(return_system_conv_rec);\n\n this.clearAllFilteredRecordedSignals();\n\n this.sourceAudioContext.close();\n let recs = this.getAllUnfilteredRecordedSignals();\n if (this.componentAttentuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.componentAttentuatorGainDB/20);\n recs = recs.map(rec => {\n return rec.map(value => value / this.linearScaleAttenuation);\n });\n }\n let unconv_rec = recs[0];\n let return_unconv_rec = unconv_rec;\n let conv_rec = component_conv_recs[component_conv_recs.length - 1];\n\n //psd of component\n let knownGain = this.oldComponentIR.Gain;\n let knownFreq = this.oldComponentIR.Freq;\n let sampleRate = this.sourceSamplingRate || 96000;\n this.addTimeStamp('Get PSD of mls recording');\n if (this.isCalibrating) return null;\n let component_unconv_rec_psd = await this.pyServerAPI\n .getSubtractedPSDWithRetry(unconv_rec, knownGain, knownFreq, sampleRate)\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD of filtered recording (component)');\n if (this.isCalibrating) return null;\n let component_conv_rec_psd = await this.pyServerAPI\n .getSubtractedPSDWithRetry(conv_rec, knownGain, knownFreq, sampleRate)\n .then(res => {\n let interpolatedGain = res.x.map((freq, index) => {\n let i = 0;\n while (i < knownFreq.length && knownFreq[i] < freq) {\n i++;\n }\n if (i === 0 || i === knownFreq.length) {\n return knownGain[i];\n }\n return (0,_utils__WEBPACK_IMPORTED_MODULE_1__.interpolate)(freq, knownFreq[i - 1], knownFreq[i], knownGain[i - 1], knownGain[i]);\n });\n\n let correctedGain = res.y.map(\n (gain, index) => 10 * Math.log10(gain) - interpolatedGain[index]\n );\n\n let filtered_psd = correctedGain.filter(\n (value, index) => res.x[index] >= this.#lowHz && res.x[index] <= this.componentFMaxHz\n );\n\n this.SDofFilteredRange['component'] = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.standardDeviation)(filtered_psd);\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n conv_rec = system_conv_recs[system_conv_recs.length - 1];\n //psd of system\n this.addTimeStamp('Get PSD of filtered recording (system) and unfiltered recording');\n if (this.isCalibrating) return null;\n let system_recs_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n let filtered_psd = res.y_conv\n .filter(\n (value, index) => res.x_conv[index] >= this.#lowHz && res.x_conv[index] <= this.#highHz\n )\n .map(value => 10 * Math.log10(value));\n\n let mls_psd = res.y_unconv\n .filter(\n (value, index) =>\n res.x_unconv[index] >= this.#lowHz && res.x_conv[index] <= this.#highHz\n )\n .map(value => 10 * Math.log10(value));\n\n this.SDofFilteredRange['mls'] = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.standardDeviation)(mls_psd);\n this.SDofFilteredRange['system'] = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.standardDeviation)(filtered_psd);\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n //iir w/ and without bandpass psd. done\n unconv_rec = this.componentInvertedImpulseResponseNoBandpass;\n conv_rec = this.componentInvertedImpulseResponse;\n this.addTimeStamp('Get PSD of component iir and component iir no band pass');\n if (this.isCalibrating) return null;\n let component_iir_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n unconv_rec = this.systemInvertedImpulseResponseNoBandpass;\n conv_rec = this.systemInvertedImpulseResponse;\n this.addTimeStamp('Get PSD of system iir and system iir no band pass');\n if (this.isCalibrating) return null;\n let system_iir_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD of mls sequence');\n if (this.isCalibrating) return null;\n let mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({mls: this.#mlsBufferView, sampleRate: this.sourceSamplingRate || 96000})\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD of filered mls (system)');\n if (this.isCalibrating) return null;\n let system_filtered_mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.systemConvolution,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n let system_no_bandpass_filtered_mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.systemConvolution,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD of filered mls (component)');\n if (this.isCalibrating) return null;\n let component_filtered_mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.componentConvolution,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n let component_no_bandpass_filtered_mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.componentConvolutionNoBandpass,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n let gainValue = this.getGainDBSPL();\n\n iir_ir_and_plots = {\n filtered_recording: {\n component: return_component_conv_rec,\n system: return_system_conv_rec,\n },\n unfiltered_recording: this.getAllUnfilteredRecordedSignals()[0],\n system: {\n iir: this.systemInvertedImpulseResponse,\n iir_no_bandpass: this.systemInvertedImpulseResponseNoBandpass,\n ir: this.systemIR,\n iir_psd: {\n y: system_iir_psd['y_conv'],\n x: system_iir_psd['x_conv'],\n y_no_bandpass: system_iir_psd['y_unconv'],\n x_no_bandpass: system_iir_psd['x_unconv'],\n },\n filtered_mls_psd: {\n x: system_filtered_mls_psd['x_mls'],\n y: system_filtered_mls_psd['y_mls'],\n },\n filtered_no_bandpass_mls_psd: {\n x: system_no_bandpass_filtered_mls_psd['x_mls'],\n y: system_no_bandpass_filtered_mls_psd['y_mls'],\n },\n convolution: this.systemConvolution,\n convolutionNoBandpass: this.systemConvolutionNoBandpass,\n psd: {\n unconv: {\n x: system_recs_psd['x_unconv'],\n y: system_recs_psd['y_unconv'],\n },\n conv: {\n x: system_recs_psd['x_conv'],\n y: system_recs_psd['y_conv'],\n },\n },\n },\n component: {\n iir: this.componentInvertedImpulseResponse,\n iir_no_bandpass: this.componentInvertedImpulseResponseNoBandpass,\n ir: this.componentIR,\n ir_origin: this.componentIROrigin,\n ir_in_time_domain: this.componentIRInTimeDomain,\n iir_psd: {\n y: component_iir_psd['y_conv'],\n x: component_iir_psd['x_conv'],\n y_no_bandpass: component_iir_psd['y_unconv'],\n x_no_bandpass: component_iir_psd['x_unconv'],\n },\n filtered_mls_psd: {\n x: component_filtered_mls_psd['x_mls'],\n y: component_filtered_mls_psd['y_mls'],\n },\n filtered_no_bandpass_mls_psd: {\n x: component_no_bandpass_filtered_mls_psd['x_mls'],\n y: component_no_bandpass_filtered_mls_psd['y_mls'],\n },\n convolution: this.componentConvolution,\n convolutionNoBandpass: this.componentConvolutionNoBandpass,\n psd: {\n unconv: {\n x: component_unconv_rec_psd['x'],\n y: component_unconv_rec_psd['y'],\n },\n conv: {\n x: component_conv_rec_psd['x'],\n y: component_conv_rec_psd['y'],\n },\n },\n gainDBSPL: gainValue,\n },\n mls: this.#mlsBufferView,\n mls_psd: {\n x: mls_psd['x_mls'],\n y: mls_psd['y_mls'],\n },\n autocorrelations: this.autocorrelations,\n impulseResponses: [],\n };\n\n return iir_ir_and_plots;\n };\n\n singleSoundCheck = async stream => {\n let iir_ir_and_plots;\n if (this._calibrateSoundCheck != 'system') {\n this.#currentConvolution = this.componentConvolution;\n this.filteredMLSRange.component.Min = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMinValue)(this.#currentConvolution);\n this.filteredMLSRange.component.Max = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMaxValue)(this.#currentConvolution);\n this.addTimeStamp('Play MLS with component IIR');\n this.soundCheck = 'component';\n if (this.isCalibrating) return null;\n await this.playMLSwithIIR(stream, this.#currentConvolution);\n this.stopCalibrationAudio();\n } else {\n this.#currentConvolution = this.systemConvolution;\n this.filteredMLSRange.system.Min = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMinValue)(this.#currentConvolution);\n this.filteredMLSRange.system.Max = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMaxValue)(this.#currentConvolution);\n this.addTimeStamp('Play MLS with system IIR');\n this.soundCheck = 'system';\n if (this.isCalibrating) return null;\n await this.playMLSwithIIR(stream, this.#currentConvolution);\n this.stopCalibrationAudio();\n }\n let conv_recs = this.getAllFilteredRecordedSignals();\n if (this._calibrateSoundCheck == 'goal'){\n if (this.componentAttentuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.componentAttentuatorGainDB/20);\n conv_recs = conv_recs.map(rec => {\n return rec.map(value => value / this.linearScaleAttenuation);\n });\n }\n }else if (this._calibrateSoundCheck == 'system'){\n if (this.systemAttentuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.systemAttentuatorGainDB/20);\n conv_recs = conv_recs.map(rec => {\n return rec.map(value => value / this.linearScaleAttenuation);\n });\n }\n }\n\n //remove the filteredMLSAttenuation from the recorded signals\n // conv_recs = conv_recs.map(rec => {\n // if (this.soundCheck === 'component') {\n // return rec.map(value => value / this.filteredMLSAttenuation.component);\n // }\n // return rec.map(value => value / this.filteredMLSAttenuation.system);\n // });\n\n\n let recs = this.getAllUnfilteredRecordedSignals();\n if (this._calibrateSoundCheck == 'goal'){\n if (this.componentAttentuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.componentAttentuatorGainDB/20);\n recs = recs.map(rec => {\n return rec.map(value => value / this.linearScaleAttenuation);\n });\n }\n }else if (this._calibrateSoundCheck == 'system'){\n if (this.systemAttentuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.systemAttentuatorGainDB/20);\n recs = recs.map(rec => {\n return rec.map(value => value / this.linearScaleAttenuation);\n });\n }\n }\n this.clearAllFilteredRecordedSignals();\n console.log('Obtaining unfiltered recording from #allHzUnfilteredRecordings to calculate PSD');\n console.log('Obtaining filtered recording from #allHzFilteredRecordings to calculate PSD');\n let unconv_rec = recs[0];\n let return_unconv_rec = unconv_rec;\n let conv_rec = conv_recs[conv_recs.length - 1];\n let return_conv_rec = conv_rec;\n this.sourceAudioContext.close();\n if (this._calibrateSoundCheck != 'system') {\n let knownGain = this.oldComponentIR.Gain;\n let knownFreq = this.oldComponentIR.Freq;\n let sampleRate = this.sourceSamplingRate || 96000;\n this.addTimeStamp('Get PSD of mls recording');\n if (this.isCalibrating) return null;\n let unconv_results = await this.pyServerAPI\n .getSubtractedPSDWithRetry(unconv_rec, knownGain, knownFreq, sampleRate)\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD recording of filtered recording (component)');\n if (this.isCalibrating) return null;\n let conv_results = await this.pyServerAPI\n .getSubtractedPSDWithRetry(conv_rec, knownGain, knownFreq, sampleRate)\n .then(res => {\n let interpolatedGain = res.x.map((freq, index) => {\n let i = 0;\n while (i < knownFreq.length && knownFreq[i] < freq) {\n i++;\n }\n if (i === 0 || i === knownFreq.length) {\n return knownGain[i];\n }\n return (0,_utils__WEBPACK_IMPORTED_MODULE_1__.interpolate)(\n freq,\n knownFreq[i - 1],\n knownFreq[i],\n knownGain[i - 1],\n knownGain[i]\n );\n });\n\n let correctedGain = res.y.map(\n (gain, index) => 10 * Math.log10(gain) - interpolatedGain[index]\n );\n let filtered_psd = correctedGain.filter(\n (value, index) => res.x[index] >= this.#lowHz && res.x[index] <= this.#highHz\n );\n\n this.SDofFilteredRange['component'] = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.standardDeviation)(filtered_psd);\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n unconv_rec = this.componentInvertedImpulseResponseNoBandpass;\n conv_rec = this.componentInvertedImpulseResponse;\n this.addTimeStamp('Get PSD of component iir and component iir no bandpass');\n if (this.isCalibrating) return null;\n let component_iir_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n unconv_rec = this.systemInvertedImpulseResponseNoBandpass;\n conv_rec = this.systemInvertedImpulseResponse;\n this.addTimeStamp('Get PSD of system iir and system iir no bandpass');\n if (this.isCalibrating) return null;\n let system_iir_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD of mls sequence');\n if (this.isCalibrating) return null;\n let mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.#mlsBufferView,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD of filtered mls (component)');\n if (this.isCalibrating) return null;\n let filtered_mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.componentConvolution,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n let filtered_no_bandpass_mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.componentConvolutionNoBandpass,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n let gainValue = this.getGainDBSPL();\n iir_ir_and_plots = {\n unfiltered_recording: return_unconv_rec,\n filtered_recording: return_conv_rec,\n system: {\n iir: this.systemInvertedImpulseResponse,\n iir_no_bandpass: this.systemInvertedImpulseResponseNoBandpass,\n ir: this.systemIR,\n iir_psd: {\n y: system_iir_psd['y_conv'],\n x: system_iir_psd['y_conv'],\n y_no_bandpass: system_iir_psd['y_unconv'],\n x_no_bandpass: system_iir_psd['x_unconv'],\n },\n filtered_recording: [],\n filtered_mls_psd: {},\n filtered_no_bandpass_mls_psd: {},\n convolution: this.systemConvolution,\n convolutionNoBandpass: this.systemConvolutionNoBandpass,\n psd: {\n unconv: {\n x: [],\n y: [],\n },\n conv: {\n x: [],\n y: [],\n },\n },\n },\n component: {\n iir: this.componentInvertedImpulseResponse,\n iir_no_bandpass: this.componentInvertedImpulseResponseNoBandpass,\n ir: this.componentIR,\n ir_origin: this.componentIROrigin,\n ir_in_time_domain: this.componentIRInTimeDomain,\n iir_psd: {\n y: component_iir_psd['y_conv'],\n x: component_iir_psd['x_conv'],\n y_no_bandpass: component_iir_psd['y_unconv'],\n x_no_bandpass: component_iir_psd['x_unconv'],\n },\n filtered_mls_psd: {\n x: filtered_mls_psd['x_mls'],\n y: filtered_mls_psd['y_mls'],\n },\n filtered_no_bandpass_mls_psd: {\n x: filtered_no_bandpass_mls_psd['x_mls'],\n y: filtered_no_bandpass_mls_psd['y_mls'],\n },\n convolution: this.componentConvolution,\n convolutionNoBandpass: this.componentConvolutionNoBandpass,\n psd: {\n unconv: {\n x: unconv_results['x'],\n y: unconv_results['y'],\n },\n conv: {\n x: conv_results['x'],\n y: conv_results['y'],\n },\n },\n gainDBSPL: gainValue,\n },\n mls: this.#mlsBufferView,\n mls_psd: {\n x: mls_psd['x_mls'],\n y: mls_psd['y_mls'],\n },\n autocorrelations: this.autocorrelations,\n impulseResponses: [],\n };\n } else {\n this.addTimeStamp('Get PSD of filtered recording (system) and unfiltered recording');\n if (this.isCalibrating) return null;\n let results = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n let filtered_psd = res.y_conv\n .filter(\n (value, index) =>\n res.x_conv[index] >= this.#lowHz && res.x_conv[index] <= this.systemFMaxHz\n )\n .map(value => 10 * Math.log10(value));\n\n let mls_psd = res.y_unconv\n .filter(\n (value, index) =>\n res.x_unconv[index] >= this.#lowHz && res.x_conv[index] <= this.systemFMaxHz\n )\n .map(value => 10 * Math.log10(value));\n\n this.SDofFilteredRange['system'] = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.standardDeviation)(filtered_psd);\n this.SDofFilteredRange['unfiltered'] = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.standardDeviation)(mls_psd);\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n //iir w/ and without bandpass psd\n unconv_rec = this.componentInvertedImpulseResponseNoBandpass;\n conv_rec = this.componentInvertedImpulseResponse;\n this.addTimeStamp('Get PSD of component iir and component iir no band pass');\n if (this.isCalibrating) return null;\n let component_iir_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n unconv_rec = this.systemInvertedImpulseResponseNoBandpass;\n conv_rec = this.systemInvertedImpulseResponse;\n this.addTimeStamp('Get PSD of system iir and system iir no band pass');\n if (this.isCalibrating) return null;\n let system_iir_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD of mls sequence');\n if (this.isCalibrating) return null;\n let mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.#mlsBufferView,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD of filtered mls (system)');\n if (this.isCalibrating) return null;\n let filtered_mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.systemConvolution,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n let filtered_no_bandpass_mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.systemConvolutionNoBandpass,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n let gainValue = this.getGainDBSPL();\n iir_ir_and_plots = {\n unfiltered_recording: return_unconv_rec,\n filtered_recording: return_conv_rec,\n system: {\n iir: this.systemInvertedImpulseResponse,\n iir_no_bandpass: this.systemInvertedImpulseResponseNoBandpass,\n ir: this.systemIR,\n iir_psd: {\n y: system_iir_psd['y_conv'],\n x: system_iir_psd['y_conv'],\n y_no_bandpass: system_iir_psd['y_unconv'],\n x_no_bandpass: system_iir_psd['x_unconv'],\n },\n filtered_recording: [],\n filtered_mls_psd: {\n x: filtered_mls_psd['x_mls'],\n y: filtered_mls_psd['y_mls'],\n },\n filtered_no_bandpass_mls_psd: {\n x: filtered_no_bandpass_mls_psd['x_mls'],\n y: filtered_no_bandpass_mls_psd['y_mls'],\n },\n convolution: this.systemConvolution,\n convolutionNoBandpass: this.systemConvolutionNoBandpass,\n psd: {\n unconv: {\n x: results['x_unconv'],\n y: results['y_unconv'],\n },\n conv: {\n x: results['x_conv'],\n y: results['y_conv'],\n },\n },\n },\n component: {\n iir: this.componentInvertedImpulseResponse,\n iir_no_bandpass: this.componentInvertedImpulseResponseNoBandpass,\n ir: this.componentIR,\n ir_origin: this.componentIROrigin,\n ir_in_time_domain: this.componentIRInTimeDomain,\n iir_psd: {\n y: component_iir_psd['y_conv'],\n x: component_iir_psd['x_conv'],\n y_no_bandpass: component_iir_psd['y_unconv'],\n x_no_bandpass: component_iir_psd['x_unconv'],\n },\n filtered_mls_psd: {},\n filtered_no_bandpass_mls_psd: {},\n convolution: this.componentConvolution,\n convolutionNoBandpass: this.componentConvolutionNoBandpass,\n psd: {\n unconv: {\n x: [],\n y: [],\n },\n conv: {\n x: [],\n y: [],\n },\n },\n gainDBSPL: gainValue,\n },\n mls: this.#mlsBufferView,\n mls_psd: {\n x: mls_psd['x_mls'],\n y: mls_psd['y_mls'],\n },\n autocorrelations: this.autocorrelations,\n impulseResponses: [],\n };\n }\n if (this.isCalibrating) return null;\n await Promise.all(this.impulseResponses).then(res => {\n for (let i = 0; i < res.length; i++) {\n if (res[i] != undefined) {\n iir_ir_and_plots['impulseResponses'].push(res[i]);\n }\n }\n });\n\n if (this.#download) {\n this.downloadSingleUnfilteredRecording();\n this.downloadSingleFilteredRecording();\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.#mls, 'MLS.csv');\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.componentConvolution, 'python_component_convolution_mls_iir.csv');\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.systemConvolution, 'python_system_convolution_mls_iir.csv');\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.componentInvertedImpulseResponse, 'componentIIR.csv');\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.systemInvertedImpulseResponse, 'systemIIR.csv');\n for (let i = 0; i < this.autocorrelations.length; i++) {\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.autocorrelations[i], `autocorrelation_${i}`);\n }\n const computedIRagain = await Promise.all(this.impulseResponses).then(res => {\n for (let i = 0; i < res.length; i++) {\n if (res[i] != undefined) {\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(res[i], `IR_${i}`);\n }\n }\n });\n }\n\n return iir_ir_and_plots;\n };\n\n /**\n * Public method to start the calibration process. Objects intialized from webassembly allocate new memory\n * and must be manually freed. This function is responsible for intializing the MlsGenInterface,\n * and wrapping the calibration steps with a garbage collection safe gaurd.\n *\n * @public\n * @param stream - The stream of audio from the Listener.\n * @example\n */\n startCalibrationImpulseResponse = async stream => {\n let desired_time = this.desired_time_per_mls;\n let checkRec = 'allhz';\n\n console.log('MLS sequence should be of length: ' + this.sourceSamplingRate * desired_time);\n\n length = this.sourceSamplingRate * desired_time;\n //get mls here\n // const calibrateSoundBurstDb = Math.pow(10, this._calibrateSoundBurstDb / 20);\n\n this.power_dB = 0;\n\n if (!this._calibrateSoundBurstLevelReTBool){\n this.power_dB =this._calibrateSoundBurstDb;\n }else{\n this.power_dB = this._calibrateSoundBurstDb+(this.T-this.gainDBSPL);\n }\n\n const amplitude = Math.pow(10,this.power_dB / 20);\n //MLSpower = Math.pow(10,this.power_dB/20);\n this.addTimeStamp('Get MLS sequence');\n if (this.isCalibrating) return null;\n await this.pyServerAPI\n .getMLSWithRetry({length, amplitude})\n .then(res => {\n console.log(res);\n this.#mlsBufferView = res['mls'];\n this.#mls = res['unscaledMLS'];\n })\n .catch(err => {\n // this.emit('InvertedImpulseResponse', {res: false});\n console.error(err);\n });\n this.numSuccessfulBackgroundCaptured = 0;\n if (this._calibrateSoundBackgroundSecs > 0) {\n this.mode = 'background';\n this.status =\n `All Hz Calibration: sampling the background noise...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n if (this.isCalibrating) return null;\n await this.recordBackground(\n stream, //stream\n () => this.numSuccessfulBackgroundCaptured < 1, //loop condition\n this.#awaitBackgroundNoiseRecording, //sleep to record\n this.sendBackgroundRecording, //send to get PSD\n this.mode,\n checkRec\n );\n this.incrementStatusBar();\n }\n this.mode = 'unfiltered';\n this.numSuccessfulCaptured = 0;\n\n if (this.isCalibrating) return null;\n await this.calibrationSteps(\n stream,\n this.#playCalibrationAudio, // play audio func (required)\n this.#createCalibrationNodeFromBuffer(this.#mlsBufferView), // before play func\n this.#awaitSignalOnset, // before record\n () => this.numSuccessfulCaptured < this.numCaptures, // loop while true\n this.#awaitDesiredMLSLength, // during record\n this.#afterMLSRecord, // after record\n this.mode,\n checkRec\n );\n this.stopCalibrationAudio();\n checkRec = false;\n\n // at this stage we've captured all the required signals,\n // and have received IRs for each one\n // so let's send all the IRs to the server to be converted to a single IIR\n if (this.isCalibrating) return null;\n await this.sendSystemImpulseResponsesToServerForProcessing();\n if (this.isCalibrating) return null;\n await this.sendComponentImpulseResponsesToServerForProcessing();\n\n this.numSuccessfulCaptured = 0;\n\n let iir_ir_and_plots;\n if (this._calibrateSoundCheck != 'none') {\n //do single check\n if (this._calibrateSoundCheck == 'goal' || this._calibrateSoundCheck == 'system') {\n if (this.isCalibrating) return null;\n iir_ir_and_plots = await this.singleSoundCheck(stream);\n if (this.isCalibrating) return null;\n } else {\n //both\n if (this.isCalibrating) return null;\n iir_ir_and_plots = await this.bothSoundCheck(stream);\n if (this.isCalibrating) return null;\n }\n } else {\n let unconv_rec = this.componentInvertedImpulseResponseNoBandpass;\n let conv_rec = this.componentInvertedImpulseResponse;\n if (this.isCalibrating) return null;\n let component_iir_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n unconv_rec = this.systemInvertedImpulseResponseNoBandpass;\n conv_rec = this.systemInvertedImpulseResponse;\n if (this.isCalibrating) return null;\n let system_iir_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n let gainValue = this.getGainDBSPL();\n iir_ir_and_plots = {\n unfiltered_recording: return_unconv_rec,\n filtered_recording: return_conv_rec,\n system: {\n iir: this.systemInvertedImpulseResponse,\n iir_no_bandpass: this.systemInvertedImpulseResponseNoBandpass,\n ir: this.systemIR,\n iir_psd: {\n y: system_iir_psd['y_conv'],\n x: system_iir_psd['y_conv'],\n y_no_bandpass: system_iir_psd['y_unconv'],\n x_no_bandpass: system_iir_psd['x_unconv'],\n },\n filtered_recording: [],\n convolution: this.systemConvolution,\n convolutionNoBandpass: this.systemConvolutionNoBandpass,\n psd: {\n unconv: {\n x: [],\n y: [],\n },\n conv: {\n x: [],\n y: [],\n },\n },\n },\n component: {\n iir: this.componentInvertedImpulseResponse,\n iir_no_bandpass: this.componentInvertedImpulseResponseNoBandpass,\n ir: this.componentIR,\n ir_in_time_domain: this.componentIRInTimeDomain,\n iir_psd: {\n y: component_iir_psd['y_conv'],\n x: component_iir_psd['x_conv'],\n y_no_bandpass: component_iir_psd['y_unconv'],\n x_no_bandpass: component_iir_psd['x_unconv'],\n },\n convolution: this.componentConvolution,\n convolutionNoBandpass: this.componentConvolutionNoBandpass,\n psd: {\n unconv: {\n x: [],\n y: [],\n },\n conv: {\n x: [],\n y: [],\n },\n },\n gainDBSPL: gainValue,\n },\n mls: this.#mlsBufferView,\n autocorrelations: this.autocorrelations,\n impulseResponses: [],\n };\n if (this.isCalibrating) return null;\n await Promise.all(this.impulseResponses).then(res => {\n for (let i = 0; i < res.length; i++) {\n if (res[i] != undefined) {\n iir_ir_and_plots['impulseResponses'].push(res[i]);\n }\n }\n });\n\n if (this.#download) {\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.#mls, 'MLS.csv');\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.componentConvolution, 'python_component_convolution_mls_iir.csv');\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.systemConvolution, 'python_system_convolution_mls_iir.csv');\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.componentInvertedImpulseResponse, 'componentIIR.csv');\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.systemInvertedImpulseResponse, 'systemIIR.csv');\n for (let i = 0; i < this.autocorrelations.length; i++) {\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.autocorrelations[i], `autocorrelation_${i}`);\n }\n const computedIRagain = await Promise.all(this.impulseResponses).then(res => {\n for (let i = 0; i < res.length; i++) {\n if (res[i] != undefined) {\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(res[i], `IR_${i}`);\n }\n }\n });\n }\n }\n if (this.isCalibrating) return null;\n this.percent_complete = 100;\n this.status = `All Hz Calibration: Finished`.toString() + this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n\n //here after calibration we have the component calibration (either loudspeaker or microphone) in the same form as the componentIR\n //that was used to calibrate\n // saveToJSON(iir_ir_and_plots);\n return iir_ir_and_plots;\n };\n\n //////////////////////volume\n\n handleIncomingData = data => {\n console.log('Received data: ', data);\n if (data.type === 'soundGainDBSPL') {\n this.soundGainDBSPL = data.value;\n } else {\n throw new Error(`Unknown data type: ${data.type}`);\n }\n };\n createSCurveBuffer = (onSetBool = true) => {\n const curve = new Float32Array(this.TAPER_SECS * this.sourceSamplingRate + 1);\n const frequency = 1 / (4 * this.TAPER_SECS);\n let j = 0;\n for (let i = 0; i < this.TAPER_SECS * this.sourceSamplingRate + 1; i += 1) {\n const phase = 2 * Math.PI * frequency * j;\n const onsetTaper = Math.pow(Math.sin(phase), 2);\n const offsetTaper = Math.pow(Math.cos(phase), 2);\n curve[i] = onSetBool ? onsetTaper : offsetTaper;\n j += 1 / this.sourceSamplingRate;\n }\n return curve;\n };\n\n #getTruncatedSignal = (left = 3.5, right = 4.5) => {\n const start = Math.floor(left * this.sourceSamplingRate);\n const end = Math.floor(right * this.sourceSamplingRate);\n const result = Array.from(this.getLastVolumeRecordedSignal().slice(start, end));\n console.log(\n 'Obtaining last 1000 hz recording from #allVolumeRecordings to send for processing'\n );\n /**\n * function to check that capture was properly made\n * @param {*} list\n */\n const checkResult = list => {\n const setItem = new Set(list);\n if (setItem.size === 1 && setItem.has(0)) {\n console.warn(\n 'The last capture failed, all recorded signal is zero',\n this.getAllVolumeRecordedSignals()\n );\n this.stopCalibrationAudio();\n this.isCalibrating = true;\n // restartButton.style.display = 'none';\n this.emit('update', {message: 'Connection failed, hit restart button to reconnect'});\n }\n if (setItem.size === 0) {\n console.warn('The last capture failed, no recorded signal');\n this.stopCalibrationAudio();\n this.isCalibrating = true;\n // restartButton.style.display = 'none';\n this.emit('update', {message: 'Connection failed, hit restart button to reconnect'});\n }\n };\n checkResult(result);\n return result;\n };\n\n /** \n * \n * \n Construct a calibration Node with the calibration parameters and given gain value\n * @param {*} gainValue\n * */\n #createCalibrationToneWithGainValue = gainValue => {\n const audioContext = this.makeNewSourceAudioContext();\n const oscilator = audioContext.createOscillator();\n const gainNode = audioContext.createGain();\n const taperGainNode = audioContext.createGain();\n const offsetGainNode = audioContext.createGain();\n const totalDuration = this.CALIBRATION_TONE_DURATION * 1.2;\n\n oscilator.frequency.value = this.#CALIBRATION_TONE_FREQUENCY;\n oscilator.type = this.#CALIBRATION_TONE_TYPE;\n gainNode.gain.value = gainValue;\n\n oscilator.connect(gainNode);\n gainNode.connect(taperGainNode);\n const onsetCurve = this.createSCurveBuffer();\n taperGainNode.gain.setValueCurveAtTime(onsetCurve, 0, this.TAPER_SECS);\n taperGainNode.connect(offsetGainNode);\n const offsetCurve = this.createSCurveBuffer(false);\n offsetGainNode.gain.setValueCurveAtTime(\n offsetCurve,\n totalDuration - this.TAPER_SECS,\n this.TAPER_SECS\n );\n offsetGainNode.connect(audioContext.destination);\n\n this.addCalibrationNode(oscilator);\n };\n\n /**\n * Construct a Calibration Node with the calibration parameters.\n *\n * @private\n * @example\n */\n #createCalibrationNode = () => {\n const audioContext = this.makeNewSourceAudioContext();\n const oscilator = audioContext.createOscillator();\n const gainNode = audioContext.createGain();\n\n oscilator.frequency.value = this.#CALIBRATION_TONE_FREQUENCY;\n oscilator.type = this.#CALIBRATION_TONE_TYPE;\n gainNode.gain.value = 0.04;\n\n oscilator.connect(gainNode);\n gainNode.connect(audioContext.destination);\n\n this.addCalibrationNode(oscilator);\n };\n\n #playCalibrationAudioVolume = async () => {\n const totalDuration = this.CALIBRATION_TONE_DURATION * 1.2;\n\n this.calibrationNodes[0].start(0);\n this.calibrationNodes[0].stop(totalDuration);\n console.log(`Playing a buffer of ${this.CALIBRATION_TONE_DURATION} seconds of audio`);\n console.log(`Waiting a total of ${totalDuration} seconds`);\n await (0,_utils__WEBPACK_IMPORTED_MODULE_1__.sleep)(totalDuration);\n };\n\n stopCalibrationAudioVolume = () => {\n if (this.calibrationNodes.length > 0) {\n this.calibrationNodes[0].stop();\n }\n };\n\n #sendToServerForProcessing = async (lCalib) => {\n console.log('Sending data to server');\n this.addTimeStamp('Send volume data to server');\n let left = this.calibrateSound1000HzPreSec;\n let right = this.calibrateSound1000HzPreSec + this.calibrateSound1000HzSec;\n if (this.isCalibrating) return null;\n this.pyServerAPI\n .getVolumeCalibration({\n sampleRate: this.sourceSamplingRate,\n payload: this.#getTruncatedSignal(left, right),\n lCalib: lCalib,\n })\n .then(res => {\n if (this.outDBSPL === null) {\n this.incrementStatusBar();\n this.outDBSPL = res['outDbSPL'];\n this.outDBSPL1000 = res['outDbSPL1000'];\n this.THD = res['thd'];\n }\n })\n .catch(err => {\n console.warn(err);\n });\n\n await this.pyServerAPI\n .volumePowerCheck({\n payload: this.getLastVolumeRecordedSignal(),\n sampleRate: this.sourceSamplingRate || 96000,\n binDesiredSec: this._calibrateSoundPowerBinDesiredSec,\n preSec: this.calibrateSound1000HzPreSec,\n Sec: this.calibrateSound1000HzSec,\n })\n .then(res => {\n if (res && res['sd'] < this._calibrateSoundPowerDbSDToleratedDb) {\n this.recordingChecks['volume'][this.inDB] = res;\n }\n });\n };\n\n startCalibrationVolume = async (stream, gainValues, lCalib, componentGainDBSPL) => {\n if (this.isCalibrating) return null;\n const trialIterations = gainValues.length;\n this.status_denominator += trialIterations;\n const thdValues = [];\n const inDBValues = [];\n let inDB = 0;\n const outDBSPLValues = [];\n const outDBSPL1000Values = [];\n let checkRec = \"loudest\";\n\n // do one calibration that will be discarded\n const soundLevelToDiscard = -60;\n const gainToDiscard = Math.pow(10, soundLevelToDiscard / 20);\n this.inDB = soundLevelToDiscard;\n this.status =\n `1000 Hz Calibration: Sound Level ${soundLevelToDiscard} dB`.toString() +\n this.generateTemplate().toString();\n //this.emit('update', {message: `1000 Hz Calibration: Sound Level ${soundLevelToDiscard} dB`});\n this.emit('update', {message: this.status});\n this.startTime = new Date().getTime();\n\n do {\n console.log('while loop');\n if (this.isCalibrating) {\n console.log('restart calibration');\n return null;\n }\n // eslint-disable-next-line no-await-in-loop\n await this.volumeCalibrationSteps(\n stream,\n this.#playCalibrationAudioVolume,\n this.#createCalibrationToneWithGainValue,\n this.#sendToServerForProcessing,\n gainToDiscard,\n lCalib, //todo make this a class parameter\n checkRec\n );\n } while (this.outDBSPL === null);\n //reset the values\n //this.incrementStatusBar();\n\n this.outDBSPL = null;\n this.outDBSPL = null;\n this.outDBSPL1000 = null;\n this.THD = null;\n\n // run the calibration at different gain values provided by the user\n for (let i = 0; i < trialIterations; i++) {\n //convert gain to DB and add to inDB\n if (i == trialIterations - 1) {\n checkRec = 'loudest';\n }\n inDB = Math.log10(gainValues[i]) * 20;\n // precision to 1 decimal place\n inDB = Math.round(inDB * 10) / 10;\n this.inDB = inDB;\n inDBValues.push(inDB);\n console.log('next update');\n this.status =\n `1000 Hz Calibration: Sound Level ${inDB} dB`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n do {\n if (this.isCalibrating) {\n console.log('restart calibration');\n return null;\n }\n // eslint-disable-next-line no-await-in-loop\n await this.volumeCalibrationSteps(\n stream,\n this.#playCalibrationAudioVolume,\n this.#createCalibrationToneWithGainValue,\n this.#sendToServerForProcessing,\n gainValues[i],\n lCalib, //todo make this a class parameter\n checkRec\n );\n } while (this.outDBSPL === null);\n outDBSPL1000Values.push(this.outDBSPL1000);\n thdValues.push(this.THD);\n outDBSPLValues.push(this.outDBSPL);\n\n this.outDBSPL = null;\n this.outDBSPL1000 = null;\n this.THD = null;\n }\n if (this.isCalibrating) return null;\n // get the volume calibration parameters from the server\n this.addTimeStamp('Get Volume Calibration Parameters');\n\n const parameters = await this.pyServerAPI\n .getVolumeCalibrationParameters({\n inDBValues: inDBValues,\n outDBSPLValues: outDBSPL1000Values,\n lCalib: lCalib,\n componentGainDBSPL,\n })\n .then(res => {\n this.incrementStatusBar();\n return res;\n });\n if (this.isCalibrating) return null;\n const result = {\n parameters: parameters,\n inDBValues: inDBValues,\n outDBSPLValues: outDBSPLValues,\n outDBSPL1000Values: outDBSPL1000Values,\n thdValues: thdValues,\n };\n\n return result;\n };\n\n writeFrqGainToFirestore = async (speakerID, frq, gain, OEM, documentID) => {\n // freq and gain are too large to take samples 1 in every 100 samples\n // const sampledFrq = [];\n // const sampledGain = [];\n // for (let i = 0; i < frq.length; i += 100) {\n // sampledFrq.push(frq[i]);\n // sampledGain.push(gain[i]);\n // }\n\n const data = {Freq: frq, Gain: gain};\n\n const docRef = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.doc)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], 'Microphones', documentID);\n await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.updateDoc)(docRef, {\n linear: data,\n });\n\n // divide frq and gain into smaller chunks and write to firestore one chunk at a time\n // use arrayUnion to append to the array\n // const chunkSize = 600;\n // const chunkedFrq = [];\n // const chunkedGain = [];\n // for (let i = 0; i < frq.length; i += chunkSize) {\n // chunkedFrq.push(frq.slice(i, i + chunkSize));\n // chunkedGain.push(gain.slice(i, i + chunkSize));\n // }\n // const docRef = doc(database, 'Microphones', documentID);\n // for (let i = 0; i < chunkedFrq.length; i++) {\n // await updateDoc(docRef, {\n // linear: {\n // Freq: arrayUnion(...chunkedFrq[i]),\n // Gain: arrayUnion(...chunkedGain[i]),\n // },\n // });\n // }\n };\n // function to write frq and gain to firebase database given speakerID\n writeFrqGain = async (speakerID, frq, gain, OEM) => {\n // freq and gain are too large to take samples 1 in every 100 samples\n\n const sampledFrq = [];\n const sampledGain = [];\n for (let i = 0; i < frq.length; i += 100) {\n sampledFrq.push(frq[i]);\n sampledGain.push(gain[i]);\n }\n\n const data = {Freq: sampledFrq, Gain: sampledGain};\n\n await (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.set)((0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.ref)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], `Microphone2/${OEM}/${speakerID}/linear`), data);\n };\n\n // Function to Read frq and gain from firebase database given speakerID\n // returns an array of frq and gain if speakerID exists, returns null otherwise\n readFrqGainFromFirestore = async (speakerID, OEM, isDefault) => {\n const collectionRef = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.collection)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], 'Microphones');\n const q = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.query)(\n collectionRef,\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('ID', '==', speakerID),\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('lowercaseOEM', '==', OEM),\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('isDefault', '==', isDefault)\n );\n const querySnapshot = await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.getDocs)(q);\n // if exists return the linear field of the first document\n if (querySnapshot.size > 0) {\n return querySnapshot.docs[0].data().linear;\n }\n return null;\n };\n readFrqGain = async (speakerID, OEM) => {\n const dbRef = (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.ref)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"]);\n const snapshot = await (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.get)((0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.child)(dbRef, `Microphone2/${OEM}/${speakerID}/linear`));\n if (snapshot.exists()) {\n return snapshot.val();\n }\n return null;\n };\n readGainat1000HzFromFirestore = async (speakerID, OEM, isDefault) => {\n const collectionRef = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.collection)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], 'Microphones');\n const q = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.query)(\n collectionRef,\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('ID', '==', speakerID),\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('lowercaseOEM', '==', OEM),\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('isDefault', '==', isDefault)\n );\n const querySnapshot = await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.getDocs)(q);\n // if exists return the Gain1000 field of the first document\n if (querySnapshot.size > 0) {\n return querySnapshot.docs[0].data().Gain1000;\n }\n return null;\n };\n\n readGainat1000Hz = async (speakerID, OEM) => {\n const dbRef = (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.ref)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"]);\n const snapshot = await (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.get)((0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.child)(dbRef, `Microphone2/${OEM}/${speakerID}/Gain1000`));\n if (snapshot.exists()) {\n return snapshot.val();\n }\n return null;\n };\n\n writeGainat1000HzToFirestore = async (speakerID, gain, OEM, documentID) => {\n const docRef = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.doc)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], 'Microphones', documentID);\n await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.updateDoc)(docRef, {\n Gain1000: gain,\n });\n };\n\n writeGainat1000Hz = async (speakerID, gain, OEM) => {\n await (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.set)((0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.ref)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], `Microphone2/${OEM}/${speakerID}/Gain1000`), gain);\n };\n\n writeIsSmartPhoneToFirestore = async (speakerID, isSmartPhone, OEM) => {\n const collectionRef = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.collection)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], 'Microphones');\n const q = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.query)(\n collectionRef,\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('ID', '==', speakerID),\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('lowercaseOEM', '==', OEM),\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('isDefault', '==', true)\n );\n const querySnapshot = await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.getDocs)(q);\n if (querySnapshot.size > 0) {\n const docRef = await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.addDoc)(collectionRef, {isSmartPhone: isSmartPhone, isDefault: false});\n return docRef.id;\n } else {\n const docRef = await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.addDoc)(collectionRef, {isSmartPhone: isSmartPhone, isDefault: true});\n return docRef.id;\n }\n };\n\n writeIsSmartPhone = async (speakerID, isSmartPhone, OEM) => {\n const data = {isSmartPhone: isSmartPhone};\n await (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.set)((0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.ref)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], `Microphone2/${OEM}/${speakerID}/isSmartPhone`), isSmartPhone);\n };\n\n writeMicrophoneInfoToFirestore = async (speakerID, micInfo, OEM, documentID) => {\n const docRef = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.doc)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], 'Microphones', documentID);\n await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.setDoc)(docRef, micInfo, {merge: true});\n };\n\n doesMicrophoneExistInFirestore = async (speakerID, OEM, documentID) => {\n const docRef = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.doc)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], 'Microphone', OEM, speakerID, documentID);\n const docSnap = await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.getDoc)(docRef);\n if (docSnap.exists()) {\n return true;\n }\n return false;\n };\n\n doesMicrophoneExist = async (speakerID, OEM) => {\n const dbRef = (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.ref)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"]);\n const snapshot = await (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.get)((0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.child)(dbRef, `Microphone2/${OEM}/${speakerID}`));\n if (snapshot.exists()) {\n return true;\n }\n return false;\n };\n\n addMicrophoneInfo = async (speakerID, OEM, micInfo) => {\n // add to database if /info does not exist\n const dbRef = (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.ref)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"]);\n const snapshot = await (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.get)((0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.child)(dbRef, `Microphone2/${OEM}/${speakerID}/info`));\n if (!snapshot.exists()) {\n await (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.set)((0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.ref)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], `Microphone2/${OEM}/${speakerID}/info`), micInfo);\n }\n };\n\n convertToDB = gain => {\n return Math.log10(gain) * 20;\n };\n\n // Function to perform linear interpolation between two points\n interpolate(x, x0, y0, x1, y1) {\n return y0 + ((x - x0) * (y1 - y0)) / (x1 - x0);\n }\n\n findGainatFrequency = (frequencies, gains, targetFrequency) => {\n // Find the index of the first frequency in the array greater than the target frequency\n let index = 0;\n while (index < frequencies.length && frequencies[index] < targetFrequency) {\n index++;\n }\n\n // Handle cases when the target frequency is outside the range of the given data\n if (index === 0) {\n return gains[0];\n } else if (index === frequencies.length) {\n return gains[gains.length - 1];\n } else {\n // Interpolate the gain based on the surrounding frequencies\n const x0 = frequencies[index - 1];\n const y0 = gains[index - 1];\n const x1 = frequencies[index];\n const y1 = gains[index];\n return this.interpolate(targetFrequency, x0, y0, x1, y1);\n }\n };\n\n\n checkPowerVariation = async () => {\n let recordings = this.getAllFilteredRecordedSignals();\n // remove filteredMLSAttenuation from the recordings\n\n // recordings = recordings.map(recording => {\n // if (this.soundCheck == 'component') {\n // return recording.map(value => value / this.filteredMLSAttenuation.component);\n // }\n // return recording.map(value => value / this.filteredMLSAttenuation.system);\n // });\n\n const rec = recordings[recordings.length - 1];\n\n await this.pyServerAPI\n .allHzPowerCheck({\n payload: rec,\n sampleRate: this.sourceSamplingRate || 96000,\n binDesiredSec: this._calibrateSoundPowerBinDesiredSec,\n burstSec: this.desired_time_per_mls,\n repeats: this.numMLSPerCapture,\n })\n .then(result => {\n if (result) {\n if (result['sd'] > this._calibrateSoundPowerDbSDToleratedDb) {\n console.log('filtered recording sd too high');\n } else {\n this.recordingChecks[this.soundCheck].push(result);\n if (this.numSuccessfulCaptured < 1) {\n this.numSuccessfulCaptured += 1;\n this.stepNum += 1;\n this.incrementStatusBar();\n console.log(\n 'after mls w iir record for some reason add numSucc capt ' + this.stepNum\n );\n this.status =\n `All Hz Calibration: ${this.numSuccessfulCaptured} recording of convolved MLS captured`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {\n message: this.status,\n });\n }\n }\n }\n });\n };\n\n getGainDBSPL = () => {\n var freqIndex = this.componentIR.Freq.indexOf(1000);\n\n // If freqIndex is not -1 (meaning 1000 is found in the freq array)\n if (freqIndex !== -1) {\n // Get the corresponding gain value using the index\n var gainValue = this.componentIR.Gain[freqIndex];\n return gainValue;\n } else {\n console.log('Freq 1000 not found in the array.');\n return null;\n }\n };\n // Example of how to use the writeFrqGain and readFrqGain functions\n // writeFrqGain('speaker1', [1, 2, 3], [4, 5, 6]);\n // Speaker1 is the speakerID you want to write to in the database\n // readFrqGain('MiniDSPUMIK_1').then(data => console.log(data));\n // MiniDSPUMIK_1 is the speakerID with some Data in the database\n //adding gainDBSPL\n startCalibration = async (\n stream,\n gainValues,\n lCalib = 104.92978421490648,\n componentIR = null,\n microphoneName = 'MiniDSP-UMIK1-711-4754-vertical',\n _calibrateSoundCheck = 'goal', //GOAL PASSed in by default\n isSmartPhone = false,\n _calibrateSoundBurstDb = -18,\n _calibrateSoundBurstFilteredExtraDb = 6,\n _calibrateSoundBurstLevelReTBool = false,\n _calibrateSoundBurstUses1000HzGainBool = false,\n _calibrateSoundBurstRepeats = 3,\n _calibrateSoundBurstSec = 1,\n _calibrateSoundBurstsWarmup = 1,\n _calibrateSoundHz = 48000,\n _calibrateSoundIIRSec = 0.2,\n _calibrateSoundIRSec = 0.2,\n calibrateSound1000HzPreSec = 3.5,\n calibrateSound1000HzSec = 1.0,\n calibrateSound1000HzPostSec = 0.5,\n _calibrateSoundBackgroundSecs = 0,\n _calibrateSoundSmoothOctaves = 0.33,\n _calibrateSoundPowerBinDesiredSec = 0.2,\n _calibrateSoundPowerDbSDToleratedDb = 1,\n micManufacturer = '',\n micSerialNumber = '',\n micModelNumber = '',\n micModelName = '',\n calibrateMicrophonesBool,\n authorEmails,\n webAudioDeviceNames = {\n loudspeaker: 'loudspeaker',\n microphone: 'microphone',\n microphoneText: 'xxx XXX',\n },\n userIDs,\n restartButton,\n reminder,\n calibrateSoundLimit,\n ) => {\n this.calibrateSoundLimit = calibrateSoundLimit;\n this._calibrateSoundBurstDb = _calibrateSoundBurstDb;\n this._calibrateSoundBurstFilteredExtraDb = _calibrateSoundBurstFilteredExtraDb;\n this._calibrateSoundBurstLevelReTBool = _calibrateSoundBurstLevelReTBool;\n this.CALIBRATION_TONE_DURATION =\n calibrateSound1000HzPreSec + calibrateSound1000HzSec + calibrateSound1000HzPostSec;\n this.calibrateSound1000HzPreSec = calibrateSound1000HzPreSec;\n this.calibrateSound1000HzSec = calibrateSound1000HzSec;\n this.calibrateSound1000HzPostSec = calibrateSound1000HzPostSec;\n this.iirLength = Math.floor(_calibrateSoundIIRSec * this.sourceSamplingRate);\n this.irLength = Math.floor(_calibrateSoundIRSec * this.sourceSamplingRate);\n this.numMLSPerCapture = _calibrateSoundBurstRepeats + 1;\n this.desired_time_per_mls = _calibrateSoundBurstSec;\n this.num_mls_to_skip = _calibrateSoundBurstsWarmup;\n this.desired_sampling_rate = _calibrateSoundHz;\n this._calibrateSoundBackgroundSecs = _calibrateSoundBackgroundSecs;\n this._calibrateSoundSmoothOctaves = _calibrateSoundSmoothOctaves;\n this._calibrateSoundPowerBinDesiredSec = _calibrateSoundPowerBinDesiredSec;\n this._calibrateSoundBurstUses1000HzGainBool = _calibrateSoundBurstUses1000HzGainBool;\n this._calibrateSoundPowerDbSDToleratedDb = _calibrateSoundPowerDbSDToleratedDb;\n this.webAudioDeviceNames = webAudioDeviceNames;\n if (isSmartPhone) this.webAudioDeviceNames.microphone = this.deviceInfo.microphoneFromAPI;\n this.webAudioDeviceNames.microphoneText = this.webAudioDeviceNames.microphoneText\n .replace('xxx', this.webAudioDeviceNames.microphone)\n .replace('XXX', this.webAudioDeviceNames.microphone);\n //feed calibration goal here\n this._calibrateSoundCheck = _calibrateSoundCheck;\n //check if a componentIR was given to the system, if it isn't check for the microphone. using dummy data here bc we need to\n //check the db based on the microphone currently connected\n\n //new lCalib found at top of calibration files *1000hz, make sure to correct\n //based on zeroing of 1000hz, search for \"*1000Hz\"\n const ID = isSmartPhone ? micModelNumber : micSerialNumber;\n const OEM = isSmartPhone\n ? micModelName === 'UMIK-1' || micModelName === 'UMIK-2'\n ? 'minidsp'\n : this.deviceInfo.OEM.toLowerCase().split(' ').join('')\n : micManufacturer.toLowerCase().split(' ').join('');\n // const ID = \"712-5669\";\n // const OEM = \"minidsp\";\n const micInfo = {\n micModelName: isSmartPhone ? micModelName : microphoneName,\n OEM: isSmartPhone\n ? micModelName === 'UMIK-1' || micModelName === 'UMIK-2'\n ? 'miniDSP'\n : this.deviceInfo.OEM\n : micManufacturer,\n ID: ID,\n createDate: new Date(),\n DateText: (0,_utils__WEBPACK_IMPORTED_MODULE_1__.getCurrentTimeString)(),\n HardwareName: isSmartPhone ? this.deviceInfo.hardwarename : microphoneName,\n hardwareFamily: isSmartPhone ? this.deviceInfo.hardwarefamily : microphoneName,\n HardwareModel: isSmartPhone ? this.deviceInfo.hardwaremodel : microphoneName,\n PlatformName: isSmartPhone ? this.deviceInfo.platformname : 'N/A',\n PlatformVersion: isSmartPhone ? this.deviceInfo.platformversion : 'N/A',\n DeviceType: isSmartPhone ? this.deviceInfo.devicetype : 'N/A',\n ID_from_51Degrees: isSmartPhone ? this.deviceInfo.DeviceId : 'N/A',\n calibrateMicrophonesBool: calibrateMicrophonesBool,\n screenHeight:this.deviceInfo.screenHeight,\n screenWidth:this.deviceInfo.screenWidth,\n webAudioDeviceNames: {\n loudspeaker: this.webAudioDeviceNames.loudspeaker,\n microphone: this.webAudioDeviceNames.microphone,\n },\n userIDs: userIDs,\n lowercaseOEM: OEM.toLowerCase().split(' ').join(''),\n };\n if (calibrateMicrophonesBool) {\n micInfo['authorEmails'] = authorEmails;\n }\n // if undefined in micInfo, set to empty string\n for (const [key, value] of Object.entries(micInfo)) {\n if (value === undefined) {\n micInfo[key] = '';\n }\n }\n\n // this.writeMicrophoneInfoToFirestore(ID, micInfo, OEM, 'default');\n // this.addMicrophoneInfo(ID, OEM, micInfo);\n if (componentIR == null) {\n //mode 'ir'\n //global variable this.componentIR must be set\n this.componentIR = await this.readFrqGainFromFirestore(ID, OEM, true).then(data => {\n return data;\n });\n // await this.readFrqGain(ID, OEM).then(data => {\n // return data;\n // });\n\n // lCalib = await this.readGainat1000Hz(ID, OEM);\n lCalib = await this.readGainat1000HzFromFirestore(ID, OEM, true);\n micInfo['gainDBSPL'] = lCalib;\n // this.componentGainDBSPL = this.convertToDB(lCalib);\n this.componentGainDBSPL = lCalib;\n //TODO: if this call to database is unknown, cannot perform experiment => return false\n if (this.componentIR == null) {\n this.status =\n `Microphone (${OEM},${ID}) is not found in the database. Please add it to the database.`.toString();\n this.emit('update', {message: this.status});\n return false;\n }\n } else {\n this.transducerType = 'Microphone';\n this.componentIR = componentIR;\n lCalib = this.findGainatFrequency(this.componentIR.Freq, this.componentIR.Gain, 1000);\n // this.componentGainDBSPL = this.convertToDB(lCalib);\n this.componentGainDBSPL = lCalib;\n // await this.writeIsSmartPhone(ID, isSmartPhone, OEM);\n }\n\n this.oldComponentIR = JSON.parse(JSON.stringify(this.componentIR));\n\n return await new Promise(async (resolve, reject) => {\n // add event listner to params.restartButton to resolve the promise with a string: 'restart'\n\n if (reminder) {\n reminder.style.display = '';\n }\n if (restartButton) {\n restartButton.style.display = '';\n restartButton.addEventListener('click', () => {\n this.stopCalibrationAudio();\n this.isCalibrating = true;\n restartButton.style.display = 'none';\n if (reminder) {\n reminder.style.display = 'none';\n }\n this.emit('update', {message: 'Restarting calibration...'});\n resolve('restart');\n });\n }\n let volumeResults = await this.startCalibrationVolume(\n stream,\n gainValues,\n lCalib,\n this.componentGainDBSPL\n );\n if (!volumeResults) return;\n\n this.T = volumeResults[\"parameters\"][\"T\"];\n this.gainDBSPL = volumeResults[\"parameters\"][\"gainDBSPL\"];\n\n // console.log(\"VOLUME RESULTS\");\n // console.log(volumeResults);\n let impulseResponseResults = await this.startCalibrationImpulseResponse(stream);\n if (!impulseResponseResults) return;\n impulseResponseResults['background_noise'] = this.background_noise;\n impulseResponseResults['system']['background_noise'] = this.background_noise;\n impulseResponseResults['component']['background_noise'] = this.background_noise;\n\n //attenuate system background noise\n if (this.systemAttenuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.systemAttenuatorGainDB/20);\n let linearScalePowerAttenuation = Math.pow(10,this.systemAttenuatorGainDB/10);\n impulseResponseResults['system']['background_noise']['recording'] = impulseResponseResults['background_noise']['recording'].map(value => value/linearScaleAttenuation);\n impulseResponseResults['system']['background_noise']['x_background'] = impulseResponseResults['background_noise']['x_background'];\n impulseResponseResults['system']['background_noise']['y_background'] = impulseResponseResults['background_noise']['y_background'].map(value => value/linearScalePowerAttenuation);\n }\n //attenuate component background noise\n if (this.componentAttentuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.componentAttenuatorGainDB/20);\n let linearScalePowerAttenuation = Math.pow(10,this.componentAttenuatorGainDB/10);\n impulseResponseResults['component']['background_noise']['recording'] = impulseResponseResults['background_noise']['recording'].map(value => value/linearScaleAttenuation);\n impulseResponseResults['component']['background_noise']['x_background'] = impulseResponseResults['background_noise']['x_background'];\n impulseResponseResults['component']['background_noise']['y_background'] = impulseResponseResults['background_noise']['y_background'].map(value => value/linearScalePowerAttenuation);\n }\n impulseResponseResults['system']['attenuatorGainDB'] = this.systemAttenuatorGainDB;\n impulseResponseResults['component']['attenuatorGainDB'] = this.componentAttenuatorGainDB;\n impulseResponseResults['system']['fMaxHz'] = this.systemFMaxHz;\n impulseResponseResults['component']['fMaxHz'] = this.componentFMaxHz;\n impulseResponseResults['L_new_n'] = this.L_new_n;\n impulseResponseResults['fs2'] = this.fs2;\n\n if (componentIR != null) {\n // I corrected microphone/loudpeaker IR scale in easyeyes,\n // but since we write microphone IR to firestore here\n // we need to correct microphone IR here\n let correctGain =\n Math.round((volumeResults.parameters.gainDBSPL - this.componentGainDBSPL) * 10) / 10;\n\n let IrFreq = impulseResponseResults?.component.ir.Freq.map(freq => Math.round(freq));\n let IrGain = impulseResponseResults?.component?.ir.Gain;\n const IrGainAt1000Hz = IrGain[IrFreq.findIndex(freq => freq === 1000)];\n const difference = Math.round(10 * (IrGainAt1000Hz - correctGain)) / 10;\n IrGain = IrGain.map(gain => gain - difference);\n micInfo['mls'] = Number(this.SDofFilteredRange['mls']);\n micInfo['systemCorrectionSD'] = Number(this.SDofFilteredRange['system']);\n micInfo['componentCorrectionSD'] = Number(this.SDofFilteredRange['component']);\n console.log(typeof micInfo['componentCorrectionSD']);\n // const id = await this.writeIsSmartPhoneToFirestore(ID, isSmartPhone, OEM);\n // await this.writeMicrophoneInfoToFirestore(ID, micInfo, OEM, id);\n // await this.writeFrqGainToFirestore(ID, IrFreq, IrGain, OEM, id);\n // micInfo['gainDBSPL'] = impulseResponseResults.component.gainDBSPL;\n // await this.writeGainat1000HzToFirestore(ID, micInfo['gainDBSPL'], OEM, id);\n // await this.writeGainat1000Hz(ID, micInfo['gainDBSPL'], OEM);\n }\n const total_results = {...volumeResults, ...impulseResponseResults};\n total_results['filteredMLSRange'] = this.filteredMLSRange;\n total_results['filteredMLSAttenuation'] = this.filteredMLSAttenuation;\n total_results['micInfo'] = micInfo;\n total_results['audioInfo'] = {};\n total_results['audioInfo']['sinkSampleRate'] = this.sinkSamplingRate;\n total_results['audioInfo']['sourceSampleRate'] = this.sourceSamplingRate;\n total_results['audioInfo']['bitsPerSample'] = this.sampleSize;\n const timeStampresult = [...this.timeStamp].join('\\n');\n total_results['timeStamps'] = timeStampresult;\n total_results['recordingChecks'] = this.recordingChecks;\n total_results['component']['phase'] = this.componentIRPhase;\n total_results['system']['phase'] = this.systemIRPhase;\n total_results['qualityMetrics'] = this.SDofFilteredRange;\n total_results['flags'] = this.flags;\n console.log('total results');\n console.log(total_results);\n console.log('Time Stamps');\n console.log(timeStampresult);\n\n resolve(total_results);\n });\n };\n}\n\n/* harmony default export */ __webpack_exports__[\"default\"] = (Combination);\n\n\n//# sourceURL=webpack://speakerCalibrator/./src/tasks/combination/combination.js?");
854
+ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _audioCalibrator__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../audioCalibrator */ \"./src/tasks/audioCalibrator.js\");\n/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../utils */ \"./src/utils.js\");\n/* harmony import */ var _config_firebase__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../../config/firebase */ \"./src/config/firebase.js\");\n/* harmony import */ var firebase_database__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! firebase/database */ \"./node_modules/firebase/database/dist/esm/index.esm.js\");\n/* harmony import */ var firebase_firestore__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! firebase/firestore */ \"./node_modules/firebase/firestore/dist/esm/index.esm.js\");\n\n\n// import * as tf from '../../../node_modules/@tensorflow/tfjs';\n\n\n\n\n\n\n/**\n *\n */\nclass Combination extends _audioCalibrator__WEBPACK_IMPORTED_MODULE_0__[\"default\"] {\n /**\n * Default constructor. Creates an instance with any number of paramters passed or the default parameters defined here.\n *\n * @param {Object<boolean, number, number, number>} calibratorParams - paramter object\n * @param {boolean} [calibratorParams.download = false] - boolean flag to download captures\n * @param {number} [calibratorParams.mlsOrder = 18] - order of the MLS to be generated\n * @param {number} [calibratorParams.numCaptures = 5] - number of captures to perform\n * @param {number} [calibratorParams.numMLSPerCapture = 2] - number of bursts of MLS per capture\n */\n constructor({\n download = false,\n mlsOrder = 18,\n numCaptures = 3,\n numMLSPerCapture = 2,\n lowHz = 20,\n highHz = 10000,\n }) {\n super(numCaptures, numMLSPerCapture);\n this.#mlsOrder = parseInt(mlsOrder, 10);\n this.#P = 2 ** mlsOrder - 1;\n this.#download = download;\n this.#mls = [];\n this.#lowHz = lowHz;\n this.#highHz = highHz;\n }\n\n /** @private */\n stepNum = 0;\n\n /** @private */\n totalSteps = 25;\n\n /** @private */\n #download;\n\n /** @private */\n #mlsGenInterface;\n\n /** @private */\n #mlsBufferView;\n\n /** @private */\n componentInvertedImpulseResponse = null;\n\n /** @private */\n systemInvertedImpulseResponse = null;\n\n //averaged and subtracted ir returned from calibration used to calculated iir\n /** @private */\n ir = null;\n\n /** @private */\n impulseResponses = [];\n\n /** @private */\n #mlsOrder;\n\n /** @private */\n #lowHz;\n\n /** @private */\n #highHz;\n\n /** @private */\n #mls;\n\n /** @private */\n #P;\n\n /** @private */\n #audioContext;\n\n /** @private */\n TAPER_SECS = 5;\n\n /** @private */\n offsetGainNode;\n\n /** @private */\n componentConvolution;\n\n /** @private */\n componentConvolutionNoBandpass;\n\n /** @private */\n componentIROrigin = {\n Freq: [],\n Gain: [],\n };\n\n /** @private */\n systemConvolution;\n\n /** @private */\n systemConvolutionNoBandpass;\n\n ////////////////////////volume\n /** @private */\n #CALIBRATION_TONE_FREQUENCY = 1000; // Hz\n\n /** @private */\n #CALIBRATION_TONE_TYPE = 'sine';\n\n CALIBRATION_TONE_DURATION = 5; // seconds\n calibrateSound1000HzPreSec = 3.5;\n calibrateSound1000HzSec = 1.0;\n calibrateSound1000HzPostSec = 0.5;\n\n /** @private */\n outDBSPL = null;\n THD = null;\n outDBSPL1000 = null;\n\n /** @private */\n TAPER_SECS = 0.01; // seconds\n\n /** @private */\n status_denominator = 8;\n\n /** @private */\n status_numerator = 0;\n\n /** @private */\n percent_complete = 0;\n\n /** @private */\n status = ``;\n\n /**@private */\n status_literal = `<div style=\"display: flex; justify-content: center;\"><div style=\"width: 200px; height: 20px; border: 2px solid #000; border-radius: 10px;\"><div style=\"width: ${this.percent_complete}%; height: 100%; background-color: #00aaff; border-radius: 8px;\"></div></div></div>`;\n\n /**@private */\n componentIR = null;\n\n /**@private */\n oldComponentIR = null;\n\n /**@private */\n systemIR = null;\n\n /**@private */\n _calibrateSoundCheck = '';\n\n deviceType = null;\n\n deviceName = null;\n\n deviceInfo = null;\n\n desired_time_per_mls = 0;\n\n num_mls_to_skip = 0;\n\n desired_sampling_rate = 0;\n\n #currentConvolution = [];\n\n mode = 'unfiltered';\n\n sourceNode;\n\n autocorrelations = [];\n\n iirLength = 0;\n\n irLength = 0;\n\n componentInvertedImpulseResponseNoBandpass = [];\n\n componentIRInTimeDomain = [];\n\n systemInvertedImpulseResponseNoBandpass = [];\n\n _calibrateSoundBackgroundSecs;\n\n _calibrateSoundSmoothOctaves;\n\n background_noise = {};\n\n numSuccessfulBackgroundCaptured;\n\n _calibrateSoundBurstDb;\n\n _calibrateSoundBurstFilteredExtraDb;\n\n _calibrateSoundBurstLevelReTBool;\n\n SDofFilteredRange = {\n mls: undefined,\n component: undefined,\n system: undefined,\n };\n\n transducerType = 'Loudspeaker';\n\n componentIRPhase = [];\n\n systemIRPhase = [];\n\n webAudioDeviceNames = {loudspeaker: '', microphone: '', loudspeakerText: '', microphoneText: ''};\n\n recordingChecks = {\n volume: {},\n unfiltered: [],\n system: [],\n component: [],\n };\n\n inDB;\n\n soundCheck = '';\n\n filteredMLSRange = {\n component: {\n Min: null,\n Max: null,\n },\n system: {\n Min: null,\n Max: null,\n },\n };\n\n /** @private */\n timeStamp = [];\n\n \n\n restartCalibration = false;\n\n calibrateSoundLimit = 1;\n\n filteredMLSAttenuation = {\n component: 1,\n system: 1,\n maxAbsSystem: 1,\n maxAbsComponent: 1,\n };\n\n //parameter result from volume calibration\n T = 0;\n //gainDBSPL result from volume calibration\n gainDBSPL = 0;\n //not always just using _calibrateSoundBurstDb for MLS so created a new parameter\n power_dB = 0;\n\n //system\n systemAttenuatorGainDB = 0;\n systemFMaxHz = 0;\n\n //component\n componentAttentuatorGainDB = 0;\n componentFMaxHz = 0;\n\n dL_n;\n L_new_n;\n fs2;\n\n /**generate string template that gets reevaluated as variable increases */\n generateTemplate = () => {\n if (this.isCalibrating) {\n return '';\n }\n if (this.percent_complete > 100) {\n this.percent_complete = 100;\n }\n let MLSsd = '';\n let componentSD = '';\n let systemSD = '';\n let flags = '';\n const reportWebAudioNames = `<br>${this.webAudioDeviceNames.loudspeakerText} <br> ${this.webAudioDeviceNames.microphoneText}`;\n const reportParameters = `<br> Sampling: Loudspeaker ${this.sourceSamplingRate} Hz, Microphone ${this.sinkSamplingRate} Hz, ${this.sampleSize} bits`;\n if (this.flags) {\n flags = `<br> autoGainControl: ${this.flags.autoGainControl} \n <br>echoCancellation: ${this.flags.echoCancellation}\n <br>noiseSuppression: ${this.flags.noiseSuppression}`\n }\n if (this.SDofFilteredRange['mls']) {\n MLSsd = `<br> Recorded MLS power SD: ${this.SDofFilteredRange['mls']} dB`;\n }\n if (this.SDofFilteredRange['system']) {\n systemSD = `<br> Loudspeaker+Microphone correction SD: ${this.SDofFilteredRange['system']} dB`;\n }\n if (this.SDofFilteredRange['component']) {\n componentSD = `<br> ${this.transducerType} correction SD: ${this.SDofFilteredRange['component']} dB`;\n }\n const template = `<div style=\"display: flex; justify-content: center; margin-top:12px;\"><div style=\"width: 800px; height: 20px; border: 2px solid #000; border-radius: 10px;\"><div style=\"width: ${this.percent_complete}%; height: 100%; background-color: #00aaff; border-radius: 8px;\"></div></div></div>`;\n return reportWebAudioNames + reportParameters + flags + MLSsd + systemSD + componentSD + template;\n };\n\n /** increment numerator and percent for status bar */\n incrementStatusBar = () => {\n this.status_numerator += 1;\n this.percent_complete = (this.status_numerator / this.status_denominator) * 100;\n };\n\n setDeviceType = deviceType => {\n this.deviceType = deviceType;\n };\n\n setDeviceName = deviceName => {\n this.deviceName = deviceName;\n };\n\n setDeviceInfo = deviceInfo => {\n this.deviceInfo = deviceInfo;\n };\n\n /** .\n * .\n * .\n * Sends all the computed impulse responses to the backend server for processing\n *\n * @returns sets the resulting inverted impulse response to the class property\n * @example\n */\n sendSystemImpulseResponsesToServerForProcessing = async () => {\n this.addTimeStamp('Get system iir');\n const computedIRs = await Promise.all(this.impulseResponses);\n const filteredComputedIRs = computedIRs.filter(element => {\n return element != undefined;\n }); //log any errors that are found in this step\n const mls = this.#mls;\n const lowHz = this.#lowHz; //gain of 1 below cutoff, need gain of 0\n const highHz = this.#highHz; //check error for anything other than 10 kHz\n const iirLength = this.iirLength;\n const num_periods = this.numMLSPerCapture + this.num_mls_to_skip;\n this.stepNum += 1;\n console.log('send impulse responses to server: ' + this.stepNum);\n this.status =\n `All Hz Calibration: computing the IIR...`.toString() + this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return await this.pyServerAPI\n .getSystemInverseImpulseResponseWithRetry({\n payload: filteredComputedIRs.slice(0, this.numCaptures),\n mls,\n lowHz,\n highHz,\n iirLength,\n num_periods,\n sampleRate: this.sourceSamplingRate || 96000,\n mlsAmplitude: Math.pow(10, this.power_dB / 20),\n calibrateSoundBurstFilteredExtraDb: this._calibrateSoundBurstFilteredExtraDb,\n })\n .then(async res => {\n this.stepNum += 1;\n console.log('got impulse response ' + this.stepNum);\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the IIR...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n this.systemInvertedImpulseResponse = res['iir'];\n this.systemIR = res['ir']; \n this.systemInvertedImpulseResponseNoBandpass = res['iirNoBandpass'];\n this.systemAttenuatorGainDB = res['attenuatorGain_dB'];\n this.systemFMaxHz = res['fMaxHz'];\n await this.pyServerAPI.checkMemory();\n await this.pyServerAPI.getConvolution({\n mls, \n inverse_response: this.systemInvertedImpulseResponse, \n inverse_response_no_bandpass: this.systemInvertedImpulseResponseNoBandpass,\n attenuatorGain_dB: this.systemAttenuatorGainDB ,\n mls_amplitude: Math.pow(10, this.power_dB / 20)\n }).then(result => {\n console.log(result);\n this.systemConvolution = result['convolution'];\n this.systemConvolutionNoBandpass = result['convolution_no_bandpass'];\n });\n \n\n // attenuate the system convolution if the amplitude is greater than this.calibrateSoundLimit\n // find max of absolute value of system convolution\n\n const max = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMaxValue)(this.systemConvolution);\n this.filteredMLSAttenuation.system = \n this.systemConvolution.reduce((a, b) => a + b**2, 0) / this.systemConvolution.length;\n this.filteredMLSAttenuation.maxAbsSystem = max;\n })\n .catch(err => {\n console.error(err);\n });\n };\n\n /** .\n * .\n * .\n * Sends all the computed impulse responses to the backend server for processing\n *\n * @returns sets the resulting inverted impulse response to the class property\n * @example\n */\n sendComponentImpulseResponsesToServerForProcessing = async () => {\n this.addTimeStamp('Get component iir');\n const computedIRs = await Promise.all(this.impulseResponses);\n const filteredComputedIRs = computedIRs.filter(element => {\n return element != undefined;\n });\n const componentIRGains = this.componentIR['Gain'];\n const componentIRFreqs = this.componentIR['Freq'];\n const mls = this.#mls;\n const lowHz = this.#lowHz;\n const iirLength = this.iirLength;\n const irLength = this.irLength;\n const num_periods = this.numMLSPerCapture + this.num_mls_to_skip;\n const highHz = this.#highHz;\n this.stepNum += 1;\n console.log('send impulse responses to server: ' + this.stepNum);\n this.status =\n `All Hz Calibration: computing the IIR...`.toString() + this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return this.pyServerAPI\n .getComponentInverseImpulseResponseWithRetry({\n payload: filteredComputedIRs.slice(0, this.numCaptures),\n mls,\n lowHz,\n highHz,\n iirLength,\n componentIRGains,\n componentIRFreqs,\n sampleRate: this.sourceSamplingRate || 96000,\n mlsAmplitude: Math.pow(10, this.power_dB / 20),\n irLength,\n calibrateSoundSmoothOctaves: this._calibrateSoundSmoothOctaves,\n calibrateSoundBurstFilteredExtraDb: this._calibrateSoundBurstFilteredExtraDb,\n })\n .then(async res => {\n this.stepNum += 1;\n console.log('got impulse response ' + this.stepNum);\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the IIR...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n this.componentInvertedImpulseResponse = res['iir'];\n this.componentInvertedImpulseResponseNoBandpass = res['iirNoBandpass'];\n this.componentIR['Gain'] = res['ir'];\n this.componentIR['Freq'] = res['frequencies'];\n this.componentIRPhase = res['component_angle'];\n this.systemIRPhase = res['system_angle'];\n this.componentIROrigin['Freq'] = res['frequencies'];\n this.componentIROrigin['Gain'] = res['irOrigin'];\n this.componentIRInTimeDomain = res['irTime'];\n this.componentAttenuatorGainDB = res['attenuatorGain_dB'];\n this.componentFMaxHz = res['fMaxHz'];\n await this.pyServerAPI.checkMemory();\n await this.pyServerAPI.getConvolution({\n mls, \n inverse_response: this.componentInvertedImpulseResponse, \n inverse_response_no_bandpass: this.componentInvertedImpulseResponseNoBandpass,\n attenuatorGain_dB: this.componentAttenuatorGainDB ,\n mls_amplitude: Math.pow(10, this.power_dB / 20)\n }).then(result => {\n console.log(result);\n this.componentConvolution = result['convolution'];\n this.componentConvolutionNoBandpass = result['convolution_no_bandpass'];\n });\n // attenuate the component convolution if the amplitude is greater than this.calibrateSoundLimit\n // find max of absolute value of component convolution\n const max = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMaxValue)(this.componentConvolution);\n // if (max > this.calibrateSoundLimit) {\n // const gain = this.calibrateSoundLimit / max;\n // // apply gain to component convolution\n // this.componentConvolution = this.componentConvolution.map(value => value * gain);\n // this.filteredMLSAttenuation.component = gain;\n // }\n this.filteredMLSAttenuation.component = \n this.componentConvolution.reduce((a, b) => a + b**2, 0) / this.componentConvolution.length;\n this.filteredMLSAttenuation.maxAbsComponent = max;\n })\n .catch(err => {\n // this.emit('InvertedImpulseResponse', {res: false});\n console.error(err);\n });\n };\n\n sendBackgroundRecording = () => {\n const allSignals = this.getAllBackgroundRecordings();\n const numSignals = allSignals.length;\n const background_rec_whole = allSignals[numSignals - 1];\n const fraction = 0.5 / (this._calibrateSoundBackgroundSecs + 0.5);\n // Calculate the starting index for slicing the array\n const startIndex = Math.round(fraction * background_rec_whole.length);\n // Slice the array from the calculated start index to the end of the array\n const background_rec = background_rec_whole.slice(startIndex);\n console.log('Sending background recording to server for processing');\n this.addTimeStamp('Get background PSD');\n this.pyServerAPI\n .getBackgroundNoisePSDWithRetry({\n background_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n if (this.numSuccessfulBackgroundCaptured < 1) {\n this.numSuccessfulBackgroundCaptured += 1;\n //storing all background data in background_psd object\n this.background_noise['x_background'] = res['x_background'];\n this.background_noise['y_background'] = res['y_background'];\n this.background_noise['recording'] = background_rec;\n }\n })\n .catch(err => {\n console.error(err);\n });\n };\n\n /** .\n * .\n * .\n * Sends the recorded signal, or a given csv string of a signal, to the back end server for processing\n *\n * @param {<array>String} signalCsv - Optional csv string of a previously recorded signal, if given, this signal will be processed\n * @example\n */\n sendRecordingToServerForProcessing = async signalCsv => {\n const allSignals = this.getAllUnfilteredRecordedSignals();\n console.log(\n 'Obtaining last all hz unfiltered recording from #allHzUnfilteredRecordings to send to server for processing'\n );\n const numSignals = allSignals.length;\n const mls = this.#mlsBufferView;\n const payload =\n signalCsv && signalCsv.length > 0 ? (0,_utils__WEBPACK_IMPORTED_MODULE_1__.csvToArray)(signalCsv) : allSignals[numSignals - 1];\n console.log('sending rec');\n this.stepNum += 1;\n console.log('send rec ' + this.stepNum);\n this.status =\n `All Hz Calibration Step: computing the IR of the last recording...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n if (this.isCalibrating) return null;\n await this.pyServerAPI\n .allHzPowerCheck({\n payload,\n sampleRate: this.sourceSamplingRate || 96000,\n binDesiredSec: this._calibrateSoundPowerBinDesiredSec,\n burstSec: this.desired_time_per_mls,\n repeats: this.numMLSPerCapture,\n })\n .then(async result => {\n if (result) {\n console.log(\"JS used memory:\", performance.memory.usedJSHeapSize/1024/1024, \"mb\");\n if (result['sd'] < this._calibrateSoundPowerDbSDToleratedDb) {\n this.recordingChecks['unfiltered'].push(result); \n await this.pyServerAPI.checkMemory();\n // let start = new Date().getTime() / 1000;\n // const payloadT = tf.tensor1d(payload);\n // payloadT.print();\n // const xfft = payloadT.rfft(); // tf.spe\n // xfft.array().then(array => {\n // console.log(\"fft:\", array);\n // let setItem = new Set(array);\n // console.log(\"all zero\", setItem.size === 1 && setItem.has(0));\n // });\n // console.log(\"dimention:\", xfft.shape);\n // let end = new Date().getTime() / 1000;\n // console.log(\"Time taken:\", end - start, \"seconds\");\n await this.pyServerAPI.getAutocorrelation({\n sampleRate: this.sourceSamplingRate || 96000,\n payload,\n mls,\n numPeriods: this.numMLSPerCapture,\n }).then(async res=>{\n this.autocorrelations.push(res['autocorrelation']);\n this.fs2 = res['fs2'];\n this.L_new_n = res['L_new_n'];\n this.dL_n = res['dL_n'];\n await this.pyServerAPI.checkMemory();\n this.impulseResponses.push(\n await this.pyServerAPI\n .getImpulseResponse({\n mls,\n sampleRate: this.sourceSamplingRate || 96000,\n numPeriods: this.numMLSPerCapture,\n sig: payload,\n fs2: this.fs2,\n L_new_n: this.L_new_n,\n dL_n: this.dL_n\n })\n .then(res => {\n if (this.numSuccessfulCaptured < this.numCaptures) {\n this.numSuccessfulCaptured += 1;\n console.log('num succ capt: ' + this.numSuccessfulCaptured);\n this.stepNum += 1;\n console.log('got impulse response ' + this.stepNum);\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: ${this.numSuccessfulCaptured}/${this.numCaptures} IRs computed...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {\n message: this.status,\n });\n return res['ir'];\n }\n })\n .catch(err => {\n console.error(err);\n })\n );\n })\n \n } else if (result['sd'] > this._calibrateSoundPowerDbSDToleratedDb) {\n this.clearLastUnfilteredRecordedSignals();\n console.log('unfiltered rec', this.getAllUnfilteredRecordedSignals.length);\n }\n }\n })\n .catch(err => {\n console.error(err);\n });\n };\n\n /**\n * Passed to the calibration steps function, awaits the desired amount of seconds to capture the desired number\n * of MLS periods defined in the constructor.\n *\n * @example\n */\n #awaitDesiredMLSLength = async () => {\n // seconds per MLS = P / SR\n // await N * P / SR\n this.stepNum += 1;\n console.log('await desired length ' + this.stepNum);\n this.status =\n `All Hz Calibration: sampling the calibration signal...`.toString() +\n `\\niteration ${this.stepNum}` +\n this.generateTemplate();\n this.emit('update', {\n message: this.status,\n });\n let time_to_wait = 0;\n if (this.mode === 'unfiltered') {\n //unfiltered\n time_to_wait = (this.#mls.length / this.sourceSamplingRate) * this.numMLSPerCapture;\n time_to_wait = time_to_wait * 1.1;\n } else if (this.mode === 'filtered') {\n //filtered\n // time_to_wait =\n // (this.#currentConvolution.length / this.sourceSamplingRate) *\n // (this.numMLSPerCapture / (this.num_mls_to_skip + this.numMLSPerCapture));\n time_to_wait =\n (this.#currentConvolution.length / this.sourceSamplingRate) * this.numMLSPerCapture;\n time_to_wait = time_to_wait * 1.1;\n } else {\n throw new Error('Mode broke in awaitDesiredMLSLength');\n }\n\n await (0,_utils__WEBPACK_IMPORTED_MODULE_1__.sleep)(time_to_wait);\n };\n\n /**\n * Passed to the background noise recording function, awaits the desired amount of seconds to capture the desired number\n * of seconds of background noise\n *\n * @example\n */\n #awaitBackgroundNoiseRecording = async () => {\n console.log(\n 'Waiting ' + this._calibrateSoundBackgroundSecs + ' second(s) to record background noise'\n );\n let time_to_wait = this._calibrateSoundBackgroundSecs + 0.5;\n await (0,_utils__WEBPACK_IMPORTED_MODULE_1__.sleep)(time_to_wait);\n };\n\n /** .\n * .\n * .\n * Passed to the calibration steps function, awaits the onset of the signal to ensure a steady state\n *\n * @example\n */\n #awaitSignalOnset = async () => {\n this.stepNum += 1;\n console.log('await signal onset ' + this.stepNum);\n this.status =\n `All Hz Calibration: waiting for the signal to stabilize...`.toString() +\n this.generateTemplate();\n this.emit('update', {\n message: this.status,\n });\n let number_of_bursts_to_skip = this.num_mls_to_skip;\n let time_to_sleep = 0;\n if (this.mode === 'unfiltered') {\n time_to_sleep = (this.#mls.length / this.sourceSamplingRate) * number_of_bursts_to_skip;\n } else if (this.mode === 'filtered') {\n console.log(this.#currentConvolution.length);\n // time_to_sleep =\n // (this.#currentConvolution.length / this.sourceSamplingRate) *\n // (number_of_bursts_to_skip / (number_of_bursts_to_skip + this.numMLSPerCapture));\n time_to_sleep =\n (this.#currentConvolution.length / this.sourceSamplingRate) * number_of_bursts_to_skip;\n } else {\n throw new Error('Mode broke in awaitSignalOnset');\n }\n await (0,_utils__WEBPACK_IMPORTED_MODULE_1__.sleep)(time_to_sleep);\n };\n\n /**\n * Called immediately after a recording is captured. Used to process the resulting signal\n * whether by sending the result to a server or by computing a result locally.\n *\n * @example\n */\n #afterMLSRecord = async () => {\n console.log('after record');\n await this.sendRecordingToServerForProcessing();\n };\n\n #afterMLSwIIRRecord = async () => {\n await this.checkPowerVariation();\n };\n\n /** .\n * .\n * .\n * Created an S Curver Buffer to taper the signal onset\n *\n * @param {*} length\n * @param {*} phase\n * @returns\n * @example\n */\n static createSCurveBuffer = (length, phase) => {\n const curve = new Float32Array(length);\n let i;\n for (i = 0; i < length; i += 1) {\n // scale the curve to be between 0-1\n curve[i] = Math.sin((Math.PI * i) / length - phase) / 2 + 0.5;\n }\n return curve;\n };\n\n static createInverseSCurveBuffer = (length, phase) => {\n const curve = new Float32Array(length);\n let i;\n let j = length - 1;\n for (i = 0; i < length; i += 1) {\n // scale the curve to be between 0-1\n curve[i] = Math.sin((Math.PI * j) / length - phase) / 2 + 0.5;\n j -= 1;\n }\n return curve;\n };\n\n /**\n * Construct a Calibration Node with the calibration parameters.\n *\n * @param dataBuffer\n * @private\n * @example\n */\n #createCalibrationNodeFromBuffer = dataBuffer => {\n console.log('length databuffer');\n console.log(dataBuffer.length);\n if (!this.sourceAudioContext) {\n this.makeNewSourceAudioContext();\n }\n\n const buffer = this.sourceAudioContext.createBuffer(\n 1, // number of channels\n dataBuffer.length,\n this.sourceAudioContext.sampleRate // sample rate\n );\n\n const data = buffer.getChannelData(0); // get data\n\n // fill the buffer with our data\n try {\n for (let i = 0; i < dataBuffer.length; i += 1) {\n data[i] = dataBuffer[i];\n }\n } catch (error) {\n console.error(error);\n }\n\n this.sourceNode = this.sourceAudioContext.createBufferSource();\n\n this.sourceNode.buffer = buffer;\n\n if (this.mode === 'filtered') {\n //used to not loop filtered\n this.sourceNode.loop = true;\n } else {\n this.sourceNode.loop = true;\n }\n\n this.sourceNode.connect(this.sourceAudioContext.destination);\n\n this.addCalibrationNode(this.sourceNode);\n };\n\n /**\n * Given a data buffer, creates the required calibration node\n *\n * @param {*} dataBufferArray\n * @example\n */\n #setCalibrationNodesFromBuffer = (dataBufferArray = [this.#mlsBufferView]) => {\n if (dataBufferArray.length === 1) {\n this.#createCalibrationNodeFromBuffer(dataBufferArray[0]);\n } else {\n throw new Error('The length of the data buffer array must be 1');\n }\n };\n\n /**\n * Creates an audio context and plays it for a few seconds.\n *\n * @private\n * @returns - Resolves when the audio is done playing.\n * @example\n */\n #playCalibrationAudio = () => {\n this.addTimeStamp('Play unfiltered mls');\n this.calibrationNodes[0].start(0);\n this.status = ``;\n if (this.mode === 'unfiltered') {\n console.log('play calibration audio ' + this.stepNum);\n this.status =\n `All Hz Calibration: playing the calibration tone...`.toString() +\n this.generateTemplate().toString();\n } else if (this.mode === 'filtered') {\n console.log('play convolved audio ' + this.stepNum);\n this.status =\n `All Hz Calibration: playing the convolved calibration tone...`.toString() +\n this.generateTemplate().toString();\n } else {\n throw new Error('Mode is incorrect');\n }\n this.emit('update', {message: this.status});\n this.stepNum += 1;\n console.log('sink sampling rate');\n console.log(this.sinkSamplingRate);\n console.log('source sampling rate');\n console.log(this.sourceSamplingRate);\n console.log('sample size');\n console.log(this.sampleSize);\n };\n\n /** .\n * .\n * .\n * Stops the audio with tapered offset\n *\n * @example\n */\n stopCalibrationAudio = () => {\n if (this.calibrationNodes.length === 0) {\n return;\n }\n this.calibrationNodes[0].stop(0);\n this.calibrationNodes = [];\n if (this.sourceNode) this.sourceNode.disconnect();\n this.stepNum += 1;\n console.log('stop calibration audio ' + this.stepNum);\n this.status =\n `All Hz Calibration: stopping the calibration tone...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n };\n\n playMLSwithIIR = async (stream, convolution) => {\n let checkRec = false;\n this.mode = 'filtered';\n console.log('play mls with iir');\n //this.invertedImpulseResponse = iir\n\n await this.calibrationSteps(\n stream,\n this.#playCalibrationAudio, // play audio func (required)\n this.#createCalibrationNodeFromBuffer(convolution), // before play func\n this.#awaitSignalOnset, // before record\n () => this.numSuccessfulCaptured < 1,\n this.#awaitDesiredMLSLength, // during record\n this.#afterMLSwIIRRecord, // after record\n this.mode,\n checkRec\n );\n };\n\n bothSoundCheck = async stream => {\n let iir_ir_and_plots;\n this.#currentConvolution = this.componentConvolution;\n this.filteredMLSRange.component.Min = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMinValue)(this.#currentConvolution);\n this.filteredMLSRange.component.Max = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMaxValue)(this.#currentConvolution);\n this.addTimeStamp('Play MLS with component IIR');\n this.soundCheck = 'component';\n if (this.isCalibrating) return null;\n await this.playMLSwithIIR(stream, this.#currentConvolution);\n this.stopCalibrationAudio();\n let component_conv_recs = this.getAllFilteredRecordedSignals();\n\n if (this.componentAttentuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.componentAttentuatorGainDB/20);\n component_conv_recs = component_conv_recs.map(rec => {\n return rec.map(value => value / this.linearScaleAttenuation);\n });\n }\n\n let return_component_conv_rec = component_conv_recs[component_conv_recs.length - 1];\n this.clearAllFilteredRecordedSignals();\n // await this.checkPowerVariation(return_component_conv_rec);\n this.numSuccessfulCaptured = 0;\n this.#currentConvolution = this.systemConvolution;\n this.filteredMLSRange.system.Min = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMinValue)(this.#currentConvolution);\n this.filteredMLSRange.system.Max = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMaxValue)(this.#currentConvolution);\n this.soundCheck = 'system';\n this.addTimeStamp('Play MLS with system IIR');\n if (this.isCalibrating) return null;\n await this.playMLSwithIIR(stream, this.#currentConvolution);\n\n this.stopCalibrationAudio();\n\n let system_conv_recs = this.getAllFilteredRecordedSignals();\n\n if (this.systemAttenuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.systemAttenuatorGainDB/20);\n system_conv_recs = system_conv_recs.map(rec => {\n return rec.map(value => value / linearScaleAttenuation);\n });\n }\n\n let return_system_conv_rec = system_conv_recs[system_conv_recs.length - 1];\n // await this.checkPowerVariation(return_system_conv_rec);\n\n this.clearAllFilteredRecordedSignals();\n\n this.sourceAudioContext.close();\n let recs = this.getAllUnfilteredRecordedSignals();\n if (this.componentAttentuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.componentAttentuatorGainDB/20);\n recs = recs.map(rec => {\n return rec.map(value => value / this.linearScaleAttenuation);\n });\n }\n let unconv_rec = recs[0];\n let return_unconv_rec = unconv_rec;\n let conv_rec = component_conv_recs[component_conv_recs.length - 1];\n\n //psd of component\n let knownGain = this.oldComponentIR.Gain;\n let knownFreq = this.oldComponentIR.Freq;\n let sampleRate = this.sourceSamplingRate || 96000;\n this.addTimeStamp('Get PSD of mls recording');\n if (this.isCalibrating) return null;\n let component_unconv_rec_psd = await this.pyServerAPI\n .getSubtractedPSDWithRetry(unconv_rec, knownGain, knownFreq, sampleRate)\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD of filtered recording (component)');\n if (this.isCalibrating) return null;\n let component_conv_rec_psd = await this.pyServerAPI\n .getSubtractedPSDWithRetry(conv_rec, knownGain, knownFreq, sampleRate)\n .then(res => {\n let interpolatedGain = res.x.map((freq, index) => {\n let i = 0;\n while (i < knownFreq.length && knownFreq[i] < freq) {\n i++;\n }\n if (i === 0 || i === knownFreq.length) {\n return knownGain[i];\n }\n return (0,_utils__WEBPACK_IMPORTED_MODULE_1__.interpolate)(freq, knownFreq[i - 1], knownFreq[i], knownGain[i - 1], knownGain[i]);\n });\n\n let correctedGain = res.y.map(\n (gain, index) => 10 * Math.log10(gain) - interpolatedGain[index]\n );\n\n let filtered_psd = correctedGain.filter(\n (value, index) => res.x[index] >= this.#lowHz && res.x[index] <= this.componentFMaxHz\n );\n\n this.SDofFilteredRange['component'] = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.standardDeviation)(filtered_psd);\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n conv_rec = system_conv_recs[system_conv_recs.length - 1];\n //psd of system\n this.addTimeStamp('Get PSD of filtered recording (system) and unfiltered recording');\n if (this.isCalibrating) return null;\n let system_recs_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n let filtered_psd = res.y_conv\n .filter(\n (value, index) => res.x_conv[index] >= this.#lowHz && res.x_conv[index] <= this.#highHz\n )\n .map(value => 10 * Math.log10(value));\n\n let mls_psd = res.y_unconv\n .filter(\n (value, index) =>\n res.x_unconv[index] >= this.#lowHz && res.x_conv[index] <= this.#highHz\n )\n .map(value => 10 * Math.log10(value));\n\n this.SDofFilteredRange['mls'] = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.standardDeviation)(mls_psd);\n this.SDofFilteredRange['system'] = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.standardDeviation)(filtered_psd);\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n //iir w/ and without bandpass psd. done\n unconv_rec = this.componentInvertedImpulseResponseNoBandpass;\n conv_rec = this.componentInvertedImpulseResponse;\n this.addTimeStamp('Get PSD of component iir and component iir no band pass');\n if (this.isCalibrating) return null;\n let component_iir_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n unconv_rec = this.systemInvertedImpulseResponseNoBandpass;\n conv_rec = this.systemInvertedImpulseResponse;\n this.addTimeStamp('Get PSD of system iir and system iir no band pass');\n if (this.isCalibrating) return null;\n let system_iir_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD of mls sequence');\n if (this.isCalibrating) return null;\n let mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({mls: this.#mlsBufferView, sampleRate: this.sourceSamplingRate || 96000})\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD of filered mls (system)');\n if (this.isCalibrating) return null;\n let system_filtered_mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.systemConvolution,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n let system_no_bandpass_filtered_mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.systemConvolution,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD of filered mls (component)');\n if (this.isCalibrating) return null;\n let component_filtered_mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.componentConvolution,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n let component_no_bandpass_filtered_mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.componentConvolutionNoBandpass,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n let gainValue = this.getGainDBSPL();\n\n iir_ir_and_plots = {\n filtered_recording: {\n component: return_component_conv_rec,\n system: return_system_conv_rec,\n },\n unfiltered_recording: this.getAllUnfilteredRecordedSignals()[0],\n system: {\n iir: this.systemInvertedImpulseResponse,\n iir_no_bandpass: this.systemInvertedImpulseResponseNoBandpass,\n ir: this.systemIR,\n iir_psd: {\n y: system_iir_psd['y_conv'],\n x: system_iir_psd['x_conv'],\n y_no_bandpass: system_iir_psd['y_unconv'],\n x_no_bandpass: system_iir_psd['x_unconv'],\n },\n filtered_mls_psd: {\n x: system_filtered_mls_psd['x_mls'],\n y: system_filtered_mls_psd['y_mls'],\n },\n filtered_no_bandpass_mls_psd: {\n x: system_no_bandpass_filtered_mls_psd['x_mls'],\n y: system_no_bandpass_filtered_mls_psd['y_mls'],\n },\n convolution: this.systemConvolution,\n convolutionNoBandpass: this.systemConvolutionNoBandpass,\n psd: {\n unconv: {\n x: system_recs_psd['x_unconv'],\n y: system_recs_psd['y_unconv'],\n },\n conv: {\n x: system_recs_psd['x_conv'],\n y: system_recs_psd['y_conv'],\n },\n },\n },\n component: {\n iir: this.componentInvertedImpulseResponse,\n iir_no_bandpass: this.componentInvertedImpulseResponseNoBandpass,\n ir: this.componentIR,\n ir_origin: this.componentIROrigin,\n ir_in_time_domain: this.componentIRInTimeDomain,\n iir_psd: {\n y: component_iir_psd['y_conv'],\n x: component_iir_psd['x_conv'],\n y_no_bandpass: component_iir_psd['y_unconv'],\n x_no_bandpass: component_iir_psd['x_unconv'],\n },\n filtered_mls_psd: {\n x: component_filtered_mls_psd['x_mls'],\n y: component_filtered_mls_psd['y_mls'],\n },\n filtered_no_bandpass_mls_psd: {\n x: component_no_bandpass_filtered_mls_psd['x_mls'],\n y: component_no_bandpass_filtered_mls_psd['y_mls'],\n },\n convolution: this.componentConvolution,\n convolutionNoBandpass: this.componentConvolutionNoBandpass,\n psd: {\n unconv: {\n x: component_unconv_rec_psd['x'],\n y: component_unconv_rec_psd['y'],\n },\n conv: {\n x: component_conv_rec_psd['x'],\n y: component_conv_rec_psd['y'],\n },\n },\n gainDBSPL: gainValue,\n },\n mls: this.#mlsBufferView,\n mls_psd: {\n x: mls_psd['x_mls'],\n y: mls_psd['y_mls'],\n },\n autocorrelations: this.autocorrelations,\n impulseResponses: [],\n };\n\n return iir_ir_and_plots;\n };\n\n singleSoundCheck = async stream => {\n let iir_ir_and_plots;\n if (this._calibrateSoundCheck != 'system') {\n this.#currentConvolution = this.componentConvolution;\n this.filteredMLSRange.component.Min = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMinValue)(this.#currentConvolution);\n this.filteredMLSRange.component.Max = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMaxValue)(this.#currentConvolution);\n this.addTimeStamp('Play MLS with component IIR');\n this.soundCheck = 'component';\n if (this.isCalibrating) return null;\n await this.playMLSwithIIR(stream, this.#currentConvolution);\n this.stopCalibrationAudio();\n } else {\n this.#currentConvolution = this.systemConvolution;\n this.filteredMLSRange.system.Min = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMinValue)(this.#currentConvolution);\n this.filteredMLSRange.system.Max = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.findMaxValue)(this.#currentConvolution);\n this.addTimeStamp('Play MLS with system IIR');\n this.soundCheck = 'system';\n if (this.isCalibrating) return null;\n await this.playMLSwithIIR(stream, this.#currentConvolution);\n this.stopCalibrationAudio();\n }\n let conv_recs = this.getAllFilteredRecordedSignals();\n if (this._calibrateSoundCheck == 'goal'){\n if (this.componentAttentuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.componentAttentuatorGainDB/20);\n conv_recs = conv_recs.map(rec => {\n return rec.map(value => value / this.linearScaleAttenuation);\n });\n }\n }else if (this._calibrateSoundCheck == 'system'){\n if (this.systemAttentuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.systemAttentuatorGainDB/20);\n conv_recs = conv_recs.map(rec => {\n return rec.map(value => value / this.linearScaleAttenuation);\n });\n }\n }\n\n //remove the filteredMLSAttenuation from the recorded signals\n // conv_recs = conv_recs.map(rec => {\n // if (this.soundCheck === 'component') {\n // return rec.map(value => value / this.filteredMLSAttenuation.component);\n // }\n // return rec.map(value => value / this.filteredMLSAttenuation.system);\n // });\n\n\n let recs = this.getAllUnfilteredRecordedSignals();\n if (this._calibrateSoundCheck == 'goal'){\n if (this.componentAttentuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.componentAttentuatorGainDB/20);\n recs = recs.map(rec => {\n return rec.map(value => value / this.linearScaleAttenuation);\n });\n }\n }else if (this._calibrateSoundCheck == 'system'){\n if (this.systemAttentuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.systemAttentuatorGainDB/20);\n recs = recs.map(rec => {\n return rec.map(value => value / this.linearScaleAttenuation);\n });\n }\n }\n this.clearAllFilteredRecordedSignals();\n console.log('Obtaining unfiltered recording from #allHzUnfilteredRecordings to calculate PSD');\n console.log('Obtaining filtered recording from #allHzFilteredRecordings to calculate PSD');\n let unconv_rec = recs[0];\n let return_unconv_rec = unconv_rec;\n let conv_rec = conv_recs[conv_recs.length - 1];\n let return_conv_rec = conv_rec;\n this.sourceAudioContext.close();\n if (this._calibrateSoundCheck != 'system') {\n let knownGain = this.oldComponentIR.Gain;\n let knownFreq = this.oldComponentIR.Freq;\n let sampleRate = this.sourceSamplingRate || 96000;\n this.addTimeStamp('Get PSD of mls recording');\n if (this.isCalibrating) return null;\n let unconv_results = await this.pyServerAPI\n .getSubtractedPSDWithRetry(unconv_rec, knownGain, knownFreq, sampleRate)\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD recording of filtered recording (component)');\n if (this.isCalibrating) return null;\n let conv_results = await this.pyServerAPI\n .getSubtractedPSDWithRetry(conv_rec, knownGain, knownFreq, sampleRate)\n .then(res => {\n let interpolatedGain = res.x.map((freq, index) => {\n let i = 0;\n while (i < knownFreq.length && knownFreq[i] < freq) {\n i++;\n }\n if (i === 0 || i === knownFreq.length) {\n return knownGain[i];\n }\n return (0,_utils__WEBPACK_IMPORTED_MODULE_1__.interpolate)(\n freq,\n knownFreq[i - 1],\n knownFreq[i],\n knownGain[i - 1],\n knownGain[i]\n );\n });\n\n let correctedGain = res.y.map(\n (gain, index) => 10 * Math.log10(gain) - interpolatedGain[index]\n );\n let filtered_psd = correctedGain.filter(\n (value, index) => res.x[index] >= this.#lowHz && res.x[index] <= this.#highHz\n );\n\n this.SDofFilteredRange['component'] = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.standardDeviation)(filtered_psd);\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n unconv_rec = this.componentInvertedImpulseResponseNoBandpass;\n conv_rec = this.componentInvertedImpulseResponse;\n this.addTimeStamp('Get PSD of component iir and component iir no bandpass');\n if (this.isCalibrating) return null;\n let component_iir_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n unconv_rec = this.systemInvertedImpulseResponseNoBandpass;\n conv_rec = this.systemInvertedImpulseResponse;\n this.addTimeStamp('Get PSD of system iir and system iir no bandpass');\n if (this.isCalibrating) return null;\n let system_iir_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD of mls sequence');\n if (this.isCalibrating) return null;\n let mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.#mlsBufferView,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD of filtered mls (component)');\n if (this.isCalibrating) return null;\n let filtered_mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.componentConvolution,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n let filtered_no_bandpass_mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.componentConvolutionNoBandpass,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n let gainValue = this.getGainDBSPL();\n iir_ir_and_plots = {\n unfiltered_recording: return_unconv_rec,\n filtered_recording: return_conv_rec,\n system: {\n iir: this.systemInvertedImpulseResponse,\n iir_no_bandpass: this.systemInvertedImpulseResponseNoBandpass,\n ir: this.systemIR,\n iir_psd: {\n y: system_iir_psd['y_conv'],\n x: system_iir_psd['y_conv'],\n y_no_bandpass: system_iir_psd['y_unconv'],\n x_no_bandpass: system_iir_psd['x_unconv'],\n },\n filtered_recording: [],\n filtered_mls_psd: {},\n filtered_no_bandpass_mls_psd: {},\n convolution: this.systemConvolution,\n convolutionNoBandpass: this.systemConvolutionNoBandpass,\n psd: {\n unconv: {\n x: [],\n y: [],\n },\n conv: {\n x: [],\n y: [],\n },\n },\n },\n component: {\n iir: this.componentInvertedImpulseResponse,\n iir_no_bandpass: this.componentInvertedImpulseResponseNoBandpass,\n ir: this.componentIR,\n ir_origin: this.componentIROrigin,\n ir_in_time_domain: this.componentIRInTimeDomain,\n iir_psd: {\n y: component_iir_psd['y_conv'],\n x: component_iir_psd['x_conv'],\n y_no_bandpass: component_iir_psd['y_unconv'],\n x_no_bandpass: component_iir_psd['x_unconv'],\n },\n filtered_mls_psd: {\n x: filtered_mls_psd['x_mls'],\n y: filtered_mls_psd['y_mls'],\n },\n filtered_no_bandpass_mls_psd: {\n x: filtered_no_bandpass_mls_psd['x_mls'],\n y: filtered_no_bandpass_mls_psd['y_mls'],\n },\n convolution: this.componentConvolution,\n convolutionNoBandpass: this.componentConvolutionNoBandpass,\n psd: {\n unconv: {\n x: unconv_results['x'],\n y: unconv_results['y'],\n },\n conv: {\n x: conv_results['x'],\n y: conv_results['y'],\n },\n },\n gainDBSPL: gainValue,\n },\n mls: this.#mlsBufferView,\n mls_psd: {\n x: mls_psd['x_mls'],\n y: mls_psd['y_mls'],\n },\n autocorrelations: this.autocorrelations,\n impulseResponses: [],\n };\n } else {\n this.addTimeStamp('Get PSD of filtered recording (system) and unfiltered recording');\n if (this.isCalibrating) return null;\n let results = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n let filtered_psd = res.y_conv\n .filter(\n (value, index) =>\n res.x_conv[index] >= this.#lowHz && res.x_conv[index] <= this.systemFMaxHz\n )\n .map(value => 10 * Math.log10(value));\n\n let mls_psd = res.y_unconv\n .filter(\n (value, index) =>\n res.x_unconv[index] >= this.#lowHz && res.x_conv[index] <= this.systemFMaxHz\n )\n .map(value => 10 * Math.log10(value));\n\n this.SDofFilteredRange['system'] = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.standardDeviation)(filtered_psd);\n this.SDofFilteredRange['unfiltered'] = (0,_utils__WEBPACK_IMPORTED_MODULE_1__.standardDeviation)(mls_psd);\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n //iir w/ and without bandpass psd\n unconv_rec = this.componentInvertedImpulseResponseNoBandpass;\n conv_rec = this.componentInvertedImpulseResponse;\n this.addTimeStamp('Get PSD of component iir and component iir no band pass');\n if (this.isCalibrating) return null;\n let component_iir_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n unconv_rec = this.systemInvertedImpulseResponseNoBandpass;\n conv_rec = this.systemInvertedImpulseResponse;\n this.addTimeStamp('Get PSD of system iir and system iir no band pass');\n if (this.isCalibrating) return null;\n let system_iir_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD of mls sequence');\n if (this.isCalibrating) return null;\n let mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.#mlsBufferView,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n this.addTimeStamp('Get PSD of filtered mls (system)');\n if (this.isCalibrating) return null;\n let filtered_mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.systemConvolution,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n let filtered_no_bandpass_mls_psd = await this.pyServerAPI\n .getMLSPSDWithRetry({\n mls: this.systemConvolutionNoBandpass,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n let gainValue = this.getGainDBSPL();\n iir_ir_and_plots = {\n unfiltered_recording: return_unconv_rec,\n filtered_recording: return_conv_rec,\n system: {\n iir: this.systemInvertedImpulseResponse,\n iir_no_bandpass: this.systemInvertedImpulseResponseNoBandpass,\n ir: this.systemIR,\n iir_psd: {\n y: system_iir_psd['y_conv'],\n x: system_iir_psd['y_conv'],\n y_no_bandpass: system_iir_psd['y_unconv'],\n x_no_bandpass: system_iir_psd['x_unconv'],\n },\n filtered_recording: [],\n filtered_mls_psd: {\n x: filtered_mls_psd['x_mls'],\n y: filtered_mls_psd['y_mls'],\n },\n filtered_no_bandpass_mls_psd: {\n x: filtered_no_bandpass_mls_psd['x_mls'],\n y: filtered_no_bandpass_mls_psd['y_mls'],\n },\n convolution: this.systemConvolution,\n convolutionNoBandpass: this.systemConvolutionNoBandpass,\n psd: {\n unconv: {\n x: results['x_unconv'],\n y: results['y_unconv'],\n },\n conv: {\n x: results['x_conv'],\n y: results['y_conv'],\n },\n },\n },\n component: {\n iir: this.componentInvertedImpulseResponse,\n iir_no_bandpass: this.componentInvertedImpulseResponseNoBandpass,\n ir: this.componentIR,\n ir_origin: this.componentIROrigin,\n ir_in_time_domain: this.componentIRInTimeDomain,\n iir_psd: {\n y: component_iir_psd['y_conv'],\n x: component_iir_psd['x_conv'],\n y_no_bandpass: component_iir_psd['y_unconv'],\n x_no_bandpass: component_iir_psd['x_unconv'],\n },\n filtered_mls_psd: {},\n filtered_no_bandpass_mls_psd: {},\n convolution: this.componentConvolution,\n convolutionNoBandpass: this.componentConvolutionNoBandpass,\n psd: {\n unconv: {\n x: [],\n y: [],\n },\n conv: {\n x: [],\n y: [],\n },\n },\n gainDBSPL: gainValue,\n },\n mls: this.#mlsBufferView,\n mls_psd: {\n x: mls_psd['x_mls'],\n y: mls_psd['y_mls'],\n },\n autocorrelations: this.autocorrelations,\n impulseResponses: [],\n };\n }\n if (this.isCalibrating) return null;\n await Promise.all(this.impulseResponses).then(res => {\n for (let i = 0; i < res.length; i++) {\n if (res[i] != undefined) {\n iir_ir_and_plots['impulseResponses'].push(res[i]);\n }\n }\n });\n\n if (this.#download) {\n this.downloadSingleUnfilteredRecording();\n this.downloadSingleFilteredRecording();\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.#mls, 'MLS.csv');\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.componentConvolution, 'python_component_convolution_mls_iir.csv');\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.systemConvolution, 'python_system_convolution_mls_iir.csv');\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.componentInvertedImpulseResponse, 'componentIIR.csv');\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.systemInvertedImpulseResponse, 'systemIIR.csv');\n for (let i = 0; i < this.autocorrelations.length; i++) {\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.autocorrelations[i], `autocorrelation_${i}`);\n }\n const computedIRagain = await Promise.all(this.impulseResponses).then(res => {\n for (let i = 0; i < res.length; i++) {\n if (res[i] != undefined) {\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(res[i], `IR_${i}`);\n }\n }\n });\n }\n\n return iir_ir_and_plots;\n };\n\n /**\n * Public method to start the calibration process. Objects intialized from webassembly allocate new memory\n * and must be manually freed. This function is responsible for intializing the MlsGenInterface,\n * and wrapping the calibration steps with a garbage collection safe gaurd.\n *\n * @public\n * @param stream - The stream of audio from the Listener.\n * @example\n */\n startCalibrationImpulseResponse = async stream => {\n console.log(\"JS used memory:\", performance.memory.usedJSHeapSize/1024/1024, \"mb\");\n let desired_time = this.desired_time_per_mls;\n let checkRec = 'allhz';\n\n console.log('MLS sequence should be of length: ' + this.sourceSamplingRate * desired_time);\n\n length = this.sourceSamplingRate * desired_time;\n //get mls here\n // const calibrateSoundBurstDb = Math.pow(10, this._calibrateSoundBurstDb / 20);\n\n this.power_dB = 0;\n\n if (!this._calibrateSoundBurstLevelReTBool){\n this.power_dB =this._calibrateSoundBurstDb;\n }else{\n this.power_dB = this._calibrateSoundBurstDb+(this.T-this.gainDBSPL);\n }\n\n const amplitude = Math.pow(10,this.power_dB / 20);\n //MLSpower = Math.pow(10,this.power_dB/20);\n this.addTimeStamp('Get MLS sequence');\n if (this.isCalibrating) return null;\n await this.pyServerAPI\n .getMLSWithRetry({length, amplitude})\n .then(res => {\n console.log(res);\n this.#mlsBufferView = res['mls'];\n this.#mls = res['unscaledMLS'];\n })\n .catch(err => {\n // this.emit('InvertedImpulseResponse', {res: false});\n console.error(err);\n });\n this.numSuccessfulBackgroundCaptured = 0;\n if (this._calibrateSoundBackgroundSecs > 0) {\n this.mode = 'background';\n this.status =\n `All Hz Calibration: sampling the background noise...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n if (this.isCalibrating) return null;\n await this.recordBackground(\n stream, //stream\n () => this.numSuccessfulBackgroundCaptured < 1, //loop condition\n this.#awaitBackgroundNoiseRecording, //sleep to record\n this.sendBackgroundRecording, //send to get PSD\n this.mode,\n checkRec\n );\n this.incrementStatusBar();\n }\n this.mode = 'unfiltered';\n this.numSuccessfulCaptured = 0;\n\n if (this.isCalibrating) return null;\n await this.calibrationSteps(\n stream,\n this.#playCalibrationAudio, // play audio func (required)\n this.#createCalibrationNodeFromBuffer(this.#mlsBufferView), // before play func\n this.#awaitSignalOnset, // before record\n () => this.numSuccessfulCaptured < this.numCaptures, // loop while true\n this.#awaitDesiredMLSLength, // during record\n this.#afterMLSRecord, // after record\n this.mode,\n checkRec\n );\n this.stopCalibrationAudio();\n checkRec = false;\n\n // at this stage we've captured all the required signals,\n // and have received IRs for each one\n // so let's send all the IRs to the server to be converted to a single IIR\n if (this.isCalibrating) return null;\n await this.sendSystemImpulseResponsesToServerForProcessing();\n await this.pyServerAPI.checkMemory();\n if (this.isCalibrating) return null;\n await this.sendComponentImpulseResponsesToServerForProcessing();\n\n this.numSuccessfulCaptured = 0;\n\n let iir_ir_and_plots;\n if (this._calibrateSoundCheck != 'none') {\n //do single check\n if (this._calibrateSoundCheck == 'goal' || this._calibrateSoundCheck == 'system') {\n if (this.isCalibrating) return null;\n iir_ir_and_plots = await this.singleSoundCheck(stream);\n if (this.isCalibrating) return null;\n } else {\n //both\n if (this.isCalibrating) return null;\n iir_ir_and_plots = await this.bothSoundCheck(stream);\n if (this.isCalibrating) return null;\n }\n } else {\n let unconv_rec = this.componentInvertedImpulseResponseNoBandpass;\n let conv_rec = this.componentInvertedImpulseResponse;\n if (this.isCalibrating) return null;\n let component_iir_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n unconv_rec = this.systemInvertedImpulseResponseNoBandpass;\n conv_rec = this.systemInvertedImpulseResponse;\n if (this.isCalibrating) return null;\n let system_iir_psd = await this.pyServerAPI\n .getPSDWithRetry({\n unconv_rec,\n conv_rec,\n sampleRate: this.sourceSamplingRate || 96000,\n })\n .then(res => {\n this.incrementStatusBar();\n this.status =\n `All Hz Calibration: done computing the PSD graphs...`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n return res;\n })\n .catch(err => {\n console.error(err);\n });\n\n let gainValue = this.getGainDBSPL();\n iir_ir_and_plots = {\n unfiltered_recording: return_unconv_rec,\n filtered_recording: return_conv_rec,\n system: {\n iir: this.systemInvertedImpulseResponse,\n iir_no_bandpass: this.systemInvertedImpulseResponseNoBandpass,\n ir: this.systemIR,\n iir_psd: {\n y: system_iir_psd['y_conv'],\n x: system_iir_psd['y_conv'],\n y_no_bandpass: system_iir_psd['y_unconv'],\n x_no_bandpass: system_iir_psd['x_unconv'],\n },\n filtered_recording: [],\n convolution: this.systemConvolution,\n convolutionNoBandpass: this.systemConvolutionNoBandpass,\n psd: {\n unconv: {\n x: [],\n y: [],\n },\n conv: {\n x: [],\n y: [],\n },\n },\n },\n component: {\n iir: this.componentInvertedImpulseResponse,\n iir_no_bandpass: this.componentInvertedImpulseResponseNoBandpass,\n ir: this.componentIR,\n ir_in_time_domain: this.componentIRInTimeDomain,\n iir_psd: {\n y: component_iir_psd['y_conv'],\n x: component_iir_psd['x_conv'],\n y_no_bandpass: component_iir_psd['y_unconv'],\n x_no_bandpass: component_iir_psd['x_unconv'],\n },\n convolution: this.componentConvolution,\n convolutionNoBandpass: this.componentConvolutionNoBandpass,\n psd: {\n unconv: {\n x: [],\n y: [],\n },\n conv: {\n x: [],\n y: [],\n },\n },\n gainDBSPL: gainValue,\n },\n mls: this.#mlsBufferView,\n autocorrelations: this.autocorrelations,\n impulseResponses: [],\n };\n if (this.isCalibrating) return null;\n await Promise.all(this.impulseResponses).then(res => {\n for (let i = 0; i < res.length; i++) {\n if (res[i] != undefined) {\n iir_ir_and_plots['impulseResponses'].push(res[i]);\n }\n }\n });\n\n if (this.#download) {\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.#mls, 'MLS.csv');\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.componentConvolution, 'python_component_convolution_mls_iir.csv');\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.systemConvolution, 'python_system_convolution_mls_iir.csv');\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.componentInvertedImpulseResponse, 'componentIIR.csv');\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.systemInvertedImpulseResponse, 'systemIIR.csv');\n for (let i = 0; i < this.autocorrelations.length; i++) {\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(this.autocorrelations[i], `autocorrelation_${i}`);\n }\n const computedIRagain = await Promise.all(this.impulseResponses).then(res => {\n for (let i = 0; i < res.length; i++) {\n if (res[i] != undefined) {\n (0,_utils__WEBPACK_IMPORTED_MODULE_1__.saveToCSV)(res[i], `IR_${i}`);\n }\n }\n });\n }\n }\n if (this.isCalibrating) return null;\n this.percent_complete = 100;\n this.status = `All Hz Calibration: Finished`.toString() + this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n\n //here after calibration we have the component calibration (either loudspeaker or microphone) in the same form as the componentIR\n //that was used to calibrate\n // saveToJSON(iir_ir_and_plots);\n return iir_ir_and_plots;\n };\n\n //////////////////////volume\n\n handleIncomingData = data => {\n console.log('Received data: ', data);\n if (data.type === 'soundGainDBSPL') {\n this.soundGainDBSPL = data.value;\n } else {\n throw new Error(`Unknown data type: ${data.type}`);\n }\n };\n createSCurveBuffer = (onSetBool = true) => {\n const curve = new Float32Array(this.TAPER_SECS * this.sourceSamplingRate + 1);\n const frequency = 1 / (4 * this.TAPER_SECS);\n let j = 0;\n for (let i = 0; i < this.TAPER_SECS * this.sourceSamplingRate + 1; i += 1) {\n const phase = 2 * Math.PI * frequency * j;\n const onsetTaper = Math.pow(Math.sin(phase), 2);\n const offsetTaper = Math.pow(Math.cos(phase), 2);\n curve[i] = onSetBool ? onsetTaper : offsetTaper;\n j += 1 / this.sourceSamplingRate;\n }\n return curve;\n };\n\n #getTruncatedSignal = (left = 3.5, right = 4.5) => {\n const start = Math.floor(left * this.sourceSamplingRate);\n const end = Math.floor(right * this.sourceSamplingRate);\n const result = Array.from(this.getLastVolumeRecordedSignal().slice(start, end));\n console.log(\n 'Obtaining last 1000 hz recording from #allVolumeRecordings to send for processing'\n );\n /**\n * function to check that capture was properly made\n * @param {*} list\n */\n const checkResult = list => {\n const setItem = new Set(list);\n if (setItem.size === 1 && setItem.has(0)) {\n console.warn(\n 'The last capture failed, all recorded signal is zero',\n this.getAllVolumeRecordedSignals()\n );\n this.stopCalibrationAudio();\n this.isCalibrating = true;\n // restartButton.style.display = 'none';\n this.emit('update', {message: 'Connection failed, hit restart button to reconnect'});\n }\n if (setItem.size === 0) {\n console.warn('The last capture failed, no recorded signal');\n this.stopCalibrationAudio();\n this.isCalibrating = true;\n // restartButton.style.display = 'none';\n this.emit('update', {message: 'Connection failed, hit restart button to reconnect'});\n }\n };\n checkResult(result);\n return result;\n };\n\n /** \n * \n * \n Construct a calibration Node with the calibration parameters and given gain value\n * @param {*} gainValue\n * */\n #createCalibrationToneWithGainValue = gainValue => {\n const audioContext = this.makeNewSourceAudioContext();\n const oscilator = audioContext.createOscillator();\n const gainNode = audioContext.createGain();\n const taperGainNode = audioContext.createGain();\n const offsetGainNode = audioContext.createGain();\n const totalDuration = this.CALIBRATION_TONE_DURATION * 1.2;\n\n oscilator.frequency.value = this.#CALIBRATION_TONE_FREQUENCY;\n oscilator.type = this.#CALIBRATION_TONE_TYPE;\n gainNode.gain.value = gainValue;\n\n oscilator.connect(gainNode);\n gainNode.connect(taperGainNode);\n const onsetCurve = this.createSCurveBuffer();\n taperGainNode.gain.setValueCurveAtTime(onsetCurve, 0, this.TAPER_SECS);\n taperGainNode.connect(offsetGainNode);\n const offsetCurve = this.createSCurveBuffer(false);\n offsetGainNode.gain.setValueCurveAtTime(\n offsetCurve,\n totalDuration - this.TAPER_SECS,\n this.TAPER_SECS\n );\n offsetGainNode.connect(audioContext.destination);\n\n this.addCalibrationNode(oscilator);\n };\n\n /**\n * Construct a Calibration Node with the calibration parameters.\n *\n * @private\n * @example\n */\n #createCalibrationNode = () => {\n const audioContext = this.makeNewSourceAudioContext();\n const oscilator = audioContext.createOscillator();\n const gainNode = audioContext.createGain();\n\n oscilator.frequency.value = this.#CALIBRATION_TONE_FREQUENCY;\n oscilator.type = this.#CALIBRATION_TONE_TYPE;\n gainNode.gain.value = 0.04;\n\n oscilator.connect(gainNode);\n gainNode.connect(audioContext.destination);\n\n this.addCalibrationNode(oscilator);\n };\n\n #playCalibrationAudioVolume = async () => {\n const totalDuration = this.CALIBRATION_TONE_DURATION * 1.2;\n\n this.calibrationNodes[0].start(0);\n this.calibrationNodes[0].stop(totalDuration);\n console.log(`Playing a buffer of ${this.CALIBRATION_TONE_DURATION} seconds of audio`);\n console.log(`Waiting a total of ${totalDuration} seconds`);\n await (0,_utils__WEBPACK_IMPORTED_MODULE_1__.sleep)(totalDuration);\n };\n\n stopCalibrationAudioVolume = () => {\n if (this.calibrationNodes.length > 0) {\n this.calibrationNodes[0].stop();\n }\n };\n\n #sendToServerForProcessing = async (lCalib) => {\n console.log('Sending data to server');\n this.addTimeStamp('Send volume data to server');\n let left = this.calibrateSound1000HzPreSec;\n let right = this.calibrateSound1000HzPreSec + this.calibrateSound1000HzSec;\n if (this.isCalibrating) return null;\n this.pyServerAPI\n .getVolumeCalibration({\n sampleRate: this.sourceSamplingRate,\n payload: this.#getTruncatedSignal(left, right),\n lCalib: lCalib,\n })\n .then(res => {\n if (this.outDBSPL === null) {\n this.incrementStatusBar();\n this.outDBSPL = res['outDbSPL'];\n this.outDBSPL1000 = res['outDbSPL1000'];\n this.THD = res['thd'];\n }\n })\n .catch(err => {\n console.warn(err);\n });\n\n await this.pyServerAPI\n .volumePowerCheck({\n payload: this.getLastVolumeRecordedSignal(),\n sampleRate: this.sourceSamplingRate || 96000,\n binDesiredSec: this._calibrateSoundPowerBinDesiredSec,\n preSec: this.calibrateSound1000HzPreSec,\n Sec: this.calibrateSound1000HzSec,\n })\n .then(res => {\n if (res && res['sd'] < this._calibrateSoundPowerDbSDToleratedDb) {\n this.recordingChecks['volume'][this.inDB] = res;\n }\n });\n };\n\n startCalibrationVolume = async (stream, gainValues, lCalib, componentGainDBSPL) => {\n console.log(\"JS used memory:\", performance.memory.usedJSHeapSize/1024/1024, \"mb\");\n if (this.isCalibrating) return null;\n const trialIterations = gainValues.length;\n this.status_denominator += trialIterations;\n const thdValues = [];\n const inDBValues = [];\n let inDB = 0;\n const outDBSPLValues = [];\n const outDBSPL1000Values = [];\n let checkRec = \"loudest\";\n // do one calibration that will be discarded\n const soundLevelToDiscard = -60;\n const gainToDiscard = Math.pow(10, soundLevelToDiscard / 20);\n this.inDB = soundLevelToDiscard;\n this.status =\n `1000 Hz Calibration: Sound Level ${soundLevelToDiscard} dB`.toString() +\n this.generateTemplate().toString();\n //this.emit('update', {message: `1000 Hz Calibration: Sound Level ${soundLevelToDiscard} dB`});\n this.emit('update', {message: this.status});\n this.startTime = new Date().getTime();\n\n do {\n console.log('while loop');\n if (this.isCalibrating) {\n console.log('restart calibration');\n return null;\n }\n // eslint-disable-next-line no-await-in-loop\n await this.volumeCalibrationSteps(\n stream,\n this.#playCalibrationAudioVolume,\n this.#createCalibrationToneWithGainValue,\n this.#sendToServerForProcessing,\n gainToDiscard,\n lCalib, //todo make this a class parameter\n checkRec\n );\n } while (this.outDBSPL === null);\n //reset the values\n //this.incrementStatusBar();\n\n this.outDBSPL = null;\n this.outDBSPL = null;\n this.outDBSPL1000 = null;\n this.THD = null;\n\n // run the calibration at different gain values provided by the user\n for (let i = 0; i < trialIterations; i++) {\n //convert gain to DB and add to inDB\n if (i == trialIterations - 1) {\n checkRec = 'loudest';\n }\n inDB = Math.log10(gainValues[i]) * 20;\n // precision to 1 decimal place\n inDB = Math.round(inDB * 10) / 10;\n this.inDB = inDB;\n inDBValues.push(inDB);\n console.log('next update');\n this.status =\n `1000 Hz Calibration: Sound Level ${inDB} dB`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {message: this.status});\n do {\n if (this.isCalibrating) {\n console.log('restart calibration');\n return null;\n }\n // eslint-disable-next-line no-await-in-loop\n await this.volumeCalibrationSteps(\n stream,\n this.#playCalibrationAudioVolume,\n this.#createCalibrationToneWithGainValue,\n this.#sendToServerForProcessing,\n gainValues[i],\n lCalib, //todo make this a class parameter\n checkRec\n );\n } while (this.outDBSPL === null);\n outDBSPL1000Values.push(this.outDBSPL1000);\n thdValues.push(this.THD);\n outDBSPLValues.push(this.outDBSPL);\n\n this.outDBSPL = null;\n this.outDBSPL1000 = null;\n this.THD = null;\n }\n if (this.isCalibrating) return null;\n // get the volume calibration parameters from the server\n this.addTimeStamp('Get Volume Calibration Parameters');\n\n const parameters = await this.pyServerAPI\n .getVolumeCalibrationParameters({\n inDBValues: inDBValues,\n outDBSPLValues: outDBSPL1000Values,\n lCalib: lCalib,\n componentGainDBSPL,\n })\n .then(res => {\n this.incrementStatusBar();\n return res;\n });\n if (this.isCalibrating) return null;\n const result = {\n parameters: parameters,\n inDBValues: inDBValues,\n outDBSPLValues: outDBSPLValues,\n outDBSPL1000Values: outDBSPL1000Values,\n thdValues: thdValues,\n };\n\n return result;\n };\n\n writeFrqGainToFirestore = async (speakerID, frq, gain, OEM, documentID) => {\n // freq and gain are too large to take samples 1 in every 100 samples\n // const sampledFrq = [];\n // const sampledGain = [];\n // for (let i = 0; i < frq.length; i += 100) {\n // sampledFrq.push(frq[i]);\n // sampledGain.push(gain[i]);\n // }\n\n const data = {Freq: frq, Gain: gain};\n\n const docRef = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.doc)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], 'Microphones', documentID);\n await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.updateDoc)(docRef, {\n linear: data,\n });\n\n // divide frq and gain into smaller chunks and write to firestore one chunk at a time\n // use arrayUnion to append to the array\n // const chunkSize = 600;\n // const chunkedFrq = [];\n // const chunkedGain = [];\n // for (let i = 0; i < frq.length; i += chunkSize) {\n // chunkedFrq.push(frq.slice(i, i + chunkSize));\n // chunkedGain.push(gain.slice(i, i + chunkSize));\n // }\n // const docRef = doc(database, 'Microphones', documentID);\n // for (let i = 0; i < chunkedFrq.length; i++) {\n // await updateDoc(docRef, {\n // linear: {\n // Freq: arrayUnion(...chunkedFrq[i]),\n // Gain: arrayUnion(...chunkedGain[i]),\n // },\n // });\n // }\n };\n // function to write frq and gain to firebase database given speakerID\n writeFrqGain = async (speakerID, frq, gain, OEM) => {\n // freq and gain are too large to take samples 1 in every 100 samples\n\n const sampledFrq = [];\n const sampledGain = [];\n for (let i = 0; i < frq.length; i += 100) {\n sampledFrq.push(frq[i]);\n sampledGain.push(gain[i]);\n }\n\n const data = {Freq: sampledFrq, Gain: sampledGain};\n\n await (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.set)((0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.ref)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], `Microphone2/${OEM}/${speakerID}/linear`), data);\n };\n\n // Function to Read frq and gain from firebase database given speakerID\n // returns an array of frq and gain if speakerID exists, returns null otherwise\n readFrqGainFromFirestore = async (speakerID, OEM, isDefault) => {\n const collectionRef = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.collection)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], 'Microphones');\n const q = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.query)(\n collectionRef,\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('ID', '==', speakerID),\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('lowercaseOEM', '==', OEM),\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('isDefault', '==', isDefault)\n );\n const querySnapshot = await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.getDocs)(q);\n // if exists return the linear field of the first document\n if (querySnapshot.size > 0) {\n return querySnapshot.docs[0].data().linear;\n }\n return null;\n };\n readFrqGain = async (speakerID, OEM) => {\n const dbRef = (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.ref)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"]);\n const snapshot = await (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.get)((0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.child)(dbRef, `Microphone2/${OEM}/${speakerID}/linear`));\n if (snapshot.exists()) {\n return snapshot.val();\n }\n return null;\n };\n readGainat1000HzFromFirestore = async (speakerID, OEM, isDefault) => {\n const collectionRef = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.collection)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], 'Microphones');\n const q = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.query)(\n collectionRef,\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('ID', '==', speakerID),\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('lowercaseOEM', '==', OEM),\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('isDefault', '==', isDefault)\n );\n const querySnapshot = await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.getDocs)(q);\n // if exists return the Gain1000 field of the first document\n if (querySnapshot.size > 0) {\n return querySnapshot.docs[0].data().Gain1000;\n }\n return null;\n };\n\n readGainat1000Hz = async (speakerID, OEM) => {\n const dbRef = (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.ref)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"]);\n const snapshot = await (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.get)((0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.child)(dbRef, `Microphone2/${OEM}/${speakerID}/Gain1000`));\n if (snapshot.exists()) {\n return snapshot.val();\n }\n return null;\n };\n\n writeGainat1000HzToFirestore = async (speakerID, gain, OEM, documentID) => {\n const docRef = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.doc)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], 'Microphones', documentID);\n await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.updateDoc)(docRef, {\n Gain1000: gain,\n });\n };\n\n writeGainat1000Hz = async (speakerID, gain, OEM) => {\n await (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.set)((0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.ref)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], `Microphone2/${OEM}/${speakerID}/Gain1000`), gain);\n };\n\n writeIsSmartPhoneToFirestore = async (speakerID, isSmartPhone, OEM) => {\n const collectionRef = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.collection)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], 'Microphones');\n const q = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.query)(\n collectionRef,\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('ID', '==', speakerID),\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('lowercaseOEM', '==', OEM),\n (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.where)('isDefault', '==', true)\n );\n const querySnapshot = await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.getDocs)(q);\n if (querySnapshot.size > 0) {\n const docRef = await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.addDoc)(collectionRef, {isSmartPhone: isSmartPhone, isDefault: false});\n return docRef.id;\n } else {\n const docRef = await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.addDoc)(collectionRef, {isSmartPhone: isSmartPhone, isDefault: true});\n return docRef.id;\n }\n };\n\n writeIsSmartPhone = async (speakerID, isSmartPhone, OEM) => {\n const data = {isSmartPhone: isSmartPhone};\n await (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.set)((0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.ref)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], `Microphone2/${OEM}/${speakerID}/isSmartPhone`), isSmartPhone);\n };\n\n writeMicrophoneInfoToFirestore = async (speakerID, micInfo, OEM, documentID) => {\n const docRef = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.doc)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], 'Microphones', documentID);\n await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.setDoc)(docRef, micInfo, {merge: true});\n };\n\n doesMicrophoneExistInFirestore = async (speakerID, OEM, documentID) => {\n const docRef = (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.doc)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], 'Microphone', OEM, speakerID, documentID);\n const docSnap = await (0,firebase_firestore__WEBPACK_IMPORTED_MODULE_4__.getDoc)(docRef);\n if (docSnap.exists()) {\n return true;\n }\n return false;\n };\n\n doesMicrophoneExist = async (speakerID, OEM) => {\n const dbRef = (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.ref)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"]);\n const snapshot = await (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.get)((0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.child)(dbRef, `Microphone2/${OEM}/${speakerID}`));\n if (snapshot.exists()) {\n return true;\n }\n return false;\n };\n\n addMicrophoneInfo = async (speakerID, OEM, micInfo) => {\n // add to database if /info does not exist\n const dbRef = (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.ref)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"]);\n const snapshot = await (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.get)((0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.child)(dbRef, `Microphone2/${OEM}/${speakerID}/info`));\n if (!snapshot.exists()) {\n await (0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.set)((0,firebase_database__WEBPACK_IMPORTED_MODULE_3__.ref)(_config_firebase__WEBPACK_IMPORTED_MODULE_2__[\"default\"], `Microphone2/${OEM}/${speakerID}/info`), micInfo);\n }\n };\n\n convertToDB = gain => {\n return Math.log10(gain) * 20;\n };\n\n // Function to perform linear interpolation between two points\n interpolate(x, x0, y0, x1, y1) {\n return y0 + ((x - x0) * (y1 - y0)) / (x1 - x0);\n }\n\n findGainatFrequency = (frequencies, gains, targetFrequency) => {\n // Find the index of the first frequency in the array greater than the target frequency\n let index = 0;\n while (index < frequencies.length && frequencies[index] < targetFrequency) {\n index++;\n }\n\n // Handle cases when the target frequency is outside the range of the given data\n if (index === 0) {\n return gains[0];\n } else if (index === frequencies.length) {\n return gains[gains.length - 1];\n } else {\n // Interpolate the gain based on the surrounding frequencies\n const x0 = frequencies[index - 1];\n const y0 = gains[index - 1];\n const x1 = frequencies[index];\n const y1 = gains[index];\n return this.interpolate(targetFrequency, x0, y0, x1, y1);\n }\n };\n\n\n checkPowerVariation = async () => {\n let recordings = this.getAllFilteredRecordedSignals();\n // remove filteredMLSAttenuation from the recordings\n\n // recordings = recordings.map(recording => {\n // if (this.soundCheck == 'component') {\n // return recording.map(value => value / this.filteredMLSAttenuation.component);\n // }\n // return recording.map(value => value / this.filteredMLSAttenuation.system);\n // });\n\n const rec = recordings[recordings.length - 1];\n\n await this.pyServerAPI\n .allHzPowerCheck({\n payload: rec,\n sampleRate: this.sourceSamplingRate || 96000,\n binDesiredSec: this._calibrateSoundPowerBinDesiredSec,\n burstSec: this.desired_time_per_mls,\n repeats: this.numMLSPerCapture,\n })\n .then(result => {\n if (result) {\n if (result['sd'] > this._calibrateSoundPowerDbSDToleratedDb) {\n console.log('filtered recording sd too high');\n } else {\n this.recordingChecks[this.soundCheck].push(result);\n if (this.numSuccessfulCaptured < 1) {\n this.numSuccessfulCaptured += 1;\n this.stepNum += 1;\n this.incrementStatusBar();\n console.log(\n 'after mls w iir record for some reason add numSucc capt ' + this.stepNum\n );\n this.status =\n `All Hz Calibration: ${this.numSuccessfulCaptured} recording of convolved MLS captured`.toString() +\n this.generateTemplate().toString();\n this.emit('update', {\n message: this.status,\n });\n }\n }\n }\n });\n };\n\n getGainDBSPL = () => {\n var freqIndex = this.componentIR.Freq.indexOf(1000);\n\n // If freqIndex is not -1 (meaning 1000 is found in the freq array)\n if (freqIndex !== -1) {\n // Get the corresponding gain value using the index\n var gainValue = this.componentIR.Gain[freqIndex];\n return gainValue;\n } else {\n console.log('Freq 1000 not found in the array.');\n return null;\n }\n };\n // Example of how to use the writeFrqGain and readFrqGain functions\n // writeFrqGain('speaker1', [1, 2, 3], [4, 5, 6]);\n // Speaker1 is the speakerID you want to write to in the database\n // readFrqGain('MiniDSPUMIK_1').then(data => console.log(data));\n // MiniDSPUMIK_1 is the speakerID with some Data in the database\n //adding gainDBSPL\n startCalibration = async (\n stream,\n gainValues,\n lCalib = 104.92978421490648,\n componentIR = null,\n microphoneName = 'MiniDSP-UMIK1-711-4754-vertical',\n _calibrateSoundCheck = 'goal', //GOAL PASSed in by default\n isSmartPhone = false,\n _calibrateSoundBurstDb = -18,\n _calibrateSoundBurstFilteredExtraDb = 6,\n _calibrateSoundBurstLevelReTBool = false,\n _calibrateSoundBurstUses1000HzGainBool = false,\n _calibrateSoundBurstRepeats = 3,\n _calibrateSoundBurstSec = 1,\n _calibrateSoundBurstsWarmup = 1,\n _calibrateSoundHz = 48000,\n _calibrateSoundIIRSec = 0.2,\n _calibrateSoundIRSec = 0.2,\n calibrateSound1000HzPreSec = 3.5,\n calibrateSound1000HzSec = 1.0,\n calibrateSound1000HzPostSec = 0.5,\n _calibrateSoundBackgroundSecs = 0,\n _calibrateSoundSmoothOctaves = 0.33,\n _calibrateSoundPowerBinDesiredSec = 0.2,\n _calibrateSoundPowerDbSDToleratedDb = 1,\n micManufacturer = '',\n micSerialNumber = '',\n micModelNumber = '',\n micModelName = '',\n calibrateMicrophonesBool,\n authorEmails,\n webAudioDeviceNames = {\n loudspeaker: 'loudspeaker',\n microphone: 'microphone',\n microphoneText: 'xxx XXX',\n },\n userIDs,\n restartButton,\n reminder,\n calibrateSoundLimit,\n ) => {\n this.calibrateSoundLimit = calibrateSoundLimit;\n this._calibrateSoundBurstDb = _calibrateSoundBurstDb;\n this._calibrateSoundBurstFilteredExtraDb = _calibrateSoundBurstFilteredExtraDb;\n this._calibrateSoundBurstLevelReTBool = _calibrateSoundBurstLevelReTBool;\n this.CALIBRATION_TONE_DURATION =\n calibrateSound1000HzPreSec + calibrateSound1000HzSec + calibrateSound1000HzPostSec;\n this.calibrateSound1000HzPreSec = calibrateSound1000HzPreSec;\n this.calibrateSound1000HzSec = calibrateSound1000HzSec;\n this.calibrateSound1000HzPostSec = calibrateSound1000HzPostSec;\n this.iirLength = Math.floor(_calibrateSoundIIRSec * this.sourceSamplingRate);\n this.irLength = Math.floor(_calibrateSoundIRSec * this.sourceSamplingRate);\n this.numMLSPerCapture = _calibrateSoundBurstRepeats + 1;\n this.desired_time_per_mls = _calibrateSoundBurstSec;\n this.num_mls_to_skip = _calibrateSoundBurstsWarmup;\n this.desired_sampling_rate = _calibrateSoundHz;\n this._calibrateSoundBackgroundSecs = _calibrateSoundBackgroundSecs;\n this._calibrateSoundSmoothOctaves = _calibrateSoundSmoothOctaves;\n this._calibrateSoundPowerBinDesiredSec = _calibrateSoundPowerBinDesiredSec;\n this._calibrateSoundBurstUses1000HzGainBool = _calibrateSoundBurstUses1000HzGainBool;\n this._calibrateSoundPowerDbSDToleratedDb = _calibrateSoundPowerDbSDToleratedDb;\n this.webAudioDeviceNames = webAudioDeviceNames;\n if (isSmartPhone) this.webAudioDeviceNames.microphone = this.deviceInfo.microphoneFromAPI;\n this.webAudioDeviceNames.microphoneText = this.webAudioDeviceNames.microphoneText\n .replace('xxx', this.webAudioDeviceNames.microphone)\n .replace('XXX', this.webAudioDeviceNames.microphone);\n //feed calibration goal here\n this._calibrateSoundCheck = _calibrateSoundCheck;\n //check if a componentIR was given to the system, if it isn't check for the microphone. using dummy data here bc we need to\n //check the db based on the microphone currently connected\n\n //new lCalib found at top of calibration files *1000hz, make sure to correct\n //based on zeroing of 1000hz, search for \"*1000Hz\"\n const ID = isSmartPhone ? micModelNumber : micSerialNumber;\n const OEM = isSmartPhone\n ? micModelName === 'UMIK-1' || micModelName === 'UMIK-2'\n ? 'minidsp'\n : this.deviceInfo.OEM.toLowerCase().split(' ').join('')\n : micManufacturer.toLowerCase().split(' ').join('');\n // const ID = \"712-5669\";\n // const OEM = \"minidsp\";\n const micInfo = {\n micModelName: isSmartPhone ? micModelName : microphoneName,\n OEM: isSmartPhone\n ? micModelName === 'UMIK-1' || micModelName === 'UMIK-2'\n ? 'miniDSP'\n : this.deviceInfo.OEM\n : micManufacturer,\n ID: ID,\n createDate: new Date(),\n DateText: (0,_utils__WEBPACK_IMPORTED_MODULE_1__.getCurrentTimeString)(),\n HardwareName: isSmartPhone ? this.deviceInfo.hardwarename : microphoneName,\n hardwareFamily: isSmartPhone ? this.deviceInfo.hardwarefamily : microphoneName,\n HardwareModel: isSmartPhone ? this.deviceInfo.hardwaremodel : microphoneName,\n PlatformName: isSmartPhone ? this.deviceInfo.platformname : 'N/A',\n PlatformVersion: isSmartPhone ? this.deviceInfo.platformversion : 'N/A',\n DeviceType: isSmartPhone ? this.deviceInfo.devicetype : 'N/A',\n ID_from_51Degrees: isSmartPhone ? this.deviceInfo.DeviceId : 'N/A',\n calibrateMicrophonesBool: calibrateMicrophonesBool,\n screenHeight:this.deviceInfo.screenHeight,\n screenWidth:this.deviceInfo.screenWidth,\n webAudioDeviceNames: {\n loudspeaker: this.webAudioDeviceNames.loudspeaker,\n microphone: this.webAudioDeviceNames.microphone,\n },\n userIDs: userIDs,\n lowercaseOEM: OEM.toLowerCase().split(' ').join(''),\n };\n if (calibrateMicrophonesBool) {\n micInfo['authorEmails'] = authorEmails;\n }\n // if undefined in micInfo, set to empty string\n for (const [key, value] of Object.entries(micInfo)) {\n if (value === undefined) {\n micInfo[key] = '';\n }\n }\n\n // this.writeMicrophoneInfoToFirestore(ID, micInfo, OEM, 'default');\n // this.addMicrophoneInfo(ID, OEM, micInfo);\n if (componentIR == null) {\n //mode 'ir'\n //global variable this.componentIR must be set\n this.componentIR = await this.readFrqGainFromFirestore(ID, OEM, true).then(data => {\n return data;\n });\n // await this.readFrqGain(ID, OEM).then(data => {\n // return data;\n // });\n\n // lCalib = await this.readGainat1000Hz(ID, OEM);\n lCalib = await this.readGainat1000HzFromFirestore(ID, OEM, true);\n micInfo['gainDBSPL'] = lCalib;\n // this.componentGainDBSPL = this.convertToDB(lCalib);\n this.componentGainDBSPL = lCalib;\n //TODO: if this call to database is unknown, cannot perform experiment => return false\n if (this.componentIR == null) {\n this.status =\n `Microphone (${OEM},${ID}) is not found in the database. Please add it to the database.`.toString();\n this.emit('update', {message: this.status});\n return false;\n }\n } else {\n this.transducerType = 'Microphone';\n this.componentIR = componentIR;\n lCalib = this.findGainatFrequency(this.componentIR.Freq, this.componentIR.Gain, 1000);\n // this.componentGainDBSPL = this.convertToDB(lCalib);\n this.componentGainDBSPL = lCalib;\n // await this.writeIsSmartPhone(ID, isSmartPhone, OEM);\n }\n\n this.oldComponentIR = JSON.parse(JSON.stringify(this.componentIR));\n\n return await new Promise(async (resolve, reject) => {\n // add event listner to params.restartButton to resolve the promise with a string: 'restart'\n\n if (reminder) {\n reminder.style.display = '';\n }\n if (restartButton) {\n restartButton.style.display = '';\n restartButton.addEventListener('click', () => {\n this.stopCalibrationAudio();\n this.isCalibrating = true;\n restartButton.style.display = 'none';\n if (reminder) {\n reminder.style.display = 'none';\n }\n this.emit('update', {message: 'Restarting calibration...'});\n resolve('restart');\n });\n }\n await this.pyServerAPI.checkMemory();\n let volumeResults = await this.startCalibrationVolume(\n stream,\n gainValues,\n lCalib,\n this.componentGainDBSPL\n );\n if (!volumeResults) return;\n\n this.T = volumeResults[\"parameters\"][\"T\"];\n this.gainDBSPL = volumeResults[\"parameters\"][\"gainDBSPL\"];\n\n let impulseResponseResults = await this.startCalibrationImpulseResponse(stream);\n if (!impulseResponseResults) return;\n impulseResponseResults['background_noise'] = this.background_noise;\n impulseResponseResults['system']['background_noise'] = this.background_noise;\n impulseResponseResults['component']['background_noise'] = this.background_noise;\n\n //attenuate system background noise\n if (this.systemAttenuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.systemAttenuatorGainDB/20);\n let linearScalePowerAttenuation = Math.pow(10,this.systemAttenuatorGainDB/10);\n impulseResponseResults['system']['background_noise']['recording'] = impulseResponseResults['background_noise']['recording'].map(value => value/linearScaleAttenuation);\n impulseResponseResults['system']['background_noise']['x_background'] = impulseResponseResults['background_noise']['x_background'];\n impulseResponseResults['system']['background_noise']['y_background'] = impulseResponseResults['background_noise']['y_background'].map(value => value/linearScalePowerAttenuation);\n }\n //attenuate component background noise\n if (this.componentAttentuatorGainDB != 0){\n let linearScaleAttenuation = Math.pow(10,this.componentAttenuatorGainDB/20);\n let linearScalePowerAttenuation = Math.pow(10,this.componentAttenuatorGainDB/10);\n impulseResponseResults['component']['background_noise']['recording'] = impulseResponseResults['background_noise']['recording'].map(value => value/linearScaleAttenuation);\n impulseResponseResults['component']['background_noise']['x_background'] = impulseResponseResults['background_noise']['x_background'];\n impulseResponseResults['component']['background_noise']['y_background'] = impulseResponseResults['background_noise']['y_background'].map(value => value/linearScalePowerAttenuation);\n }\n impulseResponseResults['system']['attenuatorGainDB'] = this.systemAttenuatorGainDB;\n impulseResponseResults['component']['attenuatorGainDB'] = this.componentAttenuatorGainDB;\n impulseResponseResults['system']['fMaxHz'] = this.systemFMaxHz;\n impulseResponseResults['component']['fMaxHz'] = this.componentFMaxHz;\n impulseResponseResults['L_new_n'] = this.L_new_n;\n impulseResponseResults['fs2'] = this.fs2;\n\n if (componentIR != null) {\n // I corrected microphone/loudpeaker IR scale in easyeyes,\n // but since we write microphone IR to firestore here\n // we need to correct microphone IR here\n let correctGain =\n Math.round((volumeResults.parameters.gainDBSPL - this.componentGainDBSPL) * 10) / 10;\n\n let IrFreq = impulseResponseResults?.component.ir.Freq.map(freq => Math.round(freq));\n let IrGain = impulseResponseResults?.component?.ir.Gain;\n const IrGainAt1000Hz = IrGain[IrFreq.findIndex(freq => freq === 1000)];\n const difference = Math.round(10 * (IrGainAt1000Hz - correctGain)) / 10;\n IrGain = IrGain.map(gain => gain - difference);\n micInfo['mls'] = Number(this.SDofFilteredRange['mls']);\n micInfo['systemCorrectionSD'] = Number(this.SDofFilteredRange['system']);\n micInfo['componentCorrectionSD'] = Number(this.SDofFilteredRange['component']);\n console.log(typeof micInfo['componentCorrectionSD']);\n // const id = await this.writeIsSmartPhoneToFirestore(ID, isSmartPhone, OEM);\n // await this.writeMicrophoneInfoToFirestore(ID, micInfo, OEM, id);\n // await this.writeFrqGainToFirestore(ID, IrFreq, IrGain, OEM, id);\n // micInfo['gainDBSPL'] = impulseResponseResults.component.gainDBSPL;\n // await this.writeGainat1000HzToFirestore(ID, micInfo['gainDBSPL'], OEM, id);\n // await this.writeGainat1000Hz(ID, micInfo['gainDBSPL'], OEM);\n }\n const total_results = {...volumeResults, ...impulseResponseResults};\n total_results['filteredMLSRange'] = this.filteredMLSRange;\n total_results['filteredMLSAttenuation'] = this.filteredMLSAttenuation;\n total_results['micInfo'] = micInfo;\n total_results['audioInfo'] = {};\n total_results['audioInfo']['sinkSampleRate'] = this.sinkSamplingRate;\n total_results['audioInfo']['sourceSampleRate'] = this.sourceSamplingRate;\n total_results['audioInfo']['bitsPerSample'] = this.sampleSize;\n const timeStampresult = [...this.timeStamp].join('\\n');\n total_results['timeStamps'] = timeStampresult;\n total_results['recordingChecks'] = this.recordingChecks;\n total_results['component']['phase'] = this.componentIRPhase;\n total_results['system']['phase'] = this.systemIRPhase;\n total_results['qualityMetrics'] = this.SDofFilteredRange;\n total_results['flags'] = this.flags;\n console.log('total results');\n console.log(total_results);\n console.log('Time Stamps');\n console.log(timeStampresult);\n\n resolve(total_results);\n });\n };\n}\n\n/* harmony default export */ __webpack_exports__[\"default\"] = (Combination);\n\n\n//# sourceURL=webpack://speakerCalibrator/./src/tasks/combination/combination.js?");
855
855
 
856
856
  /***/ }),
857
857
 
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "speaker-calibration",
3
- "version": "2.2.151",
3
+ "version": "2.2.153",
4
4
  "description": "Speaker calibration library for auditory testing",
5
5
  "main": "dist/main.js",
6
6
  "directories": {
@@ -31,6 +31,7 @@
31
31
  "license": "MIT",
32
32
  "repository": "git://github.com/EasyEyes/speaker-calibration.git",
33
33
  "dependencies": {
34
+ "@tensorflow/tfjs": "^4.17.0",
34
35
  "axios": "^0.27.2",
35
36
  "fftw-js": "^0.1.4",
36
37
  "firebase": "^9.23.0",
@@ -1,4 +1,7 @@
1
1
  import axios from 'axios';
2
+ import {
3
+ sleep
4
+ } from '../utils';
2
5
  /**
3
6
  *
4
7
  */
@@ -27,22 +30,59 @@ class PythonServerAPI {
27
30
  */
28
31
  getImpulseResponse = async ({
29
32
  mls,
30
- payload,
31
33
  sampleRate,
32
- P,
33
- numPeriods
34
+ numPeriods,
35
+ sig,
36
+ fs2,
37
+ L_new_n,
38
+ dL_n
34
39
  }) => {
35
40
  const task = 'impulse-response';
36
41
  let res = null;
37
42
 
38
- console.log({payload});
43
+ const data = JSON.stringify({
44
+ task,
45
+ 'sample-rate': sampleRate,
46
+ mls,
47
+ numPeriods,
48
+ sig,
49
+ fs2,
50
+ L_new_n,
51
+ dL_n
52
+ });
53
+
54
+ await axios({
55
+ method: 'post',
56
+ baseURL: PythonServerAPI.PYTHON_SERVER_URL,
57
+ url: `/task/${task}`,
58
+ headers: {
59
+ 'Content-Type': 'application/json',
60
+ },
61
+ data,
62
+ })
63
+ .then(response => {
64
+ res = response;
65
+ })
66
+ .catch(error => {
67
+ throw error;
68
+ });
69
+ return res.data[task];
70
+ };
71
+
72
+ getAutocorrelation = async ({
73
+ mls,
74
+ payload,
75
+ sampleRate,
76
+ numPeriods
77
+ }) => {
78
+ const task = 'autocorrelation';
79
+ let res = null;
39
80
 
40
81
  const data = JSON.stringify({
41
82
  task,
42
83
  payload,
43
84
  'sample-rate': sampleRate,
44
85
  mls,
45
- P,
46
86
  numPeriods,
47
87
  });
48
88
 
@@ -64,6 +104,43 @@ class PythonServerAPI {
64
104
  return res.data[task];
65
105
  };
66
106
 
107
+ getConvolution = async ({
108
+ mls,
109
+ inverse_response,
110
+ inverse_response_no_bandpass,
111
+ attenuatorGain_dB,
112
+ mls_amplitude
113
+ }) => {
114
+ const task = 'convolution';
115
+ let res = null;
116
+
117
+ const data = JSON.stringify({
118
+ task,
119
+ mls,
120
+ inverse_response,
121
+ inverse_response_no_bandpass,
122
+ attenuatorGain_dB,
123
+ mls_amplitude
124
+ });
125
+
126
+ await axios({
127
+ method: 'post',
128
+ baseURL: PythonServerAPI.PYTHON_SERVER_URL,
129
+ url: `/task/${task}`,
130
+ headers: {
131
+ 'Content-Type': 'application/json',
132
+ },
133
+ data,
134
+ })
135
+ .then(response => {
136
+ res = response;
137
+ })
138
+ .catch(error => {
139
+ throw error;
140
+ });
141
+ return res.data[task];
142
+ };
143
+
67
144
  getMLS = async ({length,amplitude}) => {
68
145
  const task = 'mls';
69
146
  let res = null;
@@ -92,6 +169,41 @@ class PythonServerAPI {
92
169
 
93
170
  return res.data[task];
94
171
  };
172
+
173
+ getMemory = async () => {
174
+ let res;
175
+ await axios({
176
+ method: 'post',
177
+ baseURL: PythonServerAPI.PYTHON_SERVER_URL,
178
+ url: `/memory`,
179
+ headers: {
180
+ 'Content-Type': 'application/json',
181
+ }
182
+ })
183
+ .then(response => {
184
+ console.log("memory used:", Math.round(response.data['memory']), "mb");
185
+ res = response.data['memory'];
186
+ })
187
+ .catch(error => {
188
+ throw error;
189
+ });
190
+ return res;
191
+ };
192
+
193
+
194
+ checkMemory = async () => {
195
+ console.log("wait for memory under 500 mb to continue calibration");
196
+ await this.getMemory();
197
+ // let memory = await this.getMemory();
198
+ // while (memory >= 500) {
199
+ // console.log("sleep 30s");
200
+ // await sleep(30);
201
+ // memory = await this.getMemory();
202
+ // }
203
+
204
+ };
205
+
206
+
95
207
  getMLSWithRetry = async ({length,amplitude}) => {
96
208
  let retryCount = 0;
97
209
  let response = null;
@@ -279,7 +391,6 @@ class PythonServerAPI {
279
391
  componentIRGains,
280
392
  iirLength,
281
393
  componentIRFreqs,
282
- num_periods,
283
394
  sampleRate,
284
395
  mlsAmplitude,
285
396
  irLength,
@@ -300,7 +411,6 @@ class PythonServerAPI {
300
411
  iirLength,
301
412
  componentIRGains,
302
413
  componentIRFreqs,
303
- num_periods,
304
414
  sampleRate,
305
415
  mlsAmplitude,
306
416
  irLength,
@@ -332,7 +442,6 @@ class PythonServerAPI {
332
442
  lowHz,
333
443
  highHz,
334
444
  iirLength,
335
- num_periods,
336
445
  sampleRate,
337
446
  mlsAmplitude,
338
447
  calibrateSoundBurstFilteredExtraDb
@@ -349,7 +458,6 @@ class PythonServerAPI {
349
458
  lowHz,
350
459
  iirLength,
351
460
  highHz,
352
- num_periods,
353
461
  sampleRate,
354
462
  mlsAmplitude,
355
463
  calibrateSoundBurstFilteredExtraDb
@@ -436,7 +544,6 @@ class PythonServerAPI {
436
544
  componentIRGains,
437
545
  iirLength,
438
546
  componentIRFreqs,
439
- num_periods,
440
547
  sampleRate,
441
548
  mlsAmplitude,
442
549
  irLength,
@@ -456,7 +563,6 @@ class PythonServerAPI {
456
563
  componentIRGains,
457
564
  iirLength,
458
565
  componentIRFreqs,
459
- num_periods,
460
566
  sampleRate,
461
567
  mlsAmplitude,
462
568
  irLength,
@@ -487,7 +593,6 @@ class PythonServerAPI {
487
593
  lowHz,
488
594
  highHz,
489
595
  iirLength,
490
- num_periods,
491
596
  sampleRate,
492
597
  mlsAmplitude,
493
598
  calibrateSoundBurstFilteredExtraDb
@@ -503,7 +608,6 @@ class PythonServerAPI {
503
608
  lowHz,
504
609
  highHz,
505
610
  iirLength,
506
- num_periods,
507
611
  sampleRate,
508
612
  mlsAmplitude,
509
613
  calibrateSoundBurstFilteredExtraDb
@@ -1,5 +1,7 @@
1
1
  import AudioCalibrator from '../audioCalibrator';
2
2
 
3
+ // import * as tf from '../../../node_modules/@tensorflow/tfjs';
4
+
3
5
  import {
4
6
  sleep,
5
7
  csvToArray,
@@ -284,6 +286,7 @@ class Combination extends AudioCalibrator {
284
286
  componentAttentuatorGainDB = 0;
285
287
  componentFMaxHz = 0;
286
288
 
289
+ dL_n;
287
290
  L_new_n;
288
291
  fs2;
289
292
 
@@ -361,7 +364,7 @@ class Combination extends AudioCalibrator {
361
364
  this.status =
362
365
  `All Hz Calibration: computing the IIR...`.toString() + this.generateTemplate().toString();
363
366
  this.emit('update', {message: this.status});
364
- return this.pyServerAPI
367
+ return await this.pyServerAPI
365
368
  .getSystemInverseImpulseResponseWithRetry({
366
369
  payload: filteredComputedIRs.slice(0, this.numCaptures),
367
370
  mls,
@@ -373,7 +376,7 @@ class Combination extends AudioCalibrator {
373
376
  mlsAmplitude: Math.pow(10, this.power_dB / 20),
374
377
  calibrateSoundBurstFilteredExtraDb: this._calibrateSoundBurstFilteredExtraDb,
375
378
  })
376
- .then(res => {
379
+ .then(async res => {
377
380
  this.stepNum += 1;
378
381
  console.log('got impulse response ' + this.stepNum);
379
382
  this.incrementStatusBar();
@@ -382,23 +385,28 @@ class Combination extends AudioCalibrator {
382
385
  this.generateTemplate().toString();
383
386
  this.emit('update', {message: this.status});
384
387
  this.systemInvertedImpulseResponse = res['iir'];
385
- this.systemIR = res['ir'];
386
- this.systemConvolution = res['convolution'];
388
+ this.systemIR = res['ir'];
387
389
  this.systemInvertedImpulseResponseNoBandpass = res['iirNoBandpass'];
388
390
  this.systemAttenuatorGainDB = res['attenuatorGain_dB'];
389
391
  this.systemFMaxHz = res['fMaxHz'];
390
- this.systemConvolutionNoBandpass = res['convolutionNoBandpass'];
392
+ await this.pyServerAPI.checkMemory();
393
+ await this.pyServerAPI.getConvolution({
394
+ mls,
395
+ inverse_response: this.systemInvertedImpulseResponse,
396
+ inverse_response_no_bandpass: this.systemInvertedImpulseResponseNoBandpass,
397
+ attenuatorGain_dB: this.systemAttenuatorGainDB ,
398
+ mls_amplitude: Math.pow(10, this.power_dB / 20)
399
+ }).then(result => {
400
+ console.log(result);
401
+ this.systemConvolution = result['convolution'];
402
+ this.systemConvolutionNoBandpass = result['convolution_no_bandpass'];
403
+ });
404
+
391
405
 
392
406
  // attenuate the system convolution if the amplitude is greater than this.calibrateSoundLimit
393
407
  // find max of absolute value of system convolution
394
408
 
395
- const max = Math.max(...this.systemConvolution.map(Math.abs));
396
- // if (max > this.calibrateSoundLimit) {
397
- // const gain = this.calibrateSoundLimit / max;
398
- // // apply gain to system convolution
399
- // this.systemConvolution = this.systemConvolution.map(value => value * gain);
400
- // this.filteredMLSAttenuation.system = gain;
401
- // }
409
+ const max = findMaxValue(this.systemConvolution);
402
410
  this.filteredMLSAttenuation.system =
403
411
  this.systemConvolution.reduce((a, b) => a + b**2, 0) / this.systemConvolution.length;
404
412
  this.filteredMLSAttenuation.maxAbsSystem = max;
@@ -444,14 +452,13 @@ class Combination extends AudioCalibrator {
444
452
  iirLength,
445
453
  componentIRGains,
446
454
  componentIRFreqs,
447
- num_periods,
448
455
  sampleRate: this.sourceSamplingRate || 96000,
449
456
  mlsAmplitude: Math.pow(10, this.power_dB / 20),
450
457
  irLength,
451
458
  calibrateSoundSmoothOctaves: this._calibrateSoundSmoothOctaves,
452
459
  calibrateSoundBurstFilteredExtraDb: this._calibrateSoundBurstFilteredExtraDb,
453
460
  })
454
- .then(res => {
461
+ .then(async res => {
455
462
  this.stepNum += 1;
456
463
  console.log('got impulse response ' + this.stepNum);
457
464
  this.incrementStatusBar();
@@ -460,22 +467,31 @@ class Combination extends AudioCalibrator {
460
467
  this.generateTemplate().toString();
461
468
  this.emit('update', {message: this.status});
462
469
  this.componentInvertedImpulseResponse = res['iir'];
470
+ this.componentInvertedImpulseResponseNoBandpass = res['iirNoBandpass'];
463
471
  this.componentIR['Gain'] = res['ir'];
464
472
  this.componentIR['Freq'] = res['frequencies'];
465
473
  this.componentIRPhase = res['component_angle'];
466
474
  this.systemIRPhase = res['system_angle'];
467
475
  this.componentIROrigin['Freq'] = res['frequencies'];
468
476
  this.componentIROrigin['Gain'] = res['irOrigin'];
469
- this.componentConvolution = res['convolution'];
470
- this.componentConvolutionNoBandpass = res['convolutionNoBandpass'];
471
- this.componentInvertedImpulseResponseNoBandpass = res['iirNoBandpass'];
472
477
  this.componentIRInTimeDomain = res['irTime'];
473
478
  this.componentAttenuatorGainDB = res['attenuatorGain_dB'];
474
479
  this.componentFMaxHz = res['fMaxHz'];
475
-
480
+ await this.pyServerAPI.checkMemory();
481
+ await this.pyServerAPI.getConvolution({
482
+ mls,
483
+ inverse_response: this.componentInvertedImpulseResponse,
484
+ inverse_response_no_bandpass: this.componentInvertedImpulseResponseNoBandpass,
485
+ attenuatorGain_dB: this.componentAttenuatorGainDB ,
486
+ mls_amplitude: Math.pow(10, this.power_dB / 20)
487
+ }).then(result => {
488
+ console.log(result);
489
+ this.componentConvolution = result['convolution'];
490
+ this.componentConvolutionNoBandpass = result['convolution_no_bandpass'];
491
+ });
476
492
  // attenuate the component convolution if the amplitude is greater than this.calibrateSoundLimit
477
493
  // find max of absolute value of component convolution
478
- const max = Math.max(...this.componentConvolution.map(Math.abs));
494
+ const max = findMaxValue(this.componentConvolution);
479
495
  // if (max > this.calibrateSoundLimit) {
480
496
  // const gain = this.calibrateSoundLimit / max;
481
497
  // // apply gain to component convolution
@@ -555,42 +571,68 @@ class Combination extends AudioCalibrator {
555
571
  burstSec: this.desired_time_per_mls,
556
572
  repeats: this.numMLSPerCapture,
557
573
  })
558
- .then(result => {
574
+ .then(async result => {
559
575
  if (result) {
576
+ console.log("JS used memory:", performance.memory.usedJSHeapSize/1024/1024, "mb");
560
577
  if (result['sd'] < this._calibrateSoundPowerDbSDToleratedDb) {
561
- this.recordingChecks['unfiltered'].push(result);
562
- this.impulseResponses.push(
563
- this.pyServerAPI
564
- .getImpulseResponse({
565
- sampleRate: this.sourceSamplingRate || 96000,
566
- payload,
567
- mls,
568
- P: this.#P, //get rid of this
569
- numPeriods: this.numMLSPerCapture,
570
- })
571
- .then(res => {
572
- if (this.numSuccessfulCaptured < this.numCaptures) {
573
- this.numSuccessfulCaptured += 1;
574
- console.log('num succ capt: ' + this.numSuccessfulCaptured);
575
- this.stepNum += 1;
576
- console.log('got impulse response ' + this.stepNum);
577
- this.incrementStatusBar();
578
- this.status =
579
- `All Hz Calibration: ${this.numSuccessfulCaptured}/${this.numCaptures} IRs computed...`.toString() +
580
- this.generateTemplate().toString();
581
- this.emit('update', {
582
- message: this.status,
583
- });
584
- this.autocorrelations.push(res['autocorrelation']);
585
- this.L_new_n = res['L_new_n'];
586
- this.fs2 = res['fs2'];
587
- return res['ir'];
588
- }
589
- })
590
- .catch(err => {
591
- console.error(err);
592
- })
593
- );
578
+ this.recordingChecks['unfiltered'].push(result);
579
+ await this.pyServerAPI.checkMemory();
580
+ // let start = new Date().getTime() / 1000;
581
+ // const payloadT = tf.tensor1d(payload);
582
+ // payloadT.print();
583
+ // const xfft = payloadT.rfft(); // tf.spe
584
+ // xfft.array().then(array => {
585
+ // console.log("fft:", array);
586
+ // let setItem = new Set(array);
587
+ // console.log("all zero", setItem.size === 1 && setItem.has(0));
588
+ // });
589
+ // console.log("dimention:", xfft.shape);
590
+ // let end = new Date().getTime() / 1000;
591
+ // console.log("Time taken:", end - start, "seconds");
592
+ await this.pyServerAPI.getAutocorrelation({
593
+ sampleRate: this.sourceSamplingRate || 96000,
594
+ payload,
595
+ mls,
596
+ numPeriods: this.numMLSPerCapture,
597
+ }).then(async res=>{
598
+ this.autocorrelations.push(res['autocorrelation']);
599
+ this.fs2 = res['fs2'];
600
+ this.L_new_n = res['L_new_n'];
601
+ this.dL_n = res['dL_n'];
602
+ await this.pyServerAPI.checkMemory();
603
+ this.impulseResponses.push(
604
+ await this.pyServerAPI
605
+ .getImpulseResponse({
606
+ mls,
607
+ sampleRate: this.sourceSamplingRate || 96000,
608
+ numPeriods: this.numMLSPerCapture,
609
+ sig: payload,
610
+ fs2: this.fs2,
611
+ L_new_n: this.L_new_n,
612
+ dL_n: this.dL_n
613
+ })
614
+ .then(res => {
615
+ if (this.numSuccessfulCaptured < this.numCaptures) {
616
+ this.numSuccessfulCaptured += 1;
617
+ console.log('num succ capt: ' + this.numSuccessfulCaptured);
618
+ this.stepNum += 1;
619
+ console.log('got impulse response ' + this.stepNum);
620
+ this.incrementStatusBar();
621
+ this.status =
622
+ `All Hz Calibration: ${this.numSuccessfulCaptured}/${this.numCaptures} IRs computed...`.toString() +
623
+ this.generateTemplate().toString();
624
+ this.emit('update', {
625
+ message: this.status,
626
+ });
627
+ return res['ir'];
628
+ }
629
+ })
630
+ .catch(err => {
631
+ console.error(err);
632
+ })
633
+ );
634
+ })
635
+
594
636
  } else if (result['sd'] > this._calibrateSoundPowerDbSDToleratedDb) {
595
637
  this.clearLastUnfilteredRecordedSignals();
596
638
  console.log('unfiltered rec', this.getAllUnfilteredRecordedSignals.length);
@@ -1792,6 +1834,7 @@ class Combination extends AudioCalibrator {
1792
1834
  * @example
1793
1835
  */
1794
1836
  startCalibrationImpulseResponse = async stream => {
1837
+ console.log("JS used memory:", performance.memory.usedJSHeapSize/1024/1024, "mb");
1795
1838
  let desired_time = this.desired_time_per_mls;
1796
1839
  let checkRec = 'allhz';
1797
1840
 
@@ -1865,6 +1908,7 @@ class Combination extends AudioCalibrator {
1865
1908
  // so let's send all the IRs to the server to be converted to a single IIR
1866
1909
  if (this.isCalibrating) return null;
1867
1910
  await this.sendSystemImpulseResponsesToServerForProcessing();
1911
+ await this.pyServerAPI.checkMemory();
1868
1912
  if (this.isCalibrating) return null;
1869
1913
  await this.sendComponentImpulseResponsesToServerForProcessing();
1870
1914
 
@@ -2190,6 +2234,7 @@ class Combination extends AudioCalibrator {
2190
2234
  };
2191
2235
 
2192
2236
  startCalibrationVolume = async (stream, gainValues, lCalib, componentGainDBSPL) => {
2237
+ console.log("JS used memory:", performance.memory.usedJSHeapSize/1024/1024, "mb");
2193
2238
  if (this.isCalibrating) return null;
2194
2239
  const trialIterations = gainValues.length;
2195
2240
  this.status_denominator += trialIterations;
@@ -2199,7 +2244,6 @@ class Combination extends AudioCalibrator {
2199
2244
  const outDBSPLValues = [];
2200
2245
  const outDBSPL1000Values = [];
2201
2246
  let checkRec = "loudest";
2202
-
2203
2247
  // do one calibration that will be discarded
2204
2248
  const soundLevelToDiscard = -60;
2205
2249
  const gainToDiscard = Math.pow(10, soundLevelToDiscard / 20);
@@ -2739,6 +2783,7 @@ class Combination extends AudioCalibrator {
2739
2783
  resolve('restart');
2740
2784
  });
2741
2785
  }
2786
+ await this.pyServerAPI.checkMemory();
2742
2787
  let volumeResults = await this.startCalibrationVolume(
2743
2788
  stream,
2744
2789
  gainValues,
@@ -2750,8 +2795,6 @@ class Combination extends AudioCalibrator {
2750
2795
  this.T = volumeResults["parameters"]["T"];
2751
2796
  this.gainDBSPL = volumeResults["parameters"]["gainDBSPL"];
2752
2797
 
2753
- // console.log("VOLUME RESULTS");
2754
- // console.log(volumeResults);
2755
2798
  let impulseResponseResults = await this.startCalibrationImpulseResponse(stream);
2756
2799
  if (!impulseResponseResults) return;
2757
2800
  impulseResponseResults['background_noise'] = this.background_noise;