mirror of
https://github.com/jambonz/jambonz-feature-server.git
synced 2026-02-11 17:01:30 +00:00
Compare commits
9 Commits
v0.7.5-rc2
...
v0.7.5-rc6
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
de9f2ce5ca | ||
|
|
36c97e9562 | ||
|
|
13ea559cb1 | ||
|
|
698d12a95f | ||
|
|
359cb82d80 | ||
|
|
29dec24095 | ||
|
|
6330b0d443 | ||
|
|
24a0bc547f | ||
|
|
db5486de27 |
@@ -1,10 +1,10 @@
|
|||||||
FROM node:17.7.1-slim
|
FROM node:slim
|
||||||
WORKDIR /opt/app/
|
WORKDIR /opt/app/
|
||||||
COPY package.json ./
|
COPY package.json package-lock.json ./
|
||||||
RUN npm install
|
RUN npm ci
|
||||||
RUN npm prune
|
RUN npm prune
|
||||||
COPY . /opt/app
|
COPY . /opt/app
|
||||||
ARG NODE_ENV
|
ARG NODE_ENV
|
||||||
ENV NODE_ENV $NODE_ENV
|
ENV NODE_ENV $NODE_ENV
|
||||||
|
|
||||||
CMD [ "npm", "start" ]
|
CMD [ "npm", "start" ]
|
||||||
|
|||||||
@@ -331,7 +331,7 @@ class CallSession extends Emitter {
|
|||||||
speech_credential_sid: credential.speech_credential_sid,
|
speech_credential_sid: credential.speech_credential_sid,
|
||||||
accessKeyId: credential.access_key_id,
|
accessKeyId: credential.access_key_id,
|
||||||
secretAccessKey: credential.secret_access_key,
|
secretAccessKey: credential.secret_access_key,
|
||||||
region: process.env.AWS_REGION || credential.aws_region
|
region: credential.aws_region || process.env.AWS_REGION
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
else if ('microsoft' === vendor) {
|
else if ('microsoft' === vendor) {
|
||||||
|
|||||||
@@ -362,7 +362,9 @@ class TaskGather extends Task {
|
|||||||
if ('microsoft' === this.vendor) {
|
if ('microsoft' === this.vendor) {
|
||||||
const final = evt.RecognitionStatus === 'Success';
|
const final = evt.RecognitionStatus === 'Success';
|
||||||
if (final) {
|
if (final) {
|
||||||
const nbest = evt.NBest.sort((a, b) => b.Confidence - a.Confidence);
|
// don't sort based on confidence: https://github.com/Azure-Samples/cognitive-services-speech-sdk/issues/1463
|
||||||
|
//const nbest = evt.NBest.sort((a, b) => b.Confidence - a.Confidence);
|
||||||
|
const nbest = evt.NBest;
|
||||||
evt = {
|
evt = {
|
||||||
is_final: true,
|
is_final: true,
|
||||||
alternatives: [
|
alternatives: [
|
||||||
@@ -385,7 +387,7 @@ class TaskGather extends Task {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (evt.is_final) {
|
if (evt.is_final) {
|
||||||
if (evt.alternatives[0].transcript === '') {
|
if (evt.alternatives[0].transcript === '' && !this.callSession.callGone && !this.killed) {
|
||||||
this.logger.info({evt}, 'TaskGather:_onTranscription - got empty transcript, listen again');
|
this.logger.info({evt}, 'TaskGather:_onTranscription - got empty transcript, listen again');
|
||||||
return this._startTranscribing(ep);
|
return this._startTranscribing(ep);
|
||||||
}
|
}
|
||||||
@@ -433,7 +435,10 @@ class TaskGather extends Task {
|
|||||||
}
|
}
|
||||||
|
|
||||||
_onNoSpeechDetected(cs, ep) {
|
_onNoSpeechDetected(cs, ep) {
|
||||||
this._resolve('timeout');
|
if (!this.callSession.callGone && !this.killed) {
|
||||||
|
this.logger.debug('TaskGather:_onNoSpeechDetected - listen again');
|
||||||
|
return this._startTranscribing(ep);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async _resolve(reason, evt) {
|
async _resolve(reason, evt) {
|
||||||
|
|||||||
@@ -51,55 +51,64 @@ class TaskSay extends Task {
|
|||||||
alert_type: AlertType.TTS_NOT_PROVISIONED,
|
alert_type: AlertType.TTS_NOT_PROVISIONED,
|
||||||
vendor
|
vendor
|
||||||
}).catch((err) => this.logger.info({err}, 'Error generating alert for no tts'));
|
}).catch((err) => this.logger.info({err}, 'Error generating alert for no tts'));
|
||||||
|
this.notifyError(`No speech credentials have been provisioned for ${vendor}`);
|
||||||
throw new Error('no provisioned speech credentials for TTS');
|
throw new Error('no provisioned speech credentials for TTS');
|
||||||
}
|
}
|
||||||
// synthesize all of the text elements
|
// synthesize all of the text elements
|
||||||
let lastUpdated = false;
|
let lastUpdated = false;
|
||||||
|
|
||||||
/* otel: trace time for tts */
|
/* produce an audio segment from the provided text */
|
||||||
const {span} = this.startChildSpan('tts-generation', {
|
const generateAudio = async(text) => {
|
||||||
'tts.vendor': vendor,
|
|
||||||
'tts.language': language,
|
|
||||||
'tts.voice': voice
|
|
||||||
});
|
|
||||||
this.ttsSpan = span;
|
|
||||||
|
|
||||||
const filepath = (await Promise.all(this.text.map(async(text) => {
|
|
||||||
if (this.killed) return;
|
if (this.killed) return;
|
||||||
if (text.startsWith('silence_stream://')) return text;
|
if (text.startsWith('silence_stream://')) return text;
|
||||||
const {filePath, servedFromCache} = await synthAudio(stats, {
|
|
||||||
text,
|
/* otel: trace time for tts */
|
||||||
vendor,
|
const {span} = this.startChildSpan('tts-generation', {
|
||||||
language,
|
'tts.vendor': vendor,
|
||||||
voice,
|
'tts.language': language,
|
||||||
engine,
|
'tts.voice': voice
|
||||||
salt,
|
});
|
||||||
credentials
|
try {
|
||||||
}).catch((err) => {
|
const {filePath, servedFromCache} = await synthAudio(stats, {
|
||||||
this.logger.info(err, 'Error synthesizing tts');
|
text,
|
||||||
|
vendor,
|
||||||
|
language,
|
||||||
|
voice,
|
||||||
|
engine,
|
||||||
|
salt,
|
||||||
|
credentials
|
||||||
|
});
|
||||||
|
this.logger.debug(`file ${filePath}, served from cache ${servedFromCache}`);
|
||||||
|
if (filePath) cs.trackTmpFile(filePath);
|
||||||
|
if (!servedFromCache && !lastUpdated) {
|
||||||
|
lastUpdated = true;
|
||||||
|
updateSpeechCredentialLastUsed(credentials.speech_credential_sid)
|
||||||
|
.catch(() => {/*already logged error */});
|
||||||
|
}
|
||||||
|
span.setAttributes({'tts.cached': servedFromCache});
|
||||||
|
span.end();
|
||||||
|
return filePath;
|
||||||
|
} catch (err) {
|
||||||
|
this.logger.info({err}, 'Error synthesizing tts');
|
||||||
|
span.end();
|
||||||
writeAlerts({
|
writeAlerts({
|
||||||
account_sid: cs.accountSid,
|
account_sid: cs.accountSid,
|
||||||
alert_type: AlertType.TTS_NOT_PROVISIONED,
|
alert_type: AlertType.TTS_NOT_PROVISIONED,
|
||||||
vendor,
|
vendor,
|
||||||
detail: err.message
|
detail: err.message
|
||||||
});
|
}).catch((err) => this.logger.info({err}, 'Error generating alert for tts failure'));
|
||||||
}).catch((err) => this.logger.info({err}, 'Error generating alert for tts failure'));
|
this.notifyError(err.message || err);
|
||||||
this.logger.debug(`file ${filePath}, served from cache ${servedFromCache}`);
|
return;
|
||||||
if (filePath) cs.trackTmpFile(filePath);
|
|
||||||
if (!servedFromCache && !lastUpdated) {
|
|
||||||
lastUpdated = true;
|
|
||||||
updateSpeechCredentialLastUsed(credentials.speech_credential_sid)
|
|
||||||
.catch(() => {/*already logged error */});
|
|
||||||
}
|
}
|
||||||
this.ttsSpan.setAttributes({'tts.cached': servedFromCache});
|
};
|
||||||
return filePath;
|
|
||||||
}))).filter((fp) => fp && fp.length);
|
const arr = this.text.map((t) => generateAudio(t));
|
||||||
this.ttsSpan?.end();
|
const filepath = (await Promise.all(arr)).filter((fp) => fp && fp.length);
|
||||||
this.logger.debug({filepath}, 'synthesized files for tts');
|
this.logger.debug({filepath}, 'synthesized files for tts');
|
||||||
|
|
||||||
while (!this.killed && (this.loop === 'forever' || this.loop--) && this.ep?.connected) {
|
while (!this.killed && (this.loop === 'forever' || this.loop--) && this.ep?.connected) {
|
||||||
let segment = 0;
|
let segment = 0;
|
||||||
do {
|
while (!this.killed && segment < filepath.length) {
|
||||||
if (cs.isInConference) {
|
if (cs.isInConference) {
|
||||||
const {memberId, confName, confUuid} = cs;
|
const {memberId, confName, confUuid} = cs;
|
||||||
await this.playToConfMember(this.ep, memberId, confName, confUuid, filepath[segment]);
|
await this.playToConfMember(this.ep, memberId, confName, confUuid, filepath[segment]);
|
||||||
@@ -109,10 +118,10 @@ class TaskSay extends Task {
|
|||||||
await ep.play(filepath[segment]);
|
await ep.play(filepath[segment]);
|
||||||
this.logger.debug(`Say:exec completed play file ${filepath[segment]}`);
|
this.logger.debug(`Say:exec completed play file ${filepath[segment]}`);
|
||||||
}
|
}
|
||||||
} while (!this.killed && ++segment < filepath.length);
|
segment++;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this.ttsSpan?.end();
|
|
||||||
this.logger.info(err, 'TaskSay:exec error');
|
this.logger.info(err, 'TaskSay:exec error');
|
||||||
}
|
}
|
||||||
this.emit('playDone');
|
this.emit('playDone');
|
||||||
|
|||||||
@@ -137,6 +137,12 @@ class Task extends Emitter {
|
|||||||
return this.callSession.normalizeUrl(url, method, auth);
|
return this.callSession.normalizeUrl(url, method, auth);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
notifyError(errMsg) {
|
||||||
|
const params = {error: errMsg, verb: this.name};
|
||||||
|
this.cs.requestor.request('jambonz:error', '/error', params)
|
||||||
|
.catch((err) => this.logger.info({err}, 'Task:notifyError error sending error'));
|
||||||
|
}
|
||||||
|
|
||||||
async performAction(results, expectResponse = true) {
|
async performAction(results, expectResponse = true) {
|
||||||
if (this.actionHook) {
|
if (this.actionHook) {
|
||||||
const params = results ? Object.assign(results, this.cs.callInfo.toJSON()) : this.cs.callInfo.toJSON();
|
const params = results ? Object.assign(results, this.cs.callInfo.toJSON()) : this.cs.callInfo.toJSON();
|
||||||
|
|||||||
@@ -253,6 +253,11 @@ class TaskTranscribe extends Task {
|
|||||||
evt = newEvent;
|
evt = newEvent;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (evt.alternatives[0].transcript === '' && !cs.callGone && !this.killed) {
|
||||||
|
this.logger.info({evt}, 'TaskGather:_onTranscription - got empty transcript, listen again');
|
||||||
|
return this._transcribe(ep);
|
||||||
|
}
|
||||||
|
|
||||||
if (this.transcriptionHook) {
|
if (this.transcriptionHook) {
|
||||||
const b3 = this.getTracingPropagation();
|
const b3 = this.getTracingPropagation();
|
||||||
const httpHeaders = b3 && {b3};
|
const httpHeaders = b3 && {b3};
|
||||||
|
|||||||
@@ -50,7 +50,11 @@ class HttpRequestor extends BaseRequestor {
|
|||||||
* @param {object} [params] - request parameters
|
* @param {object} [params] - request parameters
|
||||||
*/
|
*/
|
||||||
async request(type, hook, params, httpHeaders = {}) {
|
async request(type, hook, params, httpHeaders = {}) {
|
||||||
|
/* jambonz:error only sent over ws */
|
||||||
|
if (type === 'jambonz:error') return;
|
||||||
|
|
||||||
assert(HookMsgTypes.includes(type));
|
assert(HookMsgTypes.includes(type));
|
||||||
|
|
||||||
const payload = params ? snakeCaseKeys(params, ['customerData', 'sip']) : null;
|
const payload = params ? snakeCaseKeys(params, ['customerData', 'sip']) : null;
|
||||||
const url = hook.url || hook;
|
const url = hook.url || hook;
|
||||||
const method = hook.method || 'POST';
|
const method = hook.method || 'POST';
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
const Mrf = require('drachtio-fsmrf');
|
const Mrf = require('drachtio-fsmrf');
|
||||||
const ip = require('ip');
|
const ip = require('ip');
|
||||||
const localIp = ip.address();
|
|
||||||
const PORT = process.env.HTTP_PORT || 3000;
|
const PORT = process.env.HTTP_PORT || 3000;
|
||||||
const assert = require('assert');
|
const assert = require('assert');
|
||||||
|
|
||||||
@@ -167,6 +166,13 @@ function installSrfLocals(srf, logger) {
|
|||||||
commitInterval: 'test' === process.env.NODE_ENV ? 7 : 20
|
commitInterval: 'test' === process.env.NODE_ENV ? 7 : 20
|
||||||
});
|
});
|
||||||
|
|
||||||
|
let localIp;
|
||||||
|
try {
|
||||||
|
localIp = ip.address();
|
||||||
|
} catch (err) {
|
||||||
|
logger.error({err}, 'installSrfLocals - error detecting local ipv4 address');
|
||||||
|
}
|
||||||
|
|
||||||
srf.locals = {...srf.locals,
|
srf.locals = {...srf.locals,
|
||||||
dbHelpers: {
|
dbHelpers: {
|
||||||
client,
|
client,
|
||||||
@@ -201,8 +207,6 @@ function installSrfLocals(srf, logger) {
|
|||||||
getListPosition
|
getListPosition
|
||||||
},
|
},
|
||||||
parentLogger: logger,
|
parentLogger: logger,
|
||||||
ipv4: localIp,
|
|
||||||
serviceUrl: `http://${localIp}:${PORT}`,
|
|
||||||
getSBC,
|
getSBC,
|
||||||
getSmpp: () => {
|
getSmpp: () => {
|
||||||
return process.env.SMPP_URL;
|
return process.env.SMPP_URL;
|
||||||
@@ -213,6 +217,11 @@ function installSrfLocals(srf, logger) {
|
|||||||
writeAlerts,
|
writeAlerts,
|
||||||
AlertType
|
AlertType
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (localIp) {
|
||||||
|
srf.locals.ipv4 = localIp;
|
||||||
|
srf.locals.serviceUrl = `http://${localIp}:${PORT}`;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = installSrfLocals;
|
module.exports = installSrfLocals;
|
||||||
|
|||||||
@@ -105,6 +105,7 @@ class WsRequestor extends BaseRequestor {
|
|||||||
/* save the message info for reply */
|
/* save the message info for reply */
|
||||||
const startAt = process.hrtime();
|
const startAt = process.hrtime();
|
||||||
this.messagesInFlight.set(msgid, {
|
this.messagesInFlight.set(msgid, {
|
||||||
|
timer,
|
||||||
success: (response) => {
|
success: (response) => {
|
||||||
clearTimeout(timer);
|
clearTimeout(timer);
|
||||||
const rtt = this._roundTrip(startAt);
|
const rtt = this._roundTrip(startAt);
|
||||||
@@ -135,6 +136,8 @@ class WsRequestor extends BaseRequestor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for (const [msgid, obj] of this.messagesInFlight) {
|
for (const [msgid, obj] of this.messagesInFlight) {
|
||||||
|
const {timer} = obj;
|
||||||
|
clearTimeout(timer);
|
||||||
obj.failure(`abandoning msgid ${msgid} since we have closed the socket`);
|
obj.failure(`abandoning msgid ${msgid} since we have closed the socket`);
|
||||||
}
|
}
|
||||||
this.messagesInFlight.clear();
|
this.messagesInFlight.clear();
|
||||||
|
|||||||
Reference in New Issue
Block a user