mirror of
https://github.com/jambonz/jambonz-feature-server.git
synced 2026-02-10 08:21:33 +00:00
Compare commits
96 Commits
v0.7.3-rc3
...
v0.7.5-1
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
3ab4f3fdf9 | ||
|
|
a92e9d0f3e | ||
|
|
f51211b407 | ||
|
|
7f0e373e5f | ||
|
|
c3e5ffa52d | ||
|
|
0ee13fb794 | ||
|
|
4e84098036 | ||
|
|
6d34850dc6 | ||
|
|
76ff1835a6 | ||
|
|
a4e358596e | ||
|
|
c412554c6b | ||
|
|
34fe22f6e1 | ||
|
|
182ad8c716 | ||
|
|
036accab44 | ||
|
|
b37881a059 | ||
|
|
258e4b5434 | ||
|
|
aa4d72c80a | ||
|
|
5c38ace5ba | ||
|
|
dea58c2605 | ||
|
|
eb0f55e0e3 | ||
|
|
944b8a29ca | ||
|
|
daa02ac55a | ||
|
|
5134d5dbc6 | ||
|
|
a755e25568 | ||
|
|
13549286db | ||
|
|
72aaf80335 | ||
|
|
af33089a8a | ||
|
|
85d86cfdc3 | ||
|
|
de9f2ce5ca | ||
|
|
36c97e9562 | ||
|
|
13ea559cb1 | ||
|
|
698d12a95f | ||
|
|
359cb82d80 | ||
|
|
29dec24095 | ||
|
|
6330b0d443 | ||
|
|
24a0bc547f | ||
|
|
db5486de27 | ||
|
|
41d6c74c8e | ||
|
|
92ca40c9b3 | ||
|
|
3fa913215f | ||
|
|
0b132411c1 | ||
|
|
077d34dc9e | ||
|
|
49a75a3e3a | ||
|
|
6f214a66e8 | ||
|
|
3456c51118 | ||
|
|
13c38a9875 | ||
|
|
4f87cf9b38 | ||
|
|
bf21a1f9a4 | ||
|
|
81f6163aca | ||
|
|
547ca0281f | ||
|
|
3281a213c8 | ||
|
|
4f2fc70383 | ||
|
|
f72e8e654c | ||
|
|
cf2100f925 | ||
|
|
5a584f50da | ||
|
|
befe910503 | ||
|
|
040ec0db9b | ||
|
|
8459376f88 | ||
|
|
775a317821 | ||
|
|
9004f654ff | ||
|
|
6163657845 | ||
|
|
398daa87d5 | ||
|
|
4f5ab7d146 | ||
|
|
70f7775893 | ||
|
|
a950f9f738 | ||
|
|
ff8d7f3648 | ||
|
|
6e4ae69cb7 | ||
|
|
23eae34888 | ||
|
|
aaf94006db | ||
|
|
86b030db93 | ||
|
|
6abfdafe05 | ||
|
|
f1f83598ca | ||
|
|
3dd703411c | ||
|
|
8c5cdd374b | ||
|
|
15d784a4b0 | ||
|
|
7188648d3b | ||
|
|
d00ea5c95f | ||
|
|
ddcbda988f | ||
|
|
ddf00c0ddf | ||
|
|
fd8df533ab | ||
|
|
4b1199242f | ||
|
|
72225791b9 | ||
|
|
172dc1aaa7 | ||
|
|
72b74de767 | ||
|
|
9908485eb8 | ||
|
|
fb25389cd1 | ||
|
|
f317fbaa45 | ||
|
|
3c5d392407 | ||
|
|
5bfc451c85 | ||
|
|
47478fd409 | ||
|
|
c16a2662f2 | ||
|
|
c1130adf03 | ||
|
|
f982f6c7d8 | ||
|
|
f20190b0fc | ||
|
|
74e85e1b16 | ||
|
|
63e9cb985e |
@@ -1,10 +1,10 @@
|
|||||||
FROM node:17.4-slim
|
FROM node:lts-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" ]
|
||||||
|
|||||||
@@ -2,6 +2,8 @@
|
|||||||
|
|
||||||
This application implements the core feature server of the jambones platform.
|
This application implements the core feature server of the jambones platform.
|
||||||
|
|
||||||
|
> Note: If you are a developer looking to work on the code please read our [how-to for that](./docs/contributing.md).
|
||||||
|
|
||||||
## Configuration
|
## Configuration
|
||||||
|
|
||||||
Configuration is provided via environment variables:
|
Configuration is provided via environment variables:
|
||||||
@@ -84,7 +86,5 @@ module.exports = {
|
|||||||
```
|
```
|
||||||
|
|
||||||
#### Running the test suite
|
#### Running the test suite
|
||||||
The test suite currently only consists of JSON-parsing unit tests. A full end-to-end sip test suite should be added.
|
|
||||||
```
|
Please [see this]](./docs/contributing.md#run-the-regression-test-suite).
|
||||||
npm test
|
|
||||||
```
|
|
||||||
19
app.js
19
app.js
@@ -11,6 +11,10 @@ assert.ok(process.env.JAMBONES_NETWORK_CIDR || process.env.K8S, 'missing JAMBONE
|
|||||||
|
|
||||||
const Srf = require('drachtio-srf');
|
const Srf = require('drachtio-srf');
|
||||||
const srf = new Srf();
|
const srf = new Srf();
|
||||||
|
const tracer = require('./tracer')(process.env.JAMBONES_OTEL_SERVICE_NAME || 'jambonz-feature-server');
|
||||||
|
const api = require('@opentelemetry/api');
|
||||||
|
srf.locals = {...srf.locals, otel: {tracer, api}};
|
||||||
|
|
||||||
const PORT = process.env.HTTP_PORT || 3000;
|
const PORT = process.env.HTTP_PORT || 3000;
|
||||||
const opts = {
|
const opts = {
|
||||||
timestamp: () => {return `, "time": "${new Date().toISOString()}"`;},
|
timestamp: () => {return `, "time": "${new Date().toISOString()}"`;},
|
||||||
@@ -23,6 +27,7 @@ installSrfLocals(srf, logger);
|
|||||||
|
|
||||||
const {
|
const {
|
||||||
initLocals,
|
initLocals,
|
||||||
|
createRootSpan,
|
||||||
getAccountDetails,
|
getAccountDetails,
|
||||||
normalizeNumbers,
|
normalizeNumbers,
|
||||||
retrieveApplication,
|
retrieveApplication,
|
||||||
@@ -62,6 +67,7 @@ if (process.env.NODE_ENV === 'test') {
|
|||||||
|
|
||||||
srf.use('invite', [
|
srf.use('invite', [
|
||||||
initLocals,
|
initLocals,
|
||||||
|
createRootSpan,
|
||||||
getAccountDetails,
|
getAccountDetails,
|
||||||
normalizeNumbers,
|
normalizeNumbers,
|
||||||
retrieveApplication,
|
retrieveApplication,
|
||||||
@@ -124,4 +130,17 @@ function handle(signal) {
|
|||||||
srf.locals.disabled = true;
|
srf.locals.disabled = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (process.env.JAMBONZ_CLEANUP_INTERVAL_MINS) {
|
||||||
|
const {clearFiles} = require('./lib/utils/cron-jobs');
|
||||||
|
|
||||||
|
/* cleanup orphaned files or channels every so often */
|
||||||
|
setInterval(async() => {
|
||||||
|
try {
|
||||||
|
await clearFiles();
|
||||||
|
} catch (err) {
|
||||||
|
logger.error({err}, 'app.js: error clearing files');
|
||||||
|
}
|
||||||
|
}, 1000 * 60 * (process.env.JAMBONZ_CLEANUP_INTERVAL_MINS || 60));
|
||||||
|
}
|
||||||
|
|
||||||
module.exports = {srf, logger, disconnect};
|
module.exports = {srf, logger, disconnect};
|
||||||
|
|||||||
123
docs/contributing.md
Normal file
123
docs/contributing.md
Normal file
@@ -0,0 +1,123 @@
|
|||||||
|
# Contributors are welcome!
|
||||||
|
|
||||||
|
So, you want to hack on jambonz? Maybe add some features, maybe help fix some bugs? Awesome, welcome aboard!
|
||||||
|
|
||||||
|
This brief document should get you started. Here you will find instructions showing how to set up your laptop to run the regression test suite (which you should always run before committing any changes), as well as some basic info on the structure of the code.
|
||||||
|
|
||||||
|
## Getting oriented
|
||||||
|
|
||||||
|
First of all, you are in the right place to begin hacking on jambonz. The jambonz-feature-server app is kinda the center of the universe for jambonz. Most of the core logic in jambonz is implemented here: things like the [webhook verbs](../lib/tasks), [session management](../lib/session), and the [client-side webhook implementation](../lib/utils/http-requestor.js). A common thing you might want to do, for instance, is to add support for an all-new verb, and this code base is where would do that.
|
||||||
|
|
||||||
|
This jambonz-feature-server app works together quite closely with a [drachtio server](https://github.com/drachtio/drachtio-server) and a Freeswitch. In fact, these three components are bundled together into a single VM/instance (or a Deployment, in Kubernetes) that we more generally refer to as "Feature Server". The Feature Server is a horizontally-scalable unit that is deployed behind the public-facing SBC elements of a jambonz cluster (the SBC is itself a separately scalable unit). The drachtio-server handles the SIP signaling, the Freeswitch handles media operations and speech vendor integration, and the jambonz-feature-server app orchestrates all of it via the use of [drachtio-srf](https://github.com/drachtio/drachtio-srf) and [drachtio-fsmrf](https://github.com/drachtio/drachtio-fsmrf).
|
||||||
|
|
||||||
|
## How to do things
|
||||||
|
|
||||||
|
First of all, please join our [slack channel](https://joinslack.jambonz.org) in order to coordinate with us on the work, i.e. to notify us of what you are doing and make sure that no one else is already working on the same thing.
|
||||||
|
|
||||||
|
To prepare to make changes, please fork the repo to your own Github account, make changes, test them on your own running jambonz cluster, then run the regression test suite and lint check before giving us a PR.
|
||||||
|
|
||||||
|
### lint
|
||||||
|
|
||||||
|
We have some opinionated conventions that you must follow - see our [eslintrc.json](../.eslintrc.json) for details. Make sure your code passes by running:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
npm run jslint
|
||||||
|
```
|
||||||
|
|
||||||
|
### test suite
|
||||||
|
|
||||||
|
#### Generate speech credentials and create run-tests.sh
|
||||||
|
|
||||||
|
The test suite also requires you to provide speech credentials for both GCP and AWS. You will want to create a new file named `run-tests.sh` in the project folder. Make the file executable and then copy in the text below, substituting your speech credentials where indicated:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
GCP_JSON_KEY='{"type":"service_account","project_id":"...etc"}' \
|
||||||
|
AWS_ACCESS_KEY_ID='your-aws-access-key-id' \
|
||||||
|
AWS_SECRET_ACCESS_KEY='your-aws-secret-access-key' \
|
||||||
|
AWS_REGION='us-east-1' \
|
||||||
|
JWT_SECRET='foobar' \
|
||||||
|
npm test
|
||||||
|
```
|
||||||
|
>> Note: The project's .gitignore file prevents this file from being sent to Github, so you do not need to worry about exposing your credentials. Just make sure you name if run-tests.sh and create it in the project folder
|
||||||
|
|
||||||
|
The GCP credential is the JSON service key in stringified format.
|
||||||
|
|
||||||
|
#### Install Docker
|
||||||
|
|
||||||
|
The test suite ralso equires [Docker](https://www.docker.com/) and docker-compose to be installed on your laptop. Docker is used to set up a network with all of the elements required to test the jambonz-feature-server in a black-box type of fashion.
|
||||||
|
|
||||||
|
Once you have docker installed, you can optionally make sure everything Docker-wise is working properly by running this command from the project folder:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
docker-compose -f test/docker-compose-testbed.yaml up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
This may take several minutes to complete, mainly because the mysql schema needs to be installed and seeded, but if successful the output should look like this:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
$ docker-compose -f test/docker-compose-testbed.yaml up -d
|
||||||
|
Creating network "test_fs" with driver "bridge"
|
||||||
|
Creating test_webhook-transcribe_1 ... done
|
||||||
|
Creating test_webhook-decline_1 ... done
|
||||||
|
Creating test_mysql_1 ... done
|
||||||
|
Creating test_docker-host_1 ... done
|
||||||
|
Creating test_webhook-gather_1 ... done
|
||||||
|
Creating test_webhook-say_1 ... done
|
||||||
|
Creating test_freeswitch_1 ... done
|
||||||
|
Creating test_influxdb_1 ... done
|
||||||
|
Creating test_redis_1 ... done
|
||||||
|
Creating test_drachtio_1 ... done
|
||||||
|
```
|
||||||
|
|
||||||
|
At that point, you can run `docker ps` to see all of the containers running
|
||||||
|
|
||||||
|
```bash
|
||||||
|
docker ps
|
||||||
|
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
|
||||||
|
abbc3594f390 drachtio/drachtio-server:latest "/entrypoint.sh drac…" About a minute ago Up About a minute 0.0.0.0:9060->9022/tcp test_drachtio_1
|
||||||
|
1f384a274f87 redis:5-alpine "docker-entrypoint.s…" 2 minutes ago Up 2 minutes 0.0.0.0:16379->6379/tcp test_redis_1
|
||||||
|
78d0bb6ec9b1 influxdb:1.8 "/entrypoint.sh infl…" 2 minutes ago Up 2 minutes 0.0.0.0:8086->8086/tcp test_influxdb_1
|
||||||
|
9616ff790709 jambonz/webhook-test-scaffold:latest "/entrypoint.sh" 2 minutes ago Up 2 minutes 0.0.0.0:3102->3000/tcp test_webhook-gather_1
|
||||||
|
7323ab273ff4 drachtio/drachtio-freeswitch-mrf:v1.10.1-full "/entrypoint.sh free…" 2 minutes ago Up 2 minutes (healthy) 0.0.0.0:8022->8021/tcp test_freeswitch_1
|
||||||
|
e45e7d28dbc7 mysql:5.7 "docker-entrypoint.s…" 2 minutes ago Up 2 minutes (healthy) 33060/tcp, 0.0.0.0:3360->3306/tcp test_mysql_1
|
||||||
|
b626e5f3067e qoomon/docker-host "/entrypoint.sh" 2 minutes ago Up 2 minutes test_docker-host_1
|
||||||
|
b0a94b5e8941 jambonz/webhook-test-scaffold:latest "/entrypoint.sh" 2 minutes ago Up 2 minutes 0.0.0.0:3101->3000/tcp test_webhook-say_1
|
||||||
|
f80adda48eb5 jambonz/webhook-test-scaffold:latest "/entrypoint.sh" 2 minutes ago Up 2 minutes 0.0.0.0:3103->3000/tcp test_webhook-transcribe_1
|
||||||
|
223db4a9c670 jambonz/webhook-test-scaffold:latest "/entrypoint.sh" 2 minutes ago Up 2 minutes 0.0.0.0:3100->3000/tcp test_webhook-decline_1
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Run the regression test suite
|
||||||
|
|
||||||
|
The test suite has a dependency that the mysql client is installed on your laptop/machine where the test will be run. This is needed in order to seed the mysql database that is running in the docker network.
|
||||||
|
|
||||||
|
Assuming you have installed the mysql client, and done the above steps, you should now be able to run the tests:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
./run-tests.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
If the docker network has not been started (as described above) it will start now, and this will take a minute or two. Otherwise, the test suite will start running immediately.
|
||||||
|
|
||||||
|
In evaluating the test results, be advised that the output is fairly verbose, and also in the process of shutting down once the tests are complete you will see a bunch of errors from redis (`@jambonz/realtimedb-helpers - redis error`). You can ignore these errors, they are just spit out by jambonz-feature-server as the test environment is torn down and it tries and fails to reconnect to redis.
|
||||||
|
|
||||||
|
The final output will indicate the number of tests run and passed:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
1..28
|
||||||
|
# tests 28
|
||||||
|
# pass 28
|
||||||
|
|
||||||
|
# ok
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Adding your own tests
|
||||||
|
|
||||||
|
Running a successful regression test means you haven't broken anything - Great!
|
||||||
|
|
||||||
|
It doesn't, of course, mean that your shiny new feature or bugfix works. Adding a new test case to the suite is (unfortunately) non-trivial. We will add more documentation in the future with a how-to guide on that, but be advised it does require knowledge of the SIP protocol and the [SIPp](http://sipp.sourceforge.net/doc/reference.html) tool.
|
||||||
|
|
||||||
|
For now, if you are unable to add tests to the regression suite, please do test your feature as thoroughly as you can on your own jambonz cluster before giving us a pull request.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -6,17 +6,20 @@ const {CallDirection, CallStatus} = require('../../utils/constants');
|
|||||||
const { v4: uuidv4 } = require('uuid');
|
const { v4: uuidv4 } = require('uuid');
|
||||||
const SipError = require('drachtio-srf').SipError;
|
const SipError = require('drachtio-srf').SipError;
|
||||||
const sysError = require('./error');
|
const sysError = require('./error');
|
||||||
const Requestor = require('../../utils/requestor');
|
const HttpRequestor = require('../../utils/http-requestor');
|
||||||
|
const WsRequestor = require('../../utils/ws-requestor');
|
||||||
|
const RootSpan = require('../../utils/call-tracer');
|
||||||
const dbUtils = require('../../utils/db-utils');
|
const dbUtils = require('../../utils/db-utils');
|
||||||
|
|
||||||
router.post('/', async(req, res) => {
|
router.post('/', async(req, res) => {
|
||||||
const {logger} = req.app.locals;
|
const {logger} = req.app.locals;
|
||||||
|
const accountSid = req.body.account_sid;
|
||||||
|
const {srf} = require('../../..');
|
||||||
|
|
||||||
logger.debug({body: req.body}, 'got createCall request');
|
logger.debug({body: req.body}, 'got createCall request');
|
||||||
try {
|
try {
|
||||||
let uri, cs, to;
|
let uri, cs, to;
|
||||||
const restDial = makeTask(logger, {'rest:dial': req.body});
|
const restDial = makeTask(logger, {'rest:dial': req.body});
|
||||||
const {srf} = require('../../..');
|
|
||||||
const {lookupAccountDetails} = dbUtils(logger, srf);
|
const {lookupAccountDetails} = dbUtils(logger, srf);
|
||||||
const {getSBC, getFreeswitch} = srf.locals;
|
const {getSBC, getFreeswitch} = srf.locals;
|
||||||
const sbcAddress = getSBC();
|
const sbcAddress = getSBC();
|
||||||
@@ -38,7 +41,7 @@ router.post('/', async(req, res) => {
|
|||||||
'X-Jambonz-Routing': target.type,
|
'X-Jambonz-Routing': target.type,
|
||||||
'X-Jambonz-FS-UUID': srf.locals.fsUUID,
|
'X-Jambonz-FS-UUID': srf.locals.fsUUID,
|
||||||
'X-Call-Sid': callSid,
|
'X-Call-Sid': callSid,
|
||||||
'X-Account-Sid': req.body.account_sid
|
'X-Account-Sid': accountSid
|
||||||
};
|
};
|
||||||
|
|
||||||
switch (target.type) {
|
switch (target.type) {
|
||||||
@@ -47,7 +50,7 @@ router.post('/', async(req, res) => {
|
|||||||
uri = `sip:${target.number}@${sbcAddress}`;
|
uri = `sip:${target.number}@${sbcAddress}`;
|
||||||
to = target.number;
|
to = target.number;
|
||||||
if ('teams' === target.type) {
|
if ('teams' === target.type) {
|
||||||
const obj = await lookupTeamsByAccount(req.body.account_sid);
|
const obj = await lookupTeamsByAccount(accountSid);
|
||||||
if (!obj) throw new Error('dial to ms teams not allowed; account must first be configured with teams info');
|
if (!obj) throw new Error('dial to ms teams not allowed; account must first be configured with teams info');
|
||||||
Object.assign(opts.headers, {
|
Object.assign(opts.headers, {
|
||||||
'X-MS-Teams-FQDN': obj.ms_teams_fqdn,
|
'X-MS-Teams-FQDN': obj.ms_teams_fqdn,
|
||||||
@@ -71,6 +74,17 @@ router.post('/', async(req, res) => {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (target.type === 'phone' && target.trunk) {
|
||||||
|
const {lookupCarrier} = dbUtils(this.logger, srf);
|
||||||
|
const voip_carrier_sid = await lookupCarrier(req.body.account_sid, target.trunk);
|
||||||
|
logger.info(
|
||||||
|
`createCall: selected ${voip_carrier_sid} for requested carrier: ${target.trunk || 'unspecified'})`);
|
||||||
|
if (voip_carrier_sid) {
|
||||||
|
opts.headers['X-Requested-Carrier-Sid'] = voip_carrier_sid;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* create endpoint for outdial */
|
/* create endpoint for outdial */
|
||||||
const ms = getFreeswitch();
|
const ms = getFreeswitch();
|
||||||
if (!ms) throw new Error('no available Freeswitch for outbound call creation');
|
if (!ms) throw new Error('no available Freeswitch for outbound call creation');
|
||||||
@@ -104,27 +118,62 @@ router.post('/', async(req, res) => {
|
|||||||
* attach our requestor and notifier objects
|
* attach our requestor and notifier objects
|
||||||
* these will be used for all http requests we make during this call
|
* these will be used for all http requests we make during this call
|
||||||
*/
|
*/
|
||||||
app.requestor = new Requestor(logger, account.account_sid, app.call_hook, account.webhook_secret);
|
if ('WS' === app.call_hook?.method || /^wss?:/.test(app.call_hook.url)) {
|
||||||
if (app.call_status_hook) {
|
logger.debug({call_hook: app.call_hook}, 'creating websocket for call hook');
|
||||||
app.notifier = new Requestor(logger, account.account_sid, app.call_status_hook, account.webhook_secret);
|
app.requestor = new WsRequestor(logger, account.account_sid, app.call_hook, account.webhook_secret) ;
|
||||||
|
if (app.call_hook.url === app.call_status_hook.url || !app.call_status_hook?.url) {
|
||||||
|
logger.debug('reusing websocket for call status hook');
|
||||||
|
app.notifier = app.requestor;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
logger.debug({call_hook: app.call_hook}, 'creating http client for call hook');
|
||||||
|
app.requestor = new HttpRequestor(logger, account.account_sid, app.call_hook, account.webhook_secret);
|
||||||
|
}
|
||||||
|
if (!app.notifier && app.call_status_hook) {
|
||||||
|
app.notifier = new HttpRequestor(logger, account.account_sid, app.call_status_hook, account.webhook_secret);
|
||||||
|
logger.debug({call_hook: app.call_hook}, 'creating http client for call status hook');
|
||||||
|
}
|
||||||
|
else if (!app.notifier) {
|
||||||
|
logger.debug('creating null call status hook');
|
||||||
|
app.notifier = {request: () => {}};
|
||||||
}
|
}
|
||||||
else app.notifier = {request: () => {}};
|
|
||||||
|
|
||||||
/* now launch the outdial */
|
/* now launch the outdial */
|
||||||
try {
|
try {
|
||||||
const dlg = await srf.createUAC(uri, {...opts, followRedirects: true, keepUriOnRedirect: true}, {
|
const dlg = await srf.createUAC(uri, {...opts, followRedirects: true, keepUriOnRedirect: true}, {
|
||||||
cbRequest: (err, inviteReq) => {
|
cbRequest: (err, inviteReq) => {
|
||||||
/* in case of 302 redirect, this gets called twice, ignore the second */
|
/* in case of 302 redirect, this gets called twice, ignore the second
|
||||||
if (res.headersSent) return;
|
except to update the req so that it can later be canceled if need be
|
||||||
|
*/
|
||||||
|
if (res.headersSent) {
|
||||||
|
logger.info(`create-call: got redirect, updating request to new call-id ${req.get('Call-ID')}`);
|
||||||
|
if (cs) cs.req = inviteReq;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (err) {
|
if (err) {
|
||||||
logger.error(err, 'createCall Error creating call');
|
logger.error(err, 'createCall Error creating call');
|
||||||
res.status(500).send('Call Failure');
|
res.status(500).send('Call Failure');
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
inviteReq.srf = srf;
|
||||||
|
inviteReq.locals = {
|
||||||
|
...(inviteReq || {}),
|
||||||
|
callSid,
|
||||||
|
application_sid: app.application_sid
|
||||||
|
};
|
||||||
/* ok our outbound INVITE is in flight */
|
/* ok our outbound INVITE is in flight */
|
||||||
|
|
||||||
const tasks = [restDial];
|
const tasks = [restDial];
|
||||||
|
const rootSpan = new RootSpan('rest-call', inviteReq);
|
||||||
|
sipLogger = logger.child({
|
||||||
|
callSid,
|
||||||
|
callId: inviteReq.get('Call-ID'),
|
||||||
|
accountSid,
|
||||||
|
traceId: rootSpan.traceId
|
||||||
|
});
|
||||||
|
app.requestor.logger = app.notifier.logger = sipLogger;
|
||||||
const callInfo = new CallInfo({
|
const callInfo = new CallInfo({
|
||||||
direction: CallDirection.Outbound,
|
direction: CallDirection.Outbound,
|
||||||
req: inviteReq,
|
req: inviteReq,
|
||||||
@@ -132,17 +181,24 @@ router.post('/', async(req, res) => {
|
|||||||
tag: app.tag,
|
tag: app.tag,
|
||||||
callSid,
|
callSid,
|
||||||
accountSid: req.body.account_sid,
|
accountSid: req.body.account_sid,
|
||||||
applicationSid: app.application_sid
|
applicationSid: app.application_sid,
|
||||||
|
traceId: rootSpan.traceId
|
||||||
|
});
|
||||||
|
cs = new RestCallSession({
|
||||||
|
logger: sipLogger,
|
||||||
|
application: app,
|
||||||
|
srf,
|
||||||
|
req: inviteReq,
|
||||||
|
ep,
|
||||||
|
tasks,
|
||||||
|
callInfo,
|
||||||
|
accountInfo,
|
||||||
|
rootSpan
|
||||||
});
|
});
|
||||||
cs = new RestCallSession({logger, application: app, srf, req: inviteReq, ep, tasks, callInfo, accountInfo});
|
|
||||||
cs.exec(req);
|
cs.exec(req);
|
||||||
|
|
||||||
res.status(201).json({sid: cs.callSid});
|
res.status(201).json({sid: cs.callSid});
|
||||||
|
|
||||||
sipLogger = logger.child({
|
|
||||||
callSid: cs.callSid,
|
|
||||||
callId: callInfo.callId
|
|
||||||
});
|
|
||||||
sipLogger.info(`outbound REST call attempt to ${JSON.stringify(target)} has been sent`);
|
sipLogger.info(`outbound REST call attempt to ${JSON.stringify(target)} has been sent`);
|
||||||
},
|
},
|
||||||
cbProvisional: (prov) => {
|
cbProvisional: (prov) => {
|
||||||
@@ -153,7 +209,11 @@ router.post('/', async(req, res) => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
connectStream(dlg.remote.sdp);
|
connectStream(dlg.remote.sdp);
|
||||||
cs.emit('callStatusChange', {callStatus: CallStatus.InProgress, sipStatus: 200});
|
cs.emit('callStatusChange', {
|
||||||
|
callStatus: CallStatus.InProgress,
|
||||||
|
sipStatus: 200,
|
||||||
|
sipReason: 'OK'
|
||||||
|
});
|
||||||
restDial.emit('callStatus', 200);
|
restDial.emit('callStatus', 200);
|
||||||
restDial.emit('connect', dlg);
|
restDial.emit('connect', dlg);
|
||||||
}
|
}
|
||||||
@@ -164,14 +224,23 @@ router.post('/', async(req, res) => {
|
|||||||
else if (487 === err.status) callStatus = CallStatus.NoAnswer;
|
else if (487 === err.status) callStatus = CallStatus.NoAnswer;
|
||||||
if (sipLogger) sipLogger.info(`REST outdial failed with ${err.status}`);
|
if (sipLogger) sipLogger.info(`REST outdial failed with ${err.status}`);
|
||||||
else console.log(`REST outdial failed with ${err.status}`);
|
else console.log(`REST outdial failed with ${err.status}`);
|
||||||
if (cs) cs.emit('callStatusChange', {callStatus, sipStatus: err.status});
|
if (cs) cs.emit('callStatusChange', {
|
||||||
|
callStatus,
|
||||||
|
sipStatus: err.status,
|
||||||
|
sipReason: err.reason
|
||||||
|
});
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if (cs) cs.emit('callStatusChange', {callStatus, sipStatus: 500});
|
if (cs) cs.emit('callStatusChange', {
|
||||||
|
callStatus,
|
||||||
|
sipStatus: 500,
|
||||||
|
sipReason: 'Internal Server Error'
|
||||||
|
});
|
||||||
if (sipLogger) sipLogger.error({err}, 'REST outdial failed');
|
if (sipLogger) sipLogger.error({err}, 'REST outdial failed');
|
||||||
else console.error(err);
|
else console.error(err);
|
||||||
}
|
}
|
||||||
ep.destroy();
|
ep.destroy();
|
||||||
|
setTimeout(restDial.kill.bind(restDial), 5000);
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
sysError(logger, res, err);
|
sysError(logger, res, err);
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
const router = require('express').Router();
|
const router = require('express').Router();
|
||||||
const Requestor = require('../../utils/requestor');
|
const HttpRequestor = require('../../utils/http-requestor');
|
||||||
|
const WsRequestor = require('../../utils/ws-requestor');
|
||||||
const CallInfo = require('../../session/call-info');
|
const CallInfo = require('../../session/call-info');
|
||||||
const {CallDirection} = require('../../utils/constants');
|
const {CallDirection} = require('../../utils/constants');
|
||||||
const SmsSession = require('../../session/sms-call-session');
|
const SmsSession = require('../../session/sms-call-session');
|
||||||
@@ -18,7 +19,17 @@ router.post('/:partner', async(req, res) => {
|
|||||||
const app = req.body.app;
|
const app = req.body.app;
|
||||||
const account = await lookupAccountBySid(app.accountSid);
|
const account = await lookupAccountBySid(app.accountSid);
|
||||||
const hook = app.messaging_hook;
|
const hook = app.messaging_hook;
|
||||||
const requestor = new Requestor(logger, account.account_sid, hook, account.webhook_secret);
|
let requestor;
|
||||||
|
|
||||||
|
if ('WS' === hook?.method) {
|
||||||
|
app.requestor = new WsRequestor(logger, account.account_sid, hook, account.webhook_secret) ;
|
||||||
|
app.notifier = app.requestor;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
app.requestor = new HttpRequestor(logger, account.account_sid, hook, account.webhook_secret);
|
||||||
|
app.notifier = {request: () => {}};
|
||||||
|
}
|
||||||
|
|
||||||
const payload = {
|
const payload = {
|
||||||
carrier: req.params.partner,
|
carrier: req.params.partner,
|
||||||
messageSid: app.messageSid,
|
messageSid: app.messageSid,
|
||||||
@@ -33,7 +44,7 @@ router.post('/:partner', async(req, res) => {
|
|||||||
res.status(200).json({sid: req.body.messageSid});
|
res.status(200).json({sid: req.body.messageSid});
|
||||||
|
|
||||||
try {
|
try {
|
||||||
tasks = await requestor.request(hook, payload);
|
tasks = await requestor.request('session:new', hook, payload);
|
||||||
logger.info({tasks}, 'response from incoming SMS webhook');
|
logger.info({tasks}, 'response from incoming SMS webhook');
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
logger.error({err, hook}, 'Error sending incoming SMS message');
|
logger.error({err, hook}, 'Error sending incoming SMS message');
|
||||||
|
|||||||
@@ -12,6 +12,9 @@ function retrieveCallSession(callSid, opts) {
|
|||||||
throw new DbErrorBadRequest('call_status_hook can be updated only when call_hook is also being updated');
|
throw new DbErrorBadRequest('call_status_hook can be updated only when call_hook is also being updated');
|
||||||
}
|
}
|
||||||
const cs = sessionTracker.get(callSid);
|
const cs = sessionTracker.get(callSid);
|
||||||
|
if (!cs) {
|
||||||
|
throw new DbErrorUnprocessableRequest('call session is gone');
|
||||||
|
}
|
||||||
|
|
||||||
if (opts.call_status === CallStatus.Completed && !cs.hasStableDialog) {
|
if (opts.call_status === CallStatus.Completed && !cs.hasStableDialog) {
|
||||||
throw new DbErrorUnprocessableRequest('current call state is incompatible with requested action');
|
throw new DbErrorUnprocessableRequest('current call state is incompatible with requested action');
|
||||||
@@ -45,8 +48,18 @@ router.post('/:callSid', async(req, res) => {
|
|||||||
logger.info(`updateCall: callSid not found ${callSid}`);
|
logger.info(`updateCall: callSid not found ${callSid}`);
|
||||||
return res.sendStatus(404);
|
return res.sendStatus(404);
|
||||||
}
|
}
|
||||||
res.sendStatus(202);
|
|
||||||
cs.updateCall(req.body, callSid);
|
if (req.body.sip_request) {
|
||||||
|
const response = await cs.updateCall(req.body, callSid);
|
||||||
|
res.status(200).json({
|
||||||
|
status: response.status,
|
||||||
|
reason: response.reason
|
||||||
|
});
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
res.sendStatus(202);
|
||||||
|
cs.updateCall(req.body, callSid);
|
||||||
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
sysError(logger, res, err);
|
sysError(logger, res, err);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,11 +1,14 @@
|
|||||||
const { v4: uuidv4 } = require('uuid');
|
const { v4: uuidv4 } = require('uuid');
|
||||||
const {CallDirection} = require('./utils/constants');
|
const {CallDirection} = require('./utils/constants');
|
||||||
const CallInfo = require('./session/call-info');
|
const CallInfo = require('./session/call-info');
|
||||||
const Requestor = require('./utils/requestor');
|
const HttpRequestor = require('./utils/http-requestor');
|
||||||
|
const WsRequestor = require('./utils/ws-requestor');
|
||||||
const makeTask = require('./tasks/make_task');
|
const makeTask = require('./tasks/make_task');
|
||||||
const parseUri = require('drachtio-srf').parseUri;
|
const parseUri = require('drachtio-srf').parseUri;
|
||||||
const normalizeJambones = require('./utils/normalize-jambones');
|
const normalizeJambones = require('./utils/normalize-jambones');
|
||||||
const dbUtils = require('./utils/db-utils');
|
const dbUtils = require('./utils/db-utils');
|
||||||
|
const RootSpan = require('./utils/call-tracer');
|
||||||
|
const listTaskNames = require('./utils/summarize-tasks');
|
||||||
|
|
||||||
module.exports = function(srf, logger) {
|
module.exports = function(srf, logger) {
|
||||||
const {
|
const {
|
||||||
@@ -16,15 +19,18 @@ module.exports = function(srf, logger) {
|
|||||||
lookupAppByTeamsTenant
|
lookupAppByTeamsTenant
|
||||||
} = srf.locals.dbHelpers;
|
} = srf.locals.dbHelpers;
|
||||||
const {lookupAccountDetails} = dbUtils(logger, srf);
|
const {lookupAccountDetails} = dbUtils(logger, srf);
|
||||||
|
|
||||||
function initLocals(req, res, next) {
|
function initLocals(req, res, next) {
|
||||||
|
if (!req.has('X-Account-Sid')) {
|
||||||
|
logger.info('getAccountDetails - rejecting call due to missing X-Account-Sid header');
|
||||||
|
return res.send(500);
|
||||||
|
}
|
||||||
const callSid = req.has('X-Retain-Call-Sid') ? req.get('X-Retain-Call-Sid') : uuidv4();
|
const callSid = req.has('X-Retain-Call-Sid') ? req.get('X-Retain-Call-Sid') : uuidv4();
|
||||||
req.locals = {
|
const account_sid = req.get('X-Account-Sid');
|
||||||
callSid,
|
req.locals = {callSid, account_sid};
|
||||||
logger: logger.child({callId: req.get('Call-ID'), callSid})
|
|
||||||
};
|
|
||||||
if (req.has('X-Application-Sid')) {
|
if (req.has('X-Application-Sid')) {
|
||||||
const application_sid = req.get('X-Application-Sid');
|
const application_sid = req.get('X-Application-Sid');
|
||||||
req.locals.logger.debug(`got application from X-Application-Sid header: ${application_sid}`);
|
logger.debug(`got application from X-Application-Sid header: ${application_sid}`);
|
||||||
req.locals.application_sid = application_sid;
|
req.locals.application_sid = application_sid;
|
||||||
}
|
}
|
||||||
if (req.has('X-Authenticated-User')) req.locals.originatingUser = req.get('X-Authenticated-User');
|
if (req.has('X-Authenticated-User')) req.locals.originatingUser = req.get('X-Authenticated-User');
|
||||||
@@ -33,19 +39,50 @@ module.exports = function(srf, logger) {
|
|||||||
next();
|
next();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function createRootSpan(req, res, next) {
|
||||||
|
const {callSid, account_sid} = req.locals;
|
||||||
|
const rootSpan = new RootSpan('incoming-call', req);
|
||||||
|
const traceId = rootSpan.traceId;
|
||||||
|
|
||||||
|
req.locals = {
|
||||||
|
...req.locals,
|
||||||
|
traceId,
|
||||||
|
logger: logger.child({
|
||||||
|
callId: req.get('Call-ID'),
|
||||||
|
callSid,
|
||||||
|
accountSid: account_sid,
|
||||||
|
callingNumber: req.callingNumber,
|
||||||
|
calledNumber: req.calledNumber,
|
||||||
|
traceId}),
|
||||||
|
rootSpan
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* end the span on final failure or cancel from caller;
|
||||||
|
* otherwise it will be closed when sip dialog is destroyed
|
||||||
|
*/
|
||||||
|
req.once('cancel', () => {
|
||||||
|
rootSpan.setAttributes({finalStatus: 487});
|
||||||
|
rootSpan.end();
|
||||||
|
});
|
||||||
|
res.once('finish', () => {
|
||||||
|
rootSpan.setAttributes({finalStatus: res.statusCode});
|
||||||
|
res.statusCode >= 300 && rootSpan.end();
|
||||||
|
});
|
||||||
|
|
||||||
|
next();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* retrieve account information for the incoming call
|
* retrieve account information for the incoming call
|
||||||
*/
|
*/
|
||||||
async function getAccountDetails(req, res, next) {
|
async function getAccountDetails(req, res, next) {
|
||||||
|
const {rootSpan, account_sid} = req.locals;
|
||||||
|
|
||||||
if (!req.has('X-Account-Sid')) {
|
const {span} = rootSpan.startChildSpan('lookupAccountDetails');
|
||||||
logger.info('getAccountDetails - rejecting call due to missing X-Account-Sid header');
|
|
||||||
return res.send(500);
|
|
||||||
}
|
|
||||||
const account_sid = req.locals.account_sid = req.get('X-Account-Sid');
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
req.locals.accountInfo = await lookupAccountDetails(account_sid);
|
req.locals.accountInfo = await lookupAccountDetails(account_sid);
|
||||||
|
span.end();
|
||||||
if (!req.locals.accountInfo.account.is_active) {
|
if (!req.locals.accountInfo.account.is_active) {
|
||||||
logger.info(`Account is inactive or suspended ${account_sid}`);
|
logger.info(`Account is inactive or suspended ${account_sid}`);
|
||||||
// TODO: alert
|
// TODO: alert
|
||||||
@@ -54,6 +91,7 @@ module.exports = function(srf, logger) {
|
|||||||
logger.debug({accountInfo: req.locals?.accountInfo?.account}, `retrieved account info for ${account_sid}`);
|
logger.debug({accountInfo: req.locals?.accountInfo?.account}, `retrieved account info for ${account_sid}`);
|
||||||
next();
|
next();
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
|
span.end();
|
||||||
logger.info({err}, `Error retrieving account details for account ${account_sid}`);
|
logger.info({err}, `Error retrieving account details for account ${account_sid}`);
|
||||||
res.send(503, {headers: {'X-Reason': `No Account exists for sid ${account_sid}`}});
|
res.send(503, {headers: {'X-Reason': `No Account exists for sid ${account_sid}`}});
|
||||||
}
|
}
|
||||||
@@ -85,7 +123,8 @@ module.exports = function(srf, logger) {
|
|||||||
*/
|
*/
|
||||||
async function retrieveApplication(req, res, next) {
|
async function retrieveApplication(req, res, next) {
|
||||||
const logger = req.locals.logger;
|
const logger = req.locals.logger;
|
||||||
const {accountInfo, account_sid} = req.locals;
|
const {accountInfo, account_sid, rootSpan} = req.locals;
|
||||||
|
const {span} = rootSpan.startChildSpan('lookupApplication');
|
||||||
try {
|
try {
|
||||||
let app;
|
let app;
|
||||||
if (req.locals.application_sid) app = await lookupAppBySid(req.locals.application_sid);
|
if (req.locals.application_sid) app = await lookupAppBySid(req.locals.application_sid);
|
||||||
@@ -105,7 +144,7 @@ module.exports = function(srf, logger) {
|
|||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
const uri = parseUri(req.uri);
|
const uri = parseUri(req.uri);
|
||||||
const arr = /context-(.*)/.exec(uri.user);
|
const arr = /context-(.*)/.exec(uri?.user);
|
||||||
if (arr) {
|
if (arr) {
|
||||||
// this is a transfer from another feature server
|
// this is a transfer from another feature server
|
||||||
const {retrieveKey, deleteKey} = srf.locals.dbHelpers;
|
const {retrieveKey, deleteKey} = srf.locals.dbHelpers;
|
||||||
@@ -129,6 +168,11 @@ module.exports = function(srf, logger) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
span.setAttributes({
|
||||||
|
'app.hook': app?.call_hook?.url,
|
||||||
|
'application_sid': req.locals.application_sid
|
||||||
|
});
|
||||||
|
span.end();
|
||||||
if (!app || !app.call_hook || !app.call_hook.url) {
|
if (!app || !app.call_hook || !app.call_hook.url) {
|
||||||
logger.info(`rejecting call to ${req.locals.calledNumber}: no application or webhook url`);
|
logger.info(`rejecting call to ${req.locals.calledNumber}: no application or webhook url`);
|
||||||
return res.send(480, {
|
return res.send(480, {
|
||||||
@@ -142,10 +186,18 @@ module.exports = function(srf, logger) {
|
|||||||
* create a requestor that we will use for all http requests we make during the call.
|
* create a requestor that we will use for all http requests we make during the call.
|
||||||
* also create a notifier for call status events (if not needed, its a no-op).
|
* also create a notifier for call status events (if not needed, its a no-op).
|
||||||
*/
|
*/
|
||||||
app.requestor = new Requestor(logger, account_sid, app.call_hook, accountInfo.account.webhook_secret);
|
if ('WS' === app.call_hook?.method ||
|
||||||
if (app.call_status_hook) app.notifier = new Requestor(logger, account_sid, app.call_status_hook,
|
app.call_hook?.url.startsWith('ws://') || app.call_hook?.url.startsWith('wss://')) {
|
||||||
accountInfo.account.webhook_secret);
|
app.requestor = new WsRequestor(logger, account_sid, app.call_hook, accountInfo.account.webhook_secret) ;
|
||||||
else app.notifier = {request: () => {}};
|
app.notifier = app.requestor;
|
||||||
|
app.call_hook.method = 'WS';
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
app.requestor = new HttpRequestor(logger, account_sid, app.call_hook, accountInfo.account.webhook_secret);
|
||||||
|
if (app.call_status_hook) app.notifier = new HttpRequestor(logger, account_sid, app.call_status_hook,
|
||||||
|
accountInfo.account.webhook_secret);
|
||||||
|
else app.notifier = {request: () => {}};
|
||||||
|
}
|
||||||
|
|
||||||
req.locals.application = app;
|
req.locals.application = app;
|
||||||
const obj = Object.assign({}, app);
|
const obj = Object.assign({}, app);
|
||||||
@@ -154,9 +206,15 @@ module.exports = function(srf, logger) {
|
|||||||
// eslint-disable-next-line no-unused-vars
|
// eslint-disable-next-line no-unused-vars
|
||||||
const {call_hook, call_status_hook, ...appInfo} = obj; // mask sensitive data like user/pass on webhook
|
const {call_hook, call_status_hook, ...appInfo} = obj; // mask sensitive data like user/pass on webhook
|
||||||
logger.info({app: appInfo}, `retrieved application for incoming call to ${req.locals.calledNumber}`);
|
logger.info({app: appInfo}, `retrieved application for incoming call to ${req.locals.calledNumber}`);
|
||||||
req.locals.callInfo = new CallInfo({req, app, direction: CallDirection.Inbound});
|
req.locals.callInfo = new CallInfo({
|
||||||
|
req,
|
||||||
|
app,
|
||||||
|
direction: CallDirection.Inbound,
|
||||||
|
traceId: rootSpan.traceId
|
||||||
|
});
|
||||||
next();
|
next();
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
|
span.end();
|
||||||
logger.error(err, `${req.get('Call-ID')} Error looking up application for ${req.calledNumber}`);
|
logger.error(err, `${req.get('Call-ID')} Error looking up application for ${req.calledNumber}`);
|
||||||
res.send(500);
|
res.send(500);
|
||||||
}
|
}
|
||||||
@@ -167,29 +225,55 @@ module.exports = function(srf, logger) {
|
|||||||
*/
|
*/
|
||||||
async function invokeWebCallback(req, res, next) {
|
async function invokeWebCallback(req, res, next) {
|
||||||
const logger = req.locals.logger;
|
const logger = req.locals.logger;
|
||||||
const app = req.locals.application;
|
const {rootSpan, application:app} = req.locals;
|
||||||
|
let span;
|
||||||
try {
|
try {
|
||||||
|
|
||||||
if (app.tasks) {
|
if (app.tasks) {
|
||||||
app.tasks = normalizeJambones(logger, app.tasks).map((tdata) => makeTask(logger, tdata));
|
app.tasks = normalizeJambones(logger, app.tasks).map((tdata) => makeTask(logger, tdata));
|
||||||
if (0 === app.tasks.length) throw new Error('no application provided');
|
if (0 === app.tasks.length) throw new Error('no application provided');
|
||||||
return next();
|
return next();
|
||||||
}
|
}
|
||||||
/* retrieve the application to execute for this inbound call */
|
/* retrieve the application to execute for this inbound call */
|
||||||
const params = Object.assign(app.call_hook.method === 'POST' ? {sip: req.msg} : {},
|
const params = Object.assign(['POST', 'WS'].includes(app.call_hook.method) ? {sip: req.msg} : {},
|
||||||
req.locals.callInfo);
|
req.locals.callInfo, {
|
||||||
const json = await app.requestor.request(app.call_hook, params);
|
defaults: {
|
||||||
|
synthesizer: {
|
||||||
|
vendor: app.speech_synthesis_vendor,
|
||||||
|
language: app.speech_synthesis_language,
|
||||||
|
voice: app.speech_synthesis_voice
|
||||||
|
},
|
||||||
|
recognizer: {
|
||||||
|
vendor: app.speech_recognizer_vendor,
|
||||||
|
language: app.speech_recognizer_language
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
logger.debug({params}, 'sending initial webhook');
|
||||||
|
const obj = rootSpan.startChildSpan('performAppWebhook');
|
||||||
|
span = obj.span;
|
||||||
|
const b3 = rootSpan.getTracingPropagation();
|
||||||
|
const httpHeaders = b3 && {b3};
|
||||||
|
const json = await app.requestor.request('session:new', app.call_hook, params, httpHeaders);
|
||||||
app.tasks = normalizeJambones(logger, json).map((tdata) => makeTask(logger, tdata));
|
app.tasks = normalizeJambones(logger, json).map((tdata) => makeTask(logger, tdata));
|
||||||
|
span.setAttributes({
|
||||||
|
'http.statusCode': 200,
|
||||||
|
'app.tasks': listTaskNames(app.tasks)
|
||||||
|
});
|
||||||
|
span.end();
|
||||||
if (0 === app.tasks.length) throw new Error('no application provided');
|
if (0 === app.tasks.length) throw new Error('no application provided');
|
||||||
next();
|
next();
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
logger.info({err}, `Error retrieving or parsing application: ${err.message}`);
|
span?.setAttributes({webhookStatus: err.statusCode});
|
||||||
res.send(480, {headers: {'X-Reason': err.message}});
|
span?.end();
|
||||||
|
logger.info({err}, `Error retrieving or parsing application: ${err?.message}`);
|
||||||
|
res.send(480, {headers: {'X-Reason': err?.message || 'unknown'}});
|
||||||
|
app.requestor.close();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
initLocals,
|
initLocals,
|
||||||
|
createRootSpan,
|
||||||
getAccountDetails,
|
getAccountDetails,
|
||||||
normalizeNumbers,
|
normalizeNumbers,
|
||||||
retrieveApplication,
|
retrieveApplication,
|
||||||
|
|||||||
@@ -8,14 +8,15 @@ const CallSession = require('./call-session');
|
|||||||
|
|
||||||
*/
|
*/
|
||||||
class AdultingCallSession extends CallSession {
|
class AdultingCallSession extends CallSession {
|
||||||
constructor({logger, application, singleDialer, tasks, callInfo, accountInfo}) {
|
constructor({logger, application, singleDialer, tasks, callInfo, accountInfo, rootSpan}) {
|
||||||
super({
|
super({
|
||||||
logger,
|
logger,
|
||||||
application,
|
application,
|
||||||
srf: singleDialer.dlg.srf,
|
srf: singleDialer.dlg.srf,
|
||||||
tasks,
|
tasks,
|
||||||
callInfo,
|
callInfo,
|
||||||
accountInfo
|
accountInfo,
|
||||||
|
rootSpan
|
||||||
});
|
});
|
||||||
this.sd = singleDialer;
|
this.sd = singleDialer;
|
||||||
|
|
||||||
@@ -30,15 +31,25 @@ class AdultingCallSession extends CallSession {
|
|||||||
return this.sd.dlg;
|
return this.sd.dlg;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Note: this is not an error. It is only here to avoid an assert ("no setter for dlg")
|
||||||
|
* when there is a call in Session:_clearResources to null out dlg and ep
|
||||||
|
*/
|
||||||
|
set dlg(newDlg) {}
|
||||||
|
|
||||||
get ep() {
|
get ep() {
|
||||||
return this.sd.ep;
|
return this.sd.ep;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* see note above */
|
||||||
|
set ep(newEp) {}
|
||||||
|
|
||||||
get callSid() {
|
get callSid() {
|
||||||
return this.callInfo.callSid;
|
return this.callInfo.callSid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_callerHungup() {
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = AdultingCallSession;
|
module.exports = AdultingCallSession;
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ class CallInfo {
|
|||||||
let from ;
|
let from ;
|
||||||
let srf;
|
let srf;
|
||||||
this.direction = opts.direction;
|
this.direction = opts.direction;
|
||||||
|
this.traceId = opts.traceId;
|
||||||
if (opts.req) {
|
if (opts.req) {
|
||||||
const u = opts.req.getParsedHeader('from');
|
const u = opts.req.getParsedHeader('from');
|
||||||
const uri = parseUri(u.uri);
|
const uri = parseUri(u.uri);
|
||||||
@@ -27,6 +28,7 @@ class CallInfo {
|
|||||||
this.to = req.calledNumber;
|
this.to = req.calledNumber;
|
||||||
this.callId = req.get('Call-ID');
|
this.callId = req.get('Call-ID');
|
||||||
this.sipStatus = 100;
|
this.sipStatus = 100;
|
||||||
|
this.sipReason = 'Trying';
|
||||||
this.callStatus = CallStatus.Trying;
|
this.callStatus = CallStatus.Trying;
|
||||||
this.originatingSipIp = req.get('X-Forwarded-For');
|
this.originatingSipIp = req.get('X-Forwarded-For');
|
||||||
this.originatingSipTrunkName = req.get('X-Originating-Carrier');
|
this.originatingSipTrunkName = req.get('X-Originating-Carrier');
|
||||||
@@ -45,6 +47,7 @@ class CallInfo {
|
|||||||
this.callId = req.get('Call-ID');
|
this.callId = req.get('Call-ID');
|
||||||
this.callStatus = CallStatus.Trying,
|
this.callStatus = CallStatus.Trying,
|
||||||
this.sipStatus = 100;
|
this.sipStatus = 100;
|
||||||
|
this.sipReason = 'Trying';
|
||||||
}
|
}
|
||||||
else if (this.direction === CallDirection.None) {
|
else if (this.direction === CallDirection.None) {
|
||||||
// outbound SMS
|
// outbound SMS
|
||||||
@@ -65,6 +68,7 @@ class CallInfo {
|
|||||||
this.callStatus = CallStatus.Trying,
|
this.callStatus = CallStatus.Trying,
|
||||||
this.callId = req.get('Call-ID');
|
this.callId = req.get('Call-ID');
|
||||||
this.sipStatus = 100;
|
this.sipStatus = 100;
|
||||||
|
this.sipReason = 'Trying';
|
||||||
this.from = from || req.callingNumber;
|
this.from = from || req.callingNumber;
|
||||||
this.to = to;
|
this.to = to;
|
||||||
if (tag) this._customerData = tag;
|
if (tag) this._customerData = tag;
|
||||||
@@ -81,9 +85,10 @@ class CallInfo {
|
|||||||
* @param {string} callStatus - current call status
|
* @param {string} callStatus - current call status
|
||||||
* @param {number} sipStatus - current sip status
|
* @param {number} sipStatus - current sip status
|
||||||
*/
|
*/
|
||||||
updateCallStatus(callStatus, sipStatus) {
|
updateCallStatus(callStatus, sipStatus, sipReason) {
|
||||||
this.callStatus = callStatus;
|
this.callStatus = callStatus;
|
||||||
if (sipStatus) this.sipStatus = sipStatus;
|
if (sipStatus) this.sipStatus = sipStatus;
|
||||||
|
if (sipReason) this.sipReason = sipReason;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -106,9 +111,11 @@ class CallInfo {
|
|||||||
to: this.to,
|
to: this.to,
|
||||||
callId: this.callId,
|
callId: this.callId,
|
||||||
sipStatus: this.sipStatus,
|
sipStatus: this.sipStatus,
|
||||||
|
sipReason: this.sipReason,
|
||||||
callStatus: this.callStatus,
|
callStatus: this.callStatus,
|
||||||
callerId: this.callerId,
|
callerId: this.callerId,
|
||||||
accountSid: this.accountSid,
|
accountSid: this.accountSid,
|
||||||
|
traceId: this.traceId,
|
||||||
applicationSid: this.applicationSid,
|
applicationSid: this.applicationSid,
|
||||||
fsSipAddress: this.localSipAddress
|
fsSipAddress: this.localSipAddress
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -7,7 +7,8 @@ const sessionTracker = require('./session-tracker');
|
|||||||
const makeTask = require('../tasks/make_task');
|
const makeTask = require('../tasks/make_task');
|
||||||
const normalizeJambones = require('../utils/normalize-jambones');
|
const normalizeJambones = require('../utils/normalize-jambones');
|
||||||
const listTaskNames = require('../utils/summarize-tasks');
|
const listTaskNames = require('../utils/summarize-tasks');
|
||||||
const Requestor = require('../utils/requestor');
|
const HttpRequestor = require('../utils/http-requestor');
|
||||||
|
const WsRequestor = require('../utils/ws-requestor');
|
||||||
const BADPRECONDITIONS = 'preconditions not met';
|
const BADPRECONDITIONS = 'preconditions not met';
|
||||||
const CALLER_CANCELLED_ERR_MSG = 'Response not sent due to unknown transaction';
|
const CALLER_CANCELLED_ERR_MSG = 'Response not sent due to unknown transaction';
|
||||||
|
|
||||||
@@ -34,7 +35,7 @@ class CallSession extends Emitter {
|
|||||||
* @param {array} opts.tasks - tasks we are to execute
|
* @param {array} opts.tasks - tasks we are to execute
|
||||||
* @param {callInfo} opts.callInfo - information about the call
|
* @param {callInfo} opts.callInfo - information about the call
|
||||||
*/
|
*/
|
||||||
constructor({logger, application, srf, tasks, callInfo, accountInfo, memberId, confName, confUuid}) {
|
constructor({logger, application, srf, tasks, callInfo, accountInfo, rootSpan, memberId, confName, confUuid}) {
|
||||||
super();
|
super();
|
||||||
this.logger = logger;
|
this.logger = logger;
|
||||||
this.application = application;
|
this.application = application;
|
||||||
@@ -49,6 +50,9 @@ class CallSession extends Emitter {
|
|||||||
this.stackIdx = 0;
|
this.stackIdx = 0;
|
||||||
this.callGone = false;
|
this.callGone = false;
|
||||||
this.notifiedComplete = false;
|
this.notifiedComplete = false;
|
||||||
|
this.rootSpan = rootSpan;
|
||||||
|
|
||||||
|
assert(rootSpan);
|
||||||
|
|
||||||
this.tmpFiles = new Set();
|
this.tmpFiles = new Set();
|
||||||
|
|
||||||
@@ -62,6 +66,9 @@ class CallSession extends Emitter {
|
|||||||
}
|
}
|
||||||
|
|
||||||
this._pool = srf.locals.dbHelpers.pool;
|
this._pool = srf.locals.dbHelpers.pool;
|
||||||
|
|
||||||
|
this.requestor.on('command', this._onCommand.bind(this));
|
||||||
|
this.requestor.on('connection-dropped', this._onWsConnectionDropped.bind(this));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -114,18 +121,27 @@ class CallSession extends Emitter {
|
|||||||
get speechSynthesisVendor() {
|
get speechSynthesisVendor() {
|
||||||
return this.application.speech_synthesis_vendor;
|
return this.application.speech_synthesis_vendor;
|
||||||
}
|
}
|
||||||
|
set speechSynthesisVendor(vendor) {
|
||||||
|
this.application.speech_synthesis_vendor = vendor;
|
||||||
|
}
|
||||||
/**
|
/**
|
||||||
* default voice to use for speech synthesis if not provided in the app
|
* default voice to use for speech synthesis if not provided in the app
|
||||||
*/
|
*/
|
||||||
get speechSynthesisVoice() {
|
get speechSynthesisVoice() {
|
||||||
return this.application.speech_synthesis_voice;
|
return this.application.speech_synthesis_voice;
|
||||||
}
|
}
|
||||||
|
set speechSynthesisVoice(voice) {
|
||||||
|
this.application.speech_synthesis_voice = voice;
|
||||||
|
}
|
||||||
/**
|
/**
|
||||||
* default language to use for speech synthesis if not provided in the app
|
* default language to use for speech synthesis if not provided in the app
|
||||||
*/
|
*/
|
||||||
get speechSynthesisLanguage() {
|
get speechSynthesisLanguage() {
|
||||||
return this.application.speech_synthesis_language;
|
return this.application.speech_synthesis_language;
|
||||||
}
|
}
|
||||||
|
set speechSynthesisLanguage(language) {
|
||||||
|
this.application.speech_synthesis_language = language;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* default vendor to use for speech recognition if not provided in the app
|
* default vendor to use for speech recognition if not provided in the app
|
||||||
@@ -133,12 +149,18 @@ class CallSession extends Emitter {
|
|||||||
get speechRecognizerVendor() {
|
get speechRecognizerVendor() {
|
||||||
return this.application.speech_recognizer_vendor;
|
return this.application.speech_recognizer_vendor;
|
||||||
}
|
}
|
||||||
|
set speechRecognizerVendor(vendor) {
|
||||||
|
this.application.speech_recognizer_vendor = vendor;
|
||||||
|
}
|
||||||
/**
|
/**
|
||||||
* default language to use for speech recognition if not provided in the app
|
* default language to use for speech recognition if not provided in the app
|
||||||
*/
|
*/
|
||||||
get speechRecognizerLanguage() {
|
get speechRecognizerLanguage() {
|
||||||
return this.application.speech_recognizer_language;
|
return this.application.speech_recognizer_language;
|
||||||
}
|
}
|
||||||
|
set speechRecognizerLanguage(language) {
|
||||||
|
this.application.speech_recognizer_language = language;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* indicates whether the call currently in progress
|
* indicates whether the call currently in progress
|
||||||
@@ -204,6 +226,60 @@ class CallSession extends Emitter {
|
|||||||
return this.memberId && this.confName && this.confUuid;
|
return this.memberId && this.confName && this.confUuid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get isBotModeEnabled() {
|
||||||
|
return this.backgroundGatherTask;
|
||||||
|
}
|
||||||
|
|
||||||
|
get b3() {
|
||||||
|
return this.rootSpan?.getTracingPropagation();
|
||||||
|
}
|
||||||
|
|
||||||
|
async enableBotMode(gather, autoEnable) {
|
||||||
|
try {
|
||||||
|
const t = normalizeJambones(this.logger, [gather]);
|
||||||
|
this.backgroundGatherTask = makeTask(this.logger, t[0]);
|
||||||
|
this.backgroundGatherTask
|
||||||
|
.once('dtmf', this._clearTasks.bind(this))
|
||||||
|
.once('vad', this._clearTasks.bind(this))
|
||||||
|
.once('transcription', this._clearTasks.bind(this))
|
||||||
|
.once('timeout', this._clearTasks.bind(this));
|
||||||
|
this.logger.info({gather}, 'CallSession:enableBotMode - starting background gather');
|
||||||
|
const resources = await this._evaluatePreconditions(this.backgroundGatherTask);
|
||||||
|
const {span, ctx} = this.rootSpan.startChildSpan(`background-gather:${this.backgroundGatherTask.summary}`);
|
||||||
|
this.backgroundGatherTask.span = span;
|
||||||
|
this.backgroundGatherTask.ctx = ctx;
|
||||||
|
this.backgroundGatherTask.exec(this, resources)
|
||||||
|
.then(() => {
|
||||||
|
this.logger.info('CallSession:enableBotMode: gather completed');
|
||||||
|
this.backgroundGatherTask && this.backgroundGatherTask.removeAllListeners();
|
||||||
|
this.backgroundGatherTask && this.backgroundGatherTask.span.end();
|
||||||
|
this.backgroundGatherTask = null;
|
||||||
|
if (autoEnable && !this.callGone && !this._stopping) {
|
||||||
|
this.logger.info('CallSession:enableBotMode: restarting background gather');
|
||||||
|
setImmediate(() => this.enableBotMode(gather, true));
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
this.logger.info({err}, 'CallSession:enableBotMode: gather threw error');
|
||||||
|
this.backgroundGatherTask && this.backgroundGatherTask.removeAllListeners();
|
||||||
|
this.backgroundGatherTask && this.backgroundGatherTask.span.end();
|
||||||
|
this.backgroundGatherTask = null;
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
this.logger.info({err, gather}, 'CallSession:enableBotMode - Error creating gather task');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
disableBotMode() {
|
||||||
|
if (this.backgroundGatherTask) {
|
||||||
|
try {
|
||||||
|
this.backgroundGatherTask.removeAllListeners();
|
||||||
|
this.backgroundGatherTask.kill().catch((err) => {});
|
||||||
|
} catch (err) {}
|
||||||
|
this.backgroundGatherTask = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
setConferenceDetails(memberId, confName, confUuid) {
|
setConferenceDetails(memberId, confName, confUuid) {
|
||||||
assert(!this.memberId && !this.confName && !this.confUuid);
|
assert(!this.memberId && !this.confName && !this.confUuid);
|
||||||
assert (memberId && confName && confUuid);
|
assert (memberId && confName && confUuid);
|
||||||
@@ -255,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) {
|
||||||
@@ -289,6 +365,7 @@ class CallSession extends Emitter {
|
|||||||
*/
|
*/
|
||||||
async exec() {
|
async exec() {
|
||||||
this.logger.info({tasks: listTaskNames(this.tasks)}, `CallSession:exec starting ${this.tasks.length} tasks`);
|
this.logger.info({tasks: listTaskNames(this.tasks)}, `CallSession:exec starting ${this.tasks.length} tasks`);
|
||||||
|
|
||||||
while (this.tasks.length && !this.callGone) {
|
while (this.tasks.length && !this.callGone) {
|
||||||
const taskNum = ++this.taskIdx;
|
const taskNum = ++this.taskIdx;
|
||||||
const stackNum = this.stackIdx;
|
const stackNum = this.stackIdx;
|
||||||
@@ -297,12 +374,24 @@ class CallSession extends Emitter {
|
|||||||
try {
|
try {
|
||||||
const resources = await this._evaluatePreconditions(task);
|
const resources = await this._evaluatePreconditions(task);
|
||||||
this.currentTask = task;
|
this.currentTask = task;
|
||||||
await task.exec(this, resources);
|
if (TaskName.Gather === task.name && this.isBotModeEnabled) {
|
||||||
|
const timeout = task.timeout;
|
||||||
|
this.logger.info(`CallSession:exec skipping #${stackNum}:${taskNum}: ${task.name}`);
|
||||||
|
this.backgroundGatherTask.updateTimeout(timeout);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
const {span, ctx} = this.rootSpan.startChildSpan(`verb:${task.summary}`);
|
||||||
|
task.span = span;
|
||||||
|
task.ctx = ctx;
|
||||||
|
await task.exec(this, resources);
|
||||||
|
task.span.end();
|
||||||
|
}
|
||||||
this.currentTask = null;
|
this.currentTask = null;
|
||||||
this.logger.info(`CallSession:exec completed task #${stackNum}:${taskNum}: ${task.name}`);
|
this.logger.info(`CallSession:exec completed task #${stackNum}:${taskNum}: ${task.name}`);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
|
task.span?.end();
|
||||||
this.currentTask = null;
|
this.currentTask = null;
|
||||||
if (err.message.includes(BADPRECONDITIONS)) {
|
if (err.message?.includes(BADPRECONDITIONS)) {
|
||||||
this.logger.info(`CallSession:exec task #${stackNum}:${taskNum}: ${task.name}: ${err.message}`);
|
this.logger.info(`CallSession:exec task #${stackNum}:${taskNum}: ${task.name}: ${err.message}`);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@@ -310,10 +399,31 @@ class CallSession extends Emitter {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (0 === this.tasks.length && this.hasStableDialog && this.requestor instanceof WsRequestor) {
|
||||||
|
let span;
|
||||||
|
try {
|
||||||
|
const {span} = this.rootSpan.startChildSpan('waiting for commands');
|
||||||
|
const {reason, queue, command} = await this._awaitCommandsOrHangup();
|
||||||
|
span.setAttributes({
|
||||||
|
'completion.reason': reason,
|
||||||
|
'async.request.queue': queue,
|
||||||
|
'async.request.command': command
|
||||||
|
});
|
||||||
|
span.end();
|
||||||
|
if (!this.hasStableDialog || this.callGone) break;
|
||||||
|
} catch (err) {
|
||||||
|
span.end();
|
||||||
|
this.logger.info(err, 'CallSession:exec - error waiting for new commands');
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// all done - cleanup
|
// all done - cleanup
|
||||||
this.logger.info('CallSession:exec all tasks complete');
|
this.logger.info('CallSession:exec all tasks complete');
|
||||||
|
this._stopping = true;
|
||||||
|
this.disableBotMode();
|
||||||
this._onTasksDone();
|
this._onTasksDone();
|
||||||
this._clearResources();
|
this._clearResources();
|
||||||
|
|
||||||
@@ -362,12 +472,15 @@ class CallSession extends Emitter {
|
|||||||
* this is called to clean up when the call is released from one side or another
|
* this is called to clean up when the call is released from one side or another
|
||||||
*/
|
*/
|
||||||
_callReleased() {
|
_callReleased() {
|
||||||
this.logger.debug('CallSession:_callReleased - caller hung up');
|
|
||||||
this.callGone = true;
|
this.callGone = true;
|
||||||
if (this.currentTask) {
|
if (this.currentTask) {
|
||||||
this.currentTask.kill(this);
|
this.currentTask.kill(this);
|
||||||
this.currentTask = null;
|
this.currentTask = null;
|
||||||
}
|
}
|
||||||
|
if (this.wakeupResolver) {
|
||||||
|
this.wakeupResolver({reason: 'session ended'});
|
||||||
|
this.wakeupResolver = null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -404,29 +517,45 @@ class CallSession extends Emitter {
|
|||||||
*/
|
*/
|
||||||
async _lccCallHook(opts) {
|
async _lccCallHook(opts) {
|
||||||
const webhooks = [];
|
const webhooks = [];
|
||||||
let sd;
|
let sd, tasks, childTasks;
|
||||||
if (opts.call_hook) webhooks.push(this.requestor.request(opts.call_hook, this.callInfo.toJSON()));
|
const b3 = this.b3;
|
||||||
if (opts.child_call_hook) {
|
const httpHeaders = b3 && {b3};
|
||||||
/* child call hook only allowed from a connected Dial state */
|
|
||||||
const task = this.currentTask;
|
if (opts.call_hook || opts.child_call_hook) {
|
||||||
sd = task.sd;
|
if (opts.call_hook) {
|
||||||
if (task && TaskName.Dial === task.name && sd) {
|
webhooks.push(this.requestor.request('session:redirect', opts.call_hook, this.callInfo.toJSON(), httpHeaders));
|
||||||
webhooks.push(this.requestor.request(opts.child_call_hook, sd.callInfo.toJSON()));
|
|
||||||
}
|
}
|
||||||
|
if (opts.child_call_hook) {
|
||||||
|
/* child call hook only allowed from a connected Dial state */
|
||||||
|
const task = this.currentTask;
|
||||||
|
sd = task.sd;
|
||||||
|
if (task && TaskName.Dial === task.name && sd) {
|
||||||
|
webhooks.push(this.requestor.request(
|
||||||
|
'session:redirect', opts.child_call_hook, sd.callInfo.toJSON(), httpHeaders));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const [tasks1, tasks2] = await Promise.all(webhooks);
|
||||||
|
if (opts.call_hook) {
|
||||||
|
tasks = tasks1;
|
||||||
|
if (opts.child_call_hook) childTasks = tasks2;
|
||||||
|
}
|
||||||
|
else childTasks = tasks1;
|
||||||
}
|
}
|
||||||
const [tasks1, tasks2] = await Promise.all(webhooks);
|
else if (opts.parent_call || opts.child_call) {
|
||||||
let tasks, childTasks;
|
const {parent_call, child_call} = opts;
|
||||||
if (opts.call_hook) {
|
assert.ok(!parent_call || Array.isArray(parent_call), 'CallSession:_lccCallHook - parent_call must be an array');
|
||||||
tasks = tasks1;
|
assert.ok(!child_call || Array.isArray(child_call), 'CallSession:_lccCallHook - child_call must be an array');
|
||||||
if (opts.child_call_hook) childTasks = tasks2;
|
tasks = parent_call;
|
||||||
|
childTasks = child_call;
|
||||||
}
|
}
|
||||||
else childTasks = tasks1;
|
|
||||||
|
|
||||||
if (childTasks) {
|
if (childTasks) {
|
||||||
const {parentLogger} = this.srf.locals;
|
const {parentLogger} = this.srf.locals;
|
||||||
const childLogger = parentLogger.child({callId: this.callId, callSid: sd.callSid});
|
const childLogger = parentLogger.child({callId: this.callId, callSid: sd.callSid});
|
||||||
const t = normalizeJambones(childLogger, childTasks).map((tdata) => makeTask(childLogger, tdata));
|
const t = normalizeJambones(childLogger, childTasks).map((tdata) => makeTask(childLogger, tdata));
|
||||||
childLogger.info({tasks: listTaskNames(t)}, 'CallSession:_lccCallHook new task list for child call');
|
childLogger.info({tasks: listTaskNames(t)}, 'CallSession:_lccCallHook new task list for child call');
|
||||||
|
|
||||||
|
// TODO: if using websockets api, we need a new websocket for the adulting session..
|
||||||
const cs = await sd.doAdulting({
|
const cs = await sd.doAdulting({
|
||||||
logger: childLogger,
|
logger: childLogger,
|
||||||
application: this.application,
|
application: this.application,
|
||||||
@@ -474,7 +603,7 @@ class CallSession extends Emitter {
|
|||||||
task.mute(callSid, mute).catch((err) => this.logger.error(err, 'CallSession:_lccMuteStatus'));
|
task.mute(callSid, mute).catch((err) => this.logger.error(err, 'CallSession:_lccMuteStatus'));
|
||||||
}
|
}
|
||||||
|
|
||||||
async _lccConfHoldStatus(callSid, opts) {
|
async _lccConfHoldStatus(opts) {
|
||||||
const task = this.currentTask;
|
const task = this.currentTask;
|
||||||
if (!task || TaskName.Conference !== task.name || !this.isInConference) {
|
if (!task || TaskName.Conference !== task.name || !this.isInConference) {
|
||||||
return this.logger.info('CallSession:_lccConfHoldStatus - invalid command as call is not in conference');
|
return this.logger.info('CallSession:_lccConfHoldStatus - invalid command as call is not in conference');
|
||||||
@@ -482,7 +611,7 @@ class CallSession extends Emitter {
|
|||||||
task.doConferenceHold(this, opts);
|
task.doConferenceHold(this, opts);
|
||||||
}
|
}
|
||||||
|
|
||||||
async _lccConfMuteStatus(callSid, opts) {
|
async _lccConfMuteStatus(opts) {
|
||||||
const task = this.currentTask;
|
const task = this.currentTask;
|
||||||
if (!task || TaskName.Conference !== task.name || !this.isInConference) {
|
if (!task || TaskName.Conference !== task.name || !this.isInConference) {
|
||||||
return this.logger.info('CallSession:_lccConfHoldStatus - invalid command as call is not in conference');
|
return this.logger.info('CallSession:_lccConfHoldStatus - invalid command as call is not in conference');
|
||||||
@@ -490,6 +619,30 @@ class CallSession extends Emitter {
|
|||||||
task.doConferenceMuteNonModerators(this, opts);
|
task.doConferenceMuteNonModerators(this, opts);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async _lccSipRequest(opts, callSid) {
|
||||||
|
const {sip_request} = opts;
|
||||||
|
const {method, content_type, content, headers = {}} = sip_request;
|
||||||
|
if (!this.hasStableDialog) {
|
||||||
|
this.logger.info('CallSession:_lccSipRequest - invalid command as we do not have a stable call');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
const dlg = callSid === this.callSid ? this.dlg : this.currentTask.dlg;
|
||||||
|
const res = await dlg.request({
|
||||||
|
method,
|
||||||
|
headers: {
|
||||||
|
...headers,
|
||||||
|
'Content-Type': content_type
|
||||||
|
},
|
||||||
|
body: content
|
||||||
|
});
|
||||||
|
this.logger.debug({res}, `CallSession:_lccSipRequest got response to ${method}`);
|
||||||
|
return res;
|
||||||
|
} catch (err) {
|
||||||
|
this.logger.error({err}, `CallSession:_lccSipRequest - error sending ${method}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* perform live call control -- whisper to one party or the other on a call
|
* perform live call control -- whisper to one party or the other on a call
|
||||||
* @param {array} opts - array of play or say tasks
|
* @param {array} opts - array of play or say tasks
|
||||||
@@ -497,6 +650,8 @@ class CallSession extends Emitter {
|
|||||||
async _lccWhisper(opts, callSid) {
|
async _lccWhisper(opts, callSid) {
|
||||||
const {whisper} = opts;
|
const {whisper} = opts;
|
||||||
let tasks;
|
let tasks;
|
||||||
|
const b3 = this.b3;
|
||||||
|
const httpHeaders = b3 && {b3};
|
||||||
|
|
||||||
// this whole thing requires us to be in a Dial verb
|
// this whole thing requires us to be in a Dial verb
|
||||||
const task = this.currentTask;
|
const task = this.currentTask;
|
||||||
@@ -507,7 +662,7 @@ class CallSession extends Emitter {
|
|||||||
// allow user to provide a url object, a url string, an array of tasks, or a single task
|
// allow user to provide a url object, a url string, an array of tasks, or a single task
|
||||||
if (typeof whisper === 'string' || (typeof whisper === 'object' && whisper.url)) {
|
if (typeof whisper === 'string' || (typeof whisper === 'object' && whisper.url)) {
|
||||||
// retrieve a url
|
// retrieve a url
|
||||||
const json = await this.requestor(opts.call_hook, this.callInfo.toJSON());
|
const json = await this.requestor(opts.call_hook, this.callInfo.toJSON(), httpHeaders);
|
||||||
tasks = normalizeJambones(this.logger, json).map((tdata) => makeTask(this.logger, tdata));
|
tasks = normalizeJambones(this.logger, json).map((tdata) => makeTask(this.logger, tdata));
|
||||||
}
|
}
|
||||||
else if (Array.isArray(whisper)) {
|
else if (Array.isArray(whisper)) {
|
||||||
@@ -560,10 +715,14 @@ class CallSession extends Emitter {
|
|||||||
await this._lccMuteStatus(callSid, opts.mute_status === 'mute');
|
await this._lccMuteStatus(callSid, opts.mute_status === 'mute');
|
||||||
}
|
}
|
||||||
else if (opts.conf_hold_status) {
|
else if (opts.conf_hold_status) {
|
||||||
await this._lccConfHoldStatus(callSid, opts);
|
await this._lccConfHoldStatus(opts);
|
||||||
}
|
}
|
||||||
else if (opts.conf_mute_status) {
|
else if (opts.conf_mute_status) {
|
||||||
await this._lccConfMuteStatus(callSid, opts);
|
await this._lccConfMuteStatus(opts);
|
||||||
|
}
|
||||||
|
else if (opts.sip_request) {
|
||||||
|
const res = await this._lccSipRequest(opts, callSid);
|
||||||
|
return {status: res.status, reason: res.reason};
|
||||||
}
|
}
|
||||||
|
|
||||||
// whisper may be the only thing we are asked to do, or it may that
|
// whisper may be the only thing we are asked to do, or it may that
|
||||||
@@ -604,6 +763,131 @@ class CallSession extends Emitter {
|
|||||||
this.taskIdx = 0;
|
this.taskIdx = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Append tasks to the current execution stack UNLESS there is a gather in the stack.
|
||||||
|
* in that case, insert the tasks before the gather AND if the tasks include
|
||||||
|
* a gather then delete/remove the gather from the existing stack
|
||||||
|
* @param {*} t array of tasks
|
||||||
|
*/
|
||||||
|
_injectTasks(newTasks) {
|
||||||
|
const gatherPos = this.tasks.map((t) => t.name).indexOf(TaskName.Gather);
|
||||||
|
const currentlyExecutingGather = this.currentTask?.name === TaskName.Gather;
|
||||||
|
|
||||||
|
this.logger.debug({
|
||||||
|
currentTaskList: listTaskNames(this.tasks),
|
||||||
|
newContent: listTaskNames(newTasks),
|
||||||
|
currentlyExecutingGather,
|
||||||
|
gatherPos
|
||||||
|
}, 'CallSession:_injectTasks - starting');
|
||||||
|
|
||||||
|
const killGather = () => {
|
||||||
|
this.logger.debug('CallSession:_injectTasks - killing current gather because we have new content');
|
||||||
|
this.currentTask.kill(this);
|
||||||
|
};
|
||||||
|
|
||||||
|
if (-1 === gatherPos) {
|
||||||
|
/* no gather in the stack simply append tasks */
|
||||||
|
this.tasks.push(...newTasks);
|
||||||
|
this.logger.debug({
|
||||||
|
updatedTaskList: listTaskNames(this.tasks)
|
||||||
|
}, 'CallSession:_injectTasks - completed (simple append)');
|
||||||
|
|
||||||
|
/* we do need to kill the current gather if we are executing one */
|
||||||
|
if (currentlyExecutingGather) killGather();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (currentlyExecutingGather) killGather();
|
||||||
|
const newTasksHasGather = newTasks.find((t) => t.name === TaskName.Gather);
|
||||||
|
this.tasks.splice(gatherPos, newTasksHasGather ? 1 : 0, ...newTasks);
|
||||||
|
|
||||||
|
this.logger.debug({
|
||||||
|
updatedTaskList: listTaskNames(this.tasks)
|
||||||
|
}, 'CallSession:_injectTasks - completed');
|
||||||
|
}
|
||||||
|
|
||||||
|
_onCommand({msgid, command, call_sid, queueCommand, data}) {
|
||||||
|
this.logger.info({msgid, command, queueCommand}, 'CallSession:_onCommand - received command');
|
||||||
|
const resolution = {reason: 'received command', queue: queueCommand, command};
|
||||||
|
switch (command) {
|
||||||
|
case 'redirect':
|
||||||
|
if (Array.isArray(data)) {
|
||||||
|
const t = normalizeJambones(this.logger, data)
|
||||||
|
.map((tdata) => makeTask(this.logger, tdata));
|
||||||
|
if (!queueCommand) {
|
||||||
|
this.logger.info({tasks: listTaskNames(t)}, 'CallSession:_onCommand new task list');
|
||||||
|
this.replaceApplication(t);
|
||||||
|
}
|
||||||
|
else if (process.env.JAMBONES_INJECT_CONTENT) {
|
||||||
|
this.logger.debug({tasks: listTaskNames(t)}, 'CallSession:_onCommand - queueing tasks (injecting content)');
|
||||||
|
this._injectTasks(t);
|
||||||
|
this.logger.info({tasks: listTaskNames(this.tasks)}, 'CallSession:_onCommand - updated task list');
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
this.logger.debug({tasks: listTaskNames(t)}, 'CallSession:_onCommand - queueing tasks');
|
||||||
|
this.tasks.push(...t);
|
||||||
|
this.logger.info({tasks: listTaskNames(this.tasks)}, 'CallSession:_onCommand - updated task list');
|
||||||
|
}
|
||||||
|
resolution.command = listTaskNames(t);
|
||||||
|
}
|
||||||
|
else this._lccCallHook(data);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'call:status':
|
||||||
|
this._lccCallStatus(data);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'mute:status':
|
||||||
|
this._lccMuteStatus(call_sid, data);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'conf:mute-status':
|
||||||
|
this._lccConfMuteStatus(data);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'conf:hold-status':
|
||||||
|
this._lccConfHoldStatus(data);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'listen:status':
|
||||||
|
this._lccListenStatus(data);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'whisper':
|
||||||
|
this._lccWhisper(data, call_sid);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'sip:request':
|
||||||
|
this._lccSipRequest(data, call_sid)
|
||||||
|
.catch((err) => {
|
||||||
|
this.logger.info({err, data}, `CallSession:_onCommand - error sending ${data.method}`);
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
this.logger.info(`CallSession:_onCommand - invalid command ${command}`);
|
||||||
|
}
|
||||||
|
if (this.wakeupResolver) {
|
||||||
|
this.logger.debug({resolution}, 'CallSession:_onCommand - got commands, waking up..');
|
||||||
|
this.wakeupResolver(resolution);
|
||||||
|
this.wakeupResolver = null;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
const {span} = this.rootSpan.startChildSpan('async command');
|
||||||
|
const {queue, command} = resolution;
|
||||||
|
span.setAttributes({
|
||||||
|
'async.request.queue': queue,
|
||||||
|
'async.request.command': command
|
||||||
|
});
|
||||||
|
span.end();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_onWsConnectionDropped() {
|
||||||
|
const {stats} = this.srf.locals;
|
||||||
|
stats.increment('app.hook.remote_close');
|
||||||
|
}
|
||||||
|
|
||||||
_evaluatePreconditions(task) {
|
_evaluatePreconditions(task) {
|
||||||
switch (task.preconditions) {
|
switch (task.preconditions) {
|
||||||
case TaskPreconditions.None:
|
case TaskPreconditions.None:
|
||||||
@@ -668,7 +952,11 @@ class CallSession extends Emitter {
|
|||||||
} catch (err) {
|
} catch (err) {
|
||||||
if (err === CALLER_CANCELLED_ERR_MSG) {
|
if (err === CALLER_CANCELLED_ERR_MSG) {
|
||||||
this.logger.error(err, 'caller canceled quickly before we could respond, ending call');
|
this.logger.error(err, 'caller canceled quickly before we could respond, ending call');
|
||||||
this._notifyCallStatusChange({callStatus: CallStatus.NoAnswer, sipStatus: 487});
|
this._notifyCallStatusChange({
|
||||||
|
callStatus: CallStatus.NoAnswer,
|
||||||
|
sipStatus: 487,
|
||||||
|
sipReason: 'Request Terminated'
|
||||||
|
});
|
||||||
this._callReleased();
|
this._callReleased();
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@@ -740,6 +1028,9 @@ class CallSession extends Emitter {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
this.tmpFiles.clear();
|
this.tmpFiles.clear();
|
||||||
|
this.requestor && this.requestor.close();
|
||||||
|
|
||||||
|
this.rootSpan && this.rootSpan.end();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -772,14 +1063,21 @@ class CallSession extends Emitter {
|
|||||||
async propagateAnswer() {
|
async propagateAnswer() {
|
||||||
if (!this.dlg) {
|
if (!this.dlg) {
|
||||||
assert(this.ep);
|
assert(this.ep);
|
||||||
this.dlg = await this.srf.createUAS(this.req, this.res, {localSdp: this.ep.local.sdp});
|
this.dlg = await this.srf.createUAS(this.req, this.res, {
|
||||||
|
headers: {
|
||||||
|
'X-Trace-ID': this.req.locals.traceId,
|
||||||
|
'X-Call-Sid': this.req.locals.callSid
|
||||||
|
},
|
||||||
|
localSdp: this.ep.local.sdp
|
||||||
|
});
|
||||||
this.logger.debug('answered call');
|
this.logger.debug('answered call');
|
||||||
this.dlg.on('destroy', this._callerHungup.bind(this));
|
this.dlg.on('destroy', this._callerHungup.bind(this));
|
||||||
this.wrapDialog(this.dlg);
|
this.wrapDialog(this.dlg);
|
||||||
this.dlg.callSid = this.callSid;
|
this.dlg.callSid = this.callSid;
|
||||||
this.emit('callStatusChange', {sipStatus: 200, callStatus: CallStatus.InProgress});
|
this.emit('callStatusChange', {sipStatus: 200, sipReason: 'OK', callStatus: CallStatus.InProgress});
|
||||||
|
|
||||||
this.dlg.on('modify', this._onReinvite.bind(this));
|
this.dlg.on('modify', this._onReinvite.bind(this));
|
||||||
|
this.dlg.on('refer', this._onRefer.bind(this));
|
||||||
|
|
||||||
this.logger.debug(`CallSession:propagateAnswer - answered callSid ${this.callSid}`);
|
this.logger.debug(`CallSession:propagateAnswer - answered callSid ${this.callSid}`);
|
||||||
}
|
}
|
||||||
@@ -805,6 +1103,22 @@ class CallSession extends Emitter {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handle incoming REFER if we are in a dial task
|
||||||
|
* @param {*} req
|
||||||
|
* @param {*} res
|
||||||
|
*/
|
||||||
|
_onRefer(req, res) {
|
||||||
|
const task = this.currentTask;
|
||||||
|
const sd = task.sd;
|
||||||
|
if (task && TaskName.Dial === task.name && sd) {
|
||||||
|
task.handleRefer(this, req, res);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
res.send(501);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* create and endpoint if we don't have one; otherwise simply return
|
* create and endpoint if we don't have one; otherwise simply return
|
||||||
* the current media server and endpoint that are associated with this call
|
* the current media server and endpoint that are associated with this call
|
||||||
@@ -841,7 +1155,7 @@ class CallSession extends Emitter {
|
|||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
this.logger.info({accountSid: this.accountSid, webhook: r[0]}, 'performQueueWebhook: webhook found');
|
this.logger.info({accountSid: this.accountSid, webhook: r[0]}, 'performQueueWebhook: webhook found');
|
||||||
this.queueEventHookRequestor = new Requestor(this.logger, this.accountSid,
|
this.queueEventHookRequestor = new HttpRequestor(this.logger, this.accountSid,
|
||||||
r[0], this.webhook_secret);
|
r[0], this.webhook_secret);
|
||||||
this.queueEventHook = r[0];
|
this.queueEventHook = r[0];
|
||||||
}
|
}
|
||||||
@@ -855,7 +1169,7 @@ class CallSession extends Emitter {
|
|||||||
/* send webhook */
|
/* send webhook */
|
||||||
const params = {...obj, ...this.callInfo.toJSON()};
|
const params = {...obj, ...this.callInfo.toJSON()};
|
||||||
this.logger.info({accountSid: this.accountSid, params}, 'performQueueWebhook: sending webhook');
|
this.logger.info({accountSid: this.accountSid, params}, 'performQueueWebhook: sending webhook');
|
||||||
this.queueEventHookRequestor.request(this.queueEventHook, params)
|
this.queueEventHookRequestor.request('queue:status', this.queueEventHook, params)
|
||||||
.catch((err) => {
|
.catch((err) => {
|
||||||
this.logger.info({err, accountSid: this.accountSid, obj}, 'Error sending queue notification event');
|
this.logger.info({err, accountSid: this.accountSid, obj}, 'Error sending queue notification event');
|
||||||
});
|
});
|
||||||
@@ -943,7 +1257,12 @@ class CallSession extends Emitter {
|
|||||||
const duration = moment().diff(this.dlg.connectTime, 'seconds');
|
const duration = moment().diff(this.dlg.connectTime, 'seconds');
|
||||||
this.emit('callStatusChange', {callStatus: CallStatus.Completed, duration});
|
this.emit('callStatusChange', {callStatus: CallStatus.Completed, duration});
|
||||||
this.logger.debug('CallSession: call terminated by jambones');
|
this.logger.debug('CallSession: call terminated by jambones');
|
||||||
|
this.rootSpan.setAttributes({'call.termination': 'hangup by jambonz'});
|
||||||
origDestroy();
|
origDestroy();
|
||||||
|
if (this.wakeupResolver) {
|
||||||
|
this.wakeupResolver({reason: 'session ended'});
|
||||||
|
this.wakeupResolver = null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -986,7 +1305,7 @@ class CallSession extends Emitter {
|
|||||||
* @param {number} sipStatus - current sip status
|
* @param {number} sipStatus - current sip status
|
||||||
* @param {number} [duration] - duration of a completed call, in seconds
|
* @param {number} [duration] - duration of a completed call, in seconds
|
||||||
*/
|
*/
|
||||||
_notifyCallStatusChange({callStatus, sipStatus, duration}) {
|
_notifyCallStatusChange({callStatus, sipStatus, sipReason, duration}) {
|
||||||
if (this.callMoved) return;
|
if (this.callMoved) return;
|
||||||
|
|
||||||
/* race condition: we hang up at the same time as the caller */
|
/* race condition: we hang up at the same time as the caller */
|
||||||
@@ -999,11 +1318,17 @@ class CallSession extends Emitter {
|
|||||||
(!duration && callStatus !== CallStatus.Completed),
|
(!duration && callStatus !== CallStatus.Completed),
|
||||||
'duration MUST be supplied when call completed AND ONLY when call completed');
|
'duration MUST be supplied when call completed AND ONLY when call completed');
|
||||||
|
|
||||||
this.callInfo.updateCallStatus(callStatus, sipStatus);
|
this.callInfo.updateCallStatus(callStatus, sipStatus, sipReason);
|
||||||
if (typeof duration === 'number') this.callInfo.duration = duration;
|
if (typeof duration === 'number') this.callInfo.duration = duration;
|
||||||
|
const {span} = this.rootSpan.startChildSpan(`call-status:${this.callInfo.callStatus}`);
|
||||||
|
span.setAttributes(this.callInfo.toJSON());
|
||||||
try {
|
try {
|
||||||
this.notifier.request(this.call_status_hook, this.callInfo.toJSON());
|
const b3 = this.b3;
|
||||||
|
const httpHeaders = b3 && {b3};
|
||||||
|
this.notifier.request('call:status', this.call_status_hook, this.callInfo.toJSON(), httpHeaders);
|
||||||
|
span.end();
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
|
span.end();
|
||||||
this.logger.info(err, `CallSession:_notifyCallStatusChange error sending ${callStatus} ${sipStatus}`);
|
this.logger.info(err, `CallSession:_notifyCallStatusChange error sending ${callStatus} ${sipStatus}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1012,6 +1337,23 @@ class CallSession extends Emitter {
|
|||||||
this.updateCallStatus(Object.assign({}, this.callInfo.toJSON()), this.serviceUrl)
|
this.updateCallStatus(Object.assign({}, this.callInfo.toJSON()), this.serviceUrl)
|
||||||
.catch((err) => this.logger.error(err, 'redis error'));
|
.catch((err) => this.logger.error(err, 'redis error'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_awaitCommandsOrHangup() {
|
||||||
|
assert(!this.wakeupResolver);
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
this.logger.info('_awaitCommandsOrHangup - waiting...');
|
||||||
|
this.wakeupResolver = resolve;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
_clearTasks(evt) {
|
||||||
|
if (this.requestor instanceof WsRequestor) {
|
||||||
|
this.logger.info({evt}, 'CallSession:_clearTasks on event from background gather');
|
||||||
|
try {
|
||||||
|
this.kill();
|
||||||
|
} catch (err) {}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = CallSession;
|
module.exports = CallSession;
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ const CallSession = require('./call-session');
|
|||||||
|
|
||||||
*/
|
*/
|
||||||
class ConfirmCallSession extends CallSession {
|
class ConfirmCallSession extends CallSession {
|
||||||
constructor({logger, application, dlg, ep, tasks, callInfo, accountInfo, memberId, confName}) {
|
constructor({logger, application, dlg, ep, tasks, callInfo, accountInfo, memberId, confName, rootSpan}) {
|
||||||
super({
|
super({
|
||||||
logger,
|
logger,
|
||||||
application,
|
application,
|
||||||
@@ -18,7 +18,8 @@ class ConfirmCallSession extends CallSession {
|
|||||||
callInfo,
|
callInfo,
|
||||||
accountInfo,
|
accountInfo,
|
||||||
memberId,
|
memberId,
|
||||||
confName
|
confName,
|
||||||
|
rootSpan
|
||||||
});
|
});
|
||||||
this.dlg = dlg;
|
this.dlg = dlg;
|
||||||
this.ep = ep;
|
this.ep = ep;
|
||||||
@@ -30,6 +31,10 @@ class ConfirmCallSession extends CallSession {
|
|||||||
_clearResources() {
|
_clearResources() {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_callerHungup() {
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = ConfirmCallSession;
|
module.exports = ConfirmCallSession;
|
||||||
|
|||||||
@@ -16,7 +16,8 @@ class InboundCallSession extends CallSession {
|
|||||||
application: req.locals.application,
|
application: req.locals.application,
|
||||||
callInfo: req.locals.callInfo,
|
callInfo: req.locals.callInfo,
|
||||||
accountInfo: req.locals.accountInfo,
|
accountInfo: req.locals.accountInfo,
|
||||||
tasks: req.locals.application.tasks
|
tasks: req.locals.application.tasks,
|
||||||
|
rootSpan: req.locals.rootSpan
|
||||||
});
|
});
|
||||||
this.req = req;
|
this.req = req;
|
||||||
this.res = res;
|
this.res = res;
|
||||||
@@ -24,17 +25,27 @@ class InboundCallSession extends CallSession {
|
|||||||
req.once('cancel', this._onCancel.bind(this));
|
req.once('cancel', this._onCancel.bind(this));
|
||||||
|
|
||||||
this.on('callStatusChange', this._notifyCallStatusChange.bind(this));
|
this.on('callStatusChange', this._notifyCallStatusChange.bind(this));
|
||||||
this._notifyCallStatusChange({callStatus: CallStatus.Trying, sipStatus: 100});
|
this._notifyCallStatusChange({
|
||||||
|
callStatus: CallStatus.Trying,
|
||||||
|
sipStatus: 100,
|
||||||
|
sipReason: 'Trying'
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
_onCancel() {
|
_onCancel() {
|
||||||
this._notifyCallStatusChange({callStatus: CallStatus.NoAnswer, sipStatus: 487});
|
this.rootSpan.setAttributes({'call.termination': 'caller abandoned'});
|
||||||
|
this._notifyCallStatusChange({
|
||||||
|
callStatus: CallStatus.NoAnswer,
|
||||||
|
sipStatus: 487,
|
||||||
|
sipReason: 'Request Terminated'
|
||||||
|
});
|
||||||
this._callReleased();
|
this._callReleased();
|
||||||
}
|
}
|
||||||
|
|
||||||
_onTasksDone() {
|
_onTasksDone() {
|
||||||
if (!this.res.finalResponseSent) {
|
if (!this.res.finalResponseSent) {
|
||||||
if (this._mediaServerFailure) {
|
if (this._mediaServerFailure) {
|
||||||
|
this.rootSpan.setAttributes({'call.termination': 'media server failure'});
|
||||||
this.logger.info('InboundCallSession:_onTasksDone generating 480 due to media server failure');
|
this.logger.info('InboundCallSession:_onTasksDone generating 480 due to media server failure');
|
||||||
this.res.send(480, {
|
this.res.send(480, {
|
||||||
headers: {
|
headers: {
|
||||||
@@ -43,6 +54,7 @@ class InboundCallSession extends CallSession {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
this.rootSpan.setAttributes({'call.termination': 'tasks completed without answering call'});
|
||||||
this.logger.info('InboundCallSession:_onTasksDone auto-generating non-success response to invite');
|
this.logger.info('InboundCallSession:_onTasksDone auto-generating non-success response to invite');
|
||||||
this.res.send(603);
|
this.res.send(603);
|
||||||
}
|
}
|
||||||
@@ -56,8 +68,12 @@ class InboundCallSession extends CallSession {
|
|||||||
_callerHungup() {
|
_callerHungup() {
|
||||||
assert(this.dlg.connectTime);
|
assert(this.dlg.connectTime);
|
||||||
const duration = moment().diff(this.dlg.connectTime, 'seconds');
|
const duration = moment().diff(this.dlg.connectTime, 'seconds');
|
||||||
this.emit('callStatusChange', {callStatus: CallStatus.Completed, duration});
|
this.rootSpan.setAttributes({'call.termination': 'hangup by caller'});
|
||||||
this.logger.debug('InboundCallSession: caller hung up');
|
this.emit('callStatusChange', {
|
||||||
|
callStatus: CallStatus.Completed,
|
||||||
|
duration
|
||||||
|
});
|
||||||
|
this.logger.info('InboundCallSession: caller hung up');
|
||||||
this._callReleased();
|
this._callReleased();
|
||||||
this.req.removeAllListeners('cancel');
|
this.req.removeAllListeners('cancel');
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ const moment = require('moment');
|
|||||||
* @extends CallSession
|
* @extends CallSession
|
||||||
*/
|
*/
|
||||||
class RestCallSession extends CallSession {
|
class RestCallSession extends CallSession {
|
||||||
constructor({logger, application, srf, req, ep, tasks, callInfo, accountInfo}) {
|
constructor({logger, application, srf, req, ep, tasks, callInfo, accountInfo, rootSpan}) {
|
||||||
super({
|
super({
|
||||||
logger,
|
logger,
|
||||||
application,
|
application,
|
||||||
@@ -16,13 +16,18 @@ class RestCallSession extends CallSession {
|
|||||||
callSid: callInfo.callSid,
|
callSid: callInfo.callSid,
|
||||||
tasks,
|
tasks,
|
||||||
callInfo,
|
callInfo,
|
||||||
accountInfo
|
accountInfo,
|
||||||
|
rootSpan
|
||||||
});
|
});
|
||||||
this.req = req;
|
this.req = req;
|
||||||
this.ep = ep;
|
this.ep = ep;
|
||||||
|
|
||||||
this.on('callStatusChange', this._notifyCallStatusChange.bind(this));
|
this.on('callStatusChange', this._notifyCallStatusChange.bind(this));
|
||||||
this._notifyCallStatusChange({callStatus: CallStatus.Trying, sipStatus: 100});
|
this._notifyCallStatusChange({
|
||||||
|
callStatus: CallStatus.Trying,
|
||||||
|
sipStatus: 100,
|
||||||
|
sipReason: 'Trying'
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -1,248 +0,0 @@
|
|||||||
const Task = require('./task');
|
|
||||||
const {TaskName, TaskPreconditions} = require('../utils/constants');
|
|
||||||
const makeTask = require('./make_task');
|
|
||||||
const { SocketClient } = require('@cognigy/socket-client');
|
|
||||||
|
|
||||||
const parseGallery = (obj = {}) => {
|
|
||||||
const {_default} = obj;
|
|
||||||
if (_default) {
|
|
||||||
const {_gallery} = _default;
|
|
||||||
if (_gallery) return _gallery.fallbackText;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const parseQuickReplies = (obj) => {
|
|
||||||
const {_default} = obj;
|
|
||||||
if (_default) {
|
|
||||||
const {_quickReplies} = _default;
|
|
||||||
if (_quickReplies) return _quickReplies.text || _quickReplies.fallbackText;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const parseBotText = (evt) => {
|
|
||||||
const {text, data} = evt;
|
|
||||||
if (text) return text;
|
|
||||||
|
|
||||||
switch (data?.type) {
|
|
||||||
case 'quickReplies':
|
|
||||||
return parseQuickReplies(data?._cognigy);
|
|
||||||
case 'gallery':
|
|
||||||
return parseGallery(data?._cognigy);
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
class Cognigy extends Task {
|
|
||||||
constructor(logger, opts) {
|
|
||||||
super(logger, opts);
|
|
||||||
this.preconditions = TaskPreconditions.Endpoint;
|
|
||||||
|
|
||||||
this.url = this.data.url;
|
|
||||||
this.token = this.data.token;
|
|
||||||
this.prompt = this.data.prompt;
|
|
||||||
this.eventHook = this.data?.eventHook;
|
|
||||||
this.actionHook = this.data?.actionHook;
|
|
||||||
this.data = this.data.data || {};
|
|
||||||
this.prompts = [];
|
|
||||||
}
|
|
||||||
|
|
||||||
get name() { return TaskName.Cognigy; }
|
|
||||||
|
|
||||||
get hasReportedFinalAction() {
|
|
||||||
return this.reportedFinalAction || this.isReplacingApplication;
|
|
||||||
}
|
|
||||||
|
|
||||||
async exec(cs, ep) {
|
|
||||||
await super.exec(cs);
|
|
||||||
|
|
||||||
this.ep = ep;
|
|
||||||
try {
|
|
||||||
/* set event handlers and start transcribing */
|
|
||||||
this.on('transcription', this._onTranscription.bind(this, cs, ep));
|
|
||||||
this.on('error', this._onError.bind(this, cs, ep));
|
|
||||||
|
|
||||||
this.transcribeTask = this._makeTranscribeTask();
|
|
||||||
this.transcribeTask.exec(cs, ep, this)
|
|
||||||
.catch((err) => {
|
|
||||||
this.logger.info({err}, 'Cognigy transcribe task returned error');
|
|
||||||
this.notifyTaskDone();
|
|
||||||
});
|
|
||||||
if (this.prompt) {
|
|
||||||
this.sayTask = this._makeSayTask(this.prompt);
|
|
||||||
this.sayTask.exec(cs, ep, this)
|
|
||||||
.catch((err) => {
|
|
||||||
this.logger.info({err}, 'Cognigy say task returned error');
|
|
||||||
this.notifyTaskDone();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/* connect to the bot and send initial data */
|
|
||||||
this.client = new SocketClient(
|
|
||||||
this.url,
|
|
||||||
this.token,
|
|
||||||
{
|
|
||||||
sessionId: cs.callSid,
|
|
||||||
channel: 'jambonz',
|
|
||||||
forceWebsockets: true,
|
|
||||||
reconnection: true,
|
|
||||||
settings: {
|
|
||||||
enableTypingIndicator: false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
this.client.on('output', this._onBotUtterance.bind(this, cs, ep));
|
|
||||||
this.client.on('typingStatus', this._onBotTypingStatus.bind(this, cs, ep));
|
|
||||||
this.client.on('error', this._onBotError.bind(this, cs, ep));
|
|
||||||
this.client.on('finalPing', this._onBotFinalPing.bind(this, cs, ep));
|
|
||||||
await this.client.connect();
|
|
||||||
this.client.sendMessage('', {...this.data, ...cs.callInfo});
|
|
||||||
|
|
||||||
await this.awaitTaskDone();
|
|
||||||
} catch (err) {
|
|
||||||
this.logger.error({err}, 'Cognigy error');
|
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async kill(cs) {
|
|
||||||
super.kill(cs);
|
|
||||||
this.logger.debug('Cognigy:kill');
|
|
||||||
|
|
||||||
this.removeAllListeners();
|
|
||||||
this.transcribeTask && this.transcribeTask.kill();
|
|
||||||
|
|
||||||
this.client.removeAllListeners();
|
|
||||||
if (this.client && this.client.connected) this.client.disconnect();
|
|
||||||
|
|
||||||
if (!this.hasReportedFinalAction) {
|
|
||||||
this.reportedFinalAction = true;
|
|
||||||
this.performAction({cognigyResult: 'caller hungup'})
|
|
||||||
.catch((err) => this.logger.info({err}, 'cognigy - error w/ action webook'));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.ep.connected) {
|
|
||||||
await this.ep.api('uuid_break', this.ep.uuid).catch((err) => this.logger.info(err, 'Error killing audio'));
|
|
||||||
}
|
|
||||||
this.notifyTaskDone();
|
|
||||||
}
|
|
||||||
|
|
||||||
_makeTranscribeTask() {
|
|
||||||
const opts = {
|
|
||||||
recognizer: this.data.recognizer || {
|
|
||||||
vendor: 'default',
|
|
||||||
language: 'default',
|
|
||||||
outputFormat: 'detailed'
|
|
||||||
}
|
|
||||||
};
|
|
||||||
this.logger.debug({opts}, 'constructing a nested transcribe object');
|
|
||||||
const transcribe = makeTask(this.logger, {transcribe: opts}, this);
|
|
||||||
return transcribe;
|
|
||||||
}
|
|
||||||
|
|
||||||
_makeSayTask(text) {
|
|
||||||
const opts = {
|
|
||||||
text,
|
|
||||||
synthesizer: this.data.synthesizer ||
|
|
||||||
{
|
|
||||||
vendor: 'default',
|
|
||||||
language: 'default',
|
|
||||||
voice: 'default'
|
|
||||||
}
|
|
||||||
};
|
|
||||||
this.logger.debug({opts}, 'constructing a nested say object');
|
|
||||||
const say = makeTask(this.logger, {say: opts}, this);
|
|
||||||
return say;
|
|
||||||
}
|
|
||||||
|
|
||||||
async _onBotError(cs, ep, evt) {
|
|
||||||
this.logger.info({evt}, 'Cognigy:_onBotError');
|
|
||||||
this.performAction({cognigyResult: 'botError', message: evt.message });
|
|
||||||
this.reportedFinalAction = true;
|
|
||||||
this.notifyTaskDone();
|
|
||||||
}
|
|
||||||
|
|
||||||
async _onBotTypingStatus(cs, ep, evt) {
|
|
||||||
this.logger.info({evt}, 'Cognigy:_onBotTypingStatus');
|
|
||||||
}
|
|
||||||
async _onBotFinalPing(cs, ep) {
|
|
||||||
this.logger.info('Cognigy:_onBotFinalPing');
|
|
||||||
if (this.prompts.length) {
|
|
||||||
const text = this.prompts.join('.');
|
|
||||||
this.prompts = [];
|
|
||||||
if (text && !this.killed) {
|
|
||||||
this.sayTask = this._makeSayTask(text);
|
|
||||||
this.sayTask.exec(cs, ep, this)
|
|
||||||
.catch((err) => {
|
|
||||||
this.logger.info({err}, 'Cognigy say task returned error');
|
|
||||||
this.notifyTaskDone();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async _onBotUtterance(cs, ep, evt) {
|
|
||||||
this.logger.debug({evt}, 'Cognigy:_onBotUtterance');
|
|
||||||
|
|
||||||
if (this.eventHook) {
|
|
||||||
this.performHook(cs, this.eventHook, {event: 'botMessage', message: evt})
|
|
||||||
.then((redirected) => {
|
|
||||||
if (redirected) {
|
|
||||||
this.logger.info('Cognigy_onTranscription: event handler for bot message redirected us to new webhook');
|
|
||||||
this.reportedFinalAction = true;
|
|
||||||
this.performAction({cognigyResult: 'redirect'}, false);
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
})
|
|
||||||
.catch(({err}) => {
|
|
||||||
this.logger.info({err}, 'Cognigy_onTranscription: error sending event hook');
|
|
||||||
});
|
|
||||||
}
|
|
||||||
const text = parseBotText(evt);
|
|
||||||
this.prompts.push(text);
|
|
||||||
}
|
|
||||||
|
|
||||||
async _onTranscription(cs, ep, evt) {
|
|
||||||
this.logger.debug({evt}, `Cognigy: got transcription for callSid ${cs.callSid}`);
|
|
||||||
const utterance = evt.alternatives[0].transcript;
|
|
||||||
|
|
||||||
if (this.eventHook) {
|
|
||||||
this.performHook(cs, this.eventHook, {event: 'userMessage', message: utterance})
|
|
||||||
.then((redirected) => {
|
|
||||||
if (redirected) {
|
|
||||||
this.logger.info('Cognigy_onTranscription: event handler for user message redirected us to new webhook');
|
|
||||||
this.reportedFinalAction = true;
|
|
||||||
this.performAction({cognigyResult: 'redirect'}, false);
|
|
||||||
if (this.transcribeTask) this.transcribeTask.kill(cs);
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
})
|
|
||||||
.catch(({err}) => {
|
|
||||||
this.logger.info({err}, 'Cognigy_onTranscription: error sending event hook');
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/* send the user utterance to the bot */
|
|
||||||
try {
|
|
||||||
if (this.client && this.client.connected) {
|
|
||||||
this.client.sendMessage(utterance);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
this.logger.info('Cognigy_onTranscription - not sending user utterance as bot is disconnected');
|
|
||||||
}
|
|
||||||
} catch (err) {
|
|
||||||
this.logger.error({err}, 'Cognigy_onTranscription: Error sending user utterance to Cognigy - ending task');
|
|
||||||
this.performAction({cognigyResult: 'socketError'});
|
|
||||||
this.reportedFinalAction = true;
|
|
||||||
this.notifyTaskDone();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_onError(cs, ep, err) {
|
|
||||||
this.logger.debug({err}, 'Cognigy: got error');
|
|
||||||
if (!this.hasReportedFinalAction) this.performAction({cognigyResult: 'error', err});
|
|
||||||
this.reportedFinalAction = true;
|
|
||||||
this.notifyTaskDone();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = Cognigy;
|
|
||||||
@@ -529,7 +529,9 @@ class Conference extends Task {
|
|||||||
|
|
||||||
async _playHook(cs, dlg, hook, allowed = [TaskName.Play, TaskName.Say, TaskName.Pause]) {
|
async _playHook(cs, dlg, hook, allowed = [TaskName.Play, TaskName.Say, TaskName.Pause]) {
|
||||||
assert(!this._playSession);
|
assert(!this._playSession);
|
||||||
const json = await cs.application.requestor.request(hook, cs.callInfo);
|
const b3 = this.getTracingPropagation();
|
||||||
|
const httpHeaders = b3 && {b3};
|
||||||
|
const json = await cs.application.requestor.request('verb:hook', hook, cs.callInfo, httpHeaders);
|
||||||
const tasks = normalizeJambones(this.logger, json).map((tdata) => makeTask(this.logger, tdata));
|
const tasks = normalizeJambones(this.logger, json).map((tdata) => makeTask(this.logger, tdata));
|
||||||
|
|
||||||
const allowedTasks = tasks.filter((t) => allowed.includes(t.name));
|
const allowedTasks = tasks.filter((t) => allowed.includes(t.name));
|
||||||
@@ -582,11 +584,14 @@ class Conference extends Task {
|
|||||||
|
|
||||||
_notifyConferenceEvent(cs, eventName, params = {}) {
|
_notifyConferenceEvent(cs, eventName, params = {}) {
|
||||||
if (this.statusEvents.includes(eventName)) {
|
if (this.statusEvents.includes(eventName)) {
|
||||||
|
const b3 = this.getTracingPropagation();
|
||||||
|
const httpHeaders = b3 && {b3};
|
||||||
params.event = eventName;
|
params.event = eventName;
|
||||||
params.duration = (Date.now() - this.conferenceStartTime.getTime()) / 1000;
|
params.duration = (Date.now() - this.conferenceStartTime.getTime()) / 1000;
|
||||||
if (!params.time) params.time = (new Date()).toISOString();
|
if (!params.time) params.time = (new Date()).toISOString();
|
||||||
if (!params.members && typeof this.participantCount === 'number') params.members = this.participantCount;
|
if (!params.members && typeof this.participantCount === 'number') params.members = this.participantCount;
|
||||||
cs.application.requestor.request(this.statusHook, Object.assign(params, this.statusParams))
|
cs.application.requestor
|
||||||
|
.request('verb:hook', this.statusHook, Object.assign(params, this.statusParams, httpHeaders))
|
||||||
.catch((err) => this.logger.info(err, 'Conference:notifyConferenceEvent - error'));
|
.catch((err) => this.logger.info(err, 'Conference:notifyConferenceEvent - error'));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
102
lib/tasks/config.js
Normal file
102
lib/tasks/config.js
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
const Task = require('./task');
|
||||||
|
const {TaskName, TaskPreconditions} = require('../utils/constants');
|
||||||
|
|
||||||
|
class TaskConfig extends Task {
|
||||||
|
constructor(logger, opts) {
|
||||||
|
super(logger, opts);
|
||||||
|
this.preconditions = TaskPreconditions.Endpoint;
|
||||||
|
|
||||||
|
[
|
||||||
|
'synthesizer',
|
||||||
|
'recognizer',
|
||||||
|
'bargeIn'
|
||||||
|
].forEach((k) => this[k] = this.data[k] || {});
|
||||||
|
|
||||||
|
if (this.bargeIn.enable) {
|
||||||
|
this.gatherOpts = {
|
||||||
|
verb: 'gather',
|
||||||
|
timeout: 0,
|
||||||
|
bargein: true,
|
||||||
|
input: ['speech']
|
||||||
|
};
|
||||||
|
[
|
||||||
|
'finishOnKey', 'input', 'numDigits', 'minDigits', 'maxDigits',
|
||||||
|
'interDigitTimeout', 'bargein', 'dtmfBargein', 'minBargeinWordCount', 'actionHook'
|
||||||
|
].forEach((k) => {
|
||||||
|
if (this.bargeIn[k]) this.gatherOpts[k] = this.bargeIn[k];
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (this.bargeIn.sticky) this.autoEnable = true;
|
||||||
|
this.preconditions = this.bargeIn.enable ? TaskPreconditions.Endpoint : TaskPreconditions.None;
|
||||||
|
}
|
||||||
|
|
||||||
|
get name() { return TaskName.Config; }
|
||||||
|
|
||||||
|
get hasSynthesizer() { return Object.keys(this.synthesizer).length; }
|
||||||
|
|
||||||
|
get hasRecognizer() { return Object.keys(this.recognizer).length; }
|
||||||
|
|
||||||
|
get summary() {
|
||||||
|
const phrase = [];
|
||||||
|
if (this.bargeIn.enable) phrase.push('enable barge-in');
|
||||||
|
if (this.hasSynthesizer) {
|
||||||
|
const {vendor:v, language:l, voice} = this.synthesizer;
|
||||||
|
const s = `{${v},${l},${voice}}`;
|
||||||
|
phrase.push(`set synthesizer${s}`);
|
||||||
|
}
|
||||||
|
if (this.hasRecognizer) {
|
||||||
|
const {vendor:v, language:l} = this.recognizer;
|
||||||
|
const s = `{${v},${l}}`;
|
||||||
|
phrase.push(`set recognizer${s}`);
|
||||||
|
}
|
||||||
|
return `${this.name}{${phrase.join(',')}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
async exec(cs) {
|
||||||
|
await super.exec(cs);
|
||||||
|
|
||||||
|
if (this.hasSynthesizer) {
|
||||||
|
cs.speechSynthesisVendor = this.synthesizer.vendor !== 'default'
|
||||||
|
? this.synthesizer.vendor
|
||||||
|
: cs.speechSynthesisVendor;
|
||||||
|
cs.speechSynthesisLanguage = this.synthesizer.language !== 'default'
|
||||||
|
? this.synthesizer.language
|
||||||
|
: cs.speechSynthesisLanguage;
|
||||||
|
cs.speechSynthesisVoice = this.synthesizer.voice !== 'default'
|
||||||
|
? this.synthesizer.voice
|
||||||
|
: cs.speechSynthesisVoice;
|
||||||
|
this.logger.info({synthesizer: this.synthesizer}, 'Config: updated synthesizer');
|
||||||
|
}
|
||||||
|
if (this.hasRecognizer) {
|
||||||
|
cs.speechRecognizerVendor = this.recognizer.vendor !== 'default'
|
||||||
|
? this.recognizer.vendor
|
||||||
|
: cs.speechRecognizerVendor;
|
||||||
|
cs.speechRecognizerLanguage = this.recognizer.language !== 'default'
|
||||||
|
? this.recognizer.language
|
||||||
|
: cs.speechRecognizerLanguage;
|
||||||
|
this.logger.info({recognizer: this.recognizer}, 'Config: updated recognizer');
|
||||||
|
}
|
||||||
|
if ('enable' in this.bargeIn) {
|
||||||
|
if (this.gatherOpts) {
|
||||||
|
this.gatherOpts.recognizer = this.hasRecognizer ?
|
||||||
|
this.recognizer :
|
||||||
|
{
|
||||||
|
vendor: cs.speechRecognizerVendor,
|
||||||
|
language: cs.speechRecognizerLanguage
|
||||||
|
};
|
||||||
|
this.logger.info({opts: this.gatherOpts}, 'Config: enabling bargeIn');
|
||||||
|
cs.enableBotMode(this.gatherOpts, this.autoEnable);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
this.logger.info('Config: disabling bargeIn');
|
||||||
|
cs.disableBotMode();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async kill(cs) {
|
||||||
|
super.kill(cs);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = TaskConfig;
|
||||||
@@ -14,6 +14,7 @@ const sessionTracker = require('../session/session-tracker');
|
|||||||
const DtmfCollector = require('../utils/dtmf-collector');
|
const DtmfCollector = require('../utils/dtmf-collector');
|
||||||
const dbUtils = require('../utils/db-utils');
|
const dbUtils = require('../utils/db-utils');
|
||||||
const debug = require('debug')('jambonz:feature-server');
|
const debug = require('debug')('jambonz:feature-server');
|
||||||
|
const {parseUri} = require('drachtio-srf');
|
||||||
|
|
||||||
function parseDtmfOptions(logger, dtmfCapture) {
|
function parseDtmfOptions(logger, dtmfCapture) {
|
||||||
let parentDtmfCollector, childDtmfCollector;
|
let parentDtmfCollector, childDtmfCollector;
|
||||||
@@ -91,6 +92,7 @@ class TaskDial extends Task {
|
|||||||
this.timeLimit = this.data.timeLimit;
|
this.timeLimit = this.data.timeLimit;
|
||||||
this.confirmHook = this.data.confirmHook;
|
this.confirmHook = this.data.confirmHook;
|
||||||
this.confirmMethod = this.data.confirmMethod;
|
this.confirmMethod = this.data.confirmMethod;
|
||||||
|
this.referHook = this.data.referHook;
|
||||||
this.dtmfHook = this.data.dtmfHook;
|
this.dtmfHook = this.data.dtmfHook;
|
||||||
this.proxy = this.data.proxy;
|
this.proxy = this.data.proxy;
|
||||||
|
|
||||||
@@ -135,6 +137,24 @@ class TaskDial extends Task {
|
|||||||
return !process.env.ANCHOR_MEDIA_ALWAYS && !this.listenTask && !this.transcribeTask;
|
return !process.env.ANCHOR_MEDIA_ALWAYS && !this.listenTask && !this.transcribeTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get summary() {
|
||||||
|
if (this.target.length === 1) {
|
||||||
|
const target = this.target[0];
|
||||||
|
switch (target.type) {
|
||||||
|
case 'phone':
|
||||||
|
case 'teams':
|
||||||
|
return `${this.name}{type=${target.type},number=${target.number}}`;
|
||||||
|
case 'user':
|
||||||
|
return `${this.name}{type=${target.type},name=${target.name}}`;
|
||||||
|
case 'sip':
|
||||||
|
return `${this.name}{type=${target.type},sipUri=${target.sipUri}}`;
|
||||||
|
default:
|
||||||
|
return `${this.name}`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else return `${this.name}{${this.target.length} targets}`;
|
||||||
|
}
|
||||||
|
|
||||||
async exec(cs) {
|
async exec(cs) {
|
||||||
await super.exec(cs);
|
await super.exec(cs);
|
||||||
try {
|
try {
|
||||||
@@ -206,7 +226,11 @@ class TaskDial extends Task {
|
|||||||
this.logger.debug('Dial:whisper executing tasks');
|
this.logger.debug('Dial:whisper executing tasks');
|
||||||
while (tasks.length && !cs.callGone) {
|
while (tasks.length && !cs.callGone) {
|
||||||
const task = tasks.shift();
|
const task = tasks.shift();
|
||||||
|
const {span, ctx} = this.startChildSpan(`whisper:${this.sayTask.summary}`);
|
||||||
|
task.span = span;
|
||||||
|
task.ctx = ctx;
|
||||||
await task.exec(cs, callSid === this.callSid ? this.ep : this.epOther);
|
await task.exec(cs, callSid === this.callSid ? this.ep : this.epOther);
|
||||||
|
span.end();
|
||||||
}
|
}
|
||||||
this.logger.debug('Dial:whisper tasks complete');
|
this.logger.debug('Dial:whisper tasks complete');
|
||||||
if (!cs.callGone && this.epOther) {
|
if (!cs.callGone && this.epOther) {
|
||||||
@@ -240,6 +264,43 @@ class TaskDial extends Task {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async handleRefer(cs, req, res, callInfo = cs.callInfo) {
|
||||||
|
if (this.referHook) {
|
||||||
|
try {
|
||||||
|
const isChild = !!callInfo.parentCallSid;
|
||||||
|
const referring_call_sid = isChild ? callInfo.callSid : cs.callSid;
|
||||||
|
const referred_call_sid = isChild ? callInfo.parentCallSid : this.sd.callSid;
|
||||||
|
|
||||||
|
const b3 = this.getTracingPropagation();
|
||||||
|
const httpHeaders = b3 && {b3};
|
||||||
|
|
||||||
|
const to = parseUri(req.getParsedHeader('Refer-To').uri);
|
||||||
|
const by = parseUri(req.getParsedHeader('Referred-By').uri);
|
||||||
|
this.logger.info({to}, 'refer to parsed');
|
||||||
|
await cs.requestor.request('verb:hook', this.referHook, {
|
||||||
|
...callInfo,
|
||||||
|
refer_details: {
|
||||||
|
sip_refer_to: req.get('Refer-To'),
|
||||||
|
sip_referred_by: req.get('Referred-By'),
|
||||||
|
sip_user_agent: req.get('User-Agent'),
|
||||||
|
refer_to_user: to.user,
|
||||||
|
referred_by_user: by.user,
|
||||||
|
referring_call_sid,
|
||||||
|
referred_call_sid
|
||||||
|
}
|
||||||
|
}, httpHeaders);
|
||||||
|
res.send(202);
|
||||||
|
this.logger.info('DialTask:handleRefer - sent 202 Accepted');
|
||||||
|
} catch (err) {
|
||||||
|
res.send(err.statusCode || 501);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
this.logger.info('DialTask:handleRefer - got REFER but no referHook, responding 501');
|
||||||
|
res.send(501);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
_removeHandlers(sd) {
|
_removeHandlers(sd) {
|
||||||
sd.removeAllListeners('accept');
|
sd.removeAllListeners('accept');
|
||||||
sd.removeAllListeners('decline');
|
sd.removeAllListeners('decline');
|
||||||
@@ -287,8 +348,10 @@ class TaskDial extends Task {
|
|||||||
const key = arr[1];
|
const key = arr[1];
|
||||||
const match = dtmfDetector.keyPress(key);
|
const match = dtmfDetector.keyPress(key);
|
||||||
if (match) {
|
if (match) {
|
||||||
|
const b3 = this.getTracingPropagation();
|
||||||
|
const httpHeaders = b3 && {b3};
|
||||||
this.logger.info({callSid}, `Dial:_onInfo triggered dtmf match: ${match}`);
|
this.logger.info({callSid}, `Dial:_onInfo triggered dtmf match: ${match}`);
|
||||||
requestor.request(this.dtmfHook, {dtmf: match, ...callInfo.toJSON()})
|
requestor.request('verb:hook', this.dtmfHook, {dtmf: match, ...callInfo.toJSON()}, httpHeaders)
|
||||||
.catch((err) => this.logger.info(err, 'Dial:_onDtmf - error'));
|
.catch((err) => this.logger.info(err, 'Dial:_onDtmf - error'));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -343,10 +406,11 @@ class TaskDial extends Task {
|
|||||||
this._killOutdials();
|
this._killOutdials();
|
||||||
}, this.timeout * 1000);
|
}, this.timeout * 1000);
|
||||||
|
|
||||||
|
this.span.setAttributes({'dial.target': JSON.stringify(this.target)});
|
||||||
this.target.forEach(async(t) => {
|
this.target.forEach(async(t) => {
|
||||||
try {
|
try {
|
||||||
t.url = t.url || this.confirmUrl;
|
t.confirmHook = t.confirmHook || this.confirmHook;
|
||||||
t.method = t.method || this.confirmMethod || 'POST';
|
//t.method = t.method || this.confirmMethod || 'POST';
|
||||||
if (t.type === 'teams') t.teamsInfo = teamsInfo;
|
if (t.type === 'teams') t.teamsInfo = teamsInfo;
|
||||||
if (t.type === 'user' && !t.name.includes('@') && !fqdn) {
|
if (t.type === 'user' && !t.name.includes('@') && !fqdn) {
|
||||||
const user = t.name;
|
const user = t.name;
|
||||||
@@ -379,11 +443,14 @@ class TaskDial extends Task {
|
|||||||
target: t,
|
target: t,
|
||||||
opts,
|
opts,
|
||||||
callInfo: cs.callInfo,
|
callInfo: cs.callInfo,
|
||||||
accountInfo: cs.accountInfo
|
accountInfo: cs.accountInfo,
|
||||||
|
rootSpan: cs.rootSpan,
|
||||||
|
startSpan: this.startSpan.bind(this)
|
||||||
});
|
});
|
||||||
this.dials.set(sd.callSid, sd);
|
this.dials.set(sd.callSid, sd);
|
||||||
|
|
||||||
sd
|
sd
|
||||||
|
.on('refer', (callInfo, req, res) => this.handleRefer(cs, req, res, callInfo))
|
||||||
.on('callCreateFail', () => {
|
.on('callCreateFail', () => {
|
||||||
clearTimeout(this.timerRing);
|
clearTimeout(this.timerRing);
|
||||||
this.dials.delete(sd.callSid);
|
this.dials.delete(sd.callSid);
|
||||||
@@ -452,6 +519,9 @@ class TaskDial extends Task {
|
|||||||
} catch (err) {
|
} catch (err) {
|
||||||
this.logger.error(err, 'Error in dial einvite from B leg');
|
this.logger.error(err, 'Error in dial einvite from B leg');
|
||||||
}
|
}
|
||||||
|
})
|
||||||
|
.on('refer', (callInfo, req, res) => {
|
||||||
|
|
||||||
})
|
})
|
||||||
.once('adulting', () => {
|
.once('adulting', () => {
|
||||||
/* child call just adulted and got its own session */
|
/* child call just adulted and got its own session */
|
||||||
@@ -536,8 +606,8 @@ class TaskDial extends Task {
|
|||||||
if (this.parentDtmfCollector) this._installDtmfDetection(cs, cs.dlg);
|
if (this.parentDtmfCollector) this._installDtmfDetection(cs, cs.dlg);
|
||||||
if (this.childDtmfCollector) this._installDtmfDetection(cs, this.dlg);
|
if (this.childDtmfCollector) this._installDtmfDetection(cs, this.dlg);
|
||||||
|
|
||||||
if (this.transcribeTask) this.transcribeTask.exec(cs, this.ep);
|
if (this.transcribeTask) this.transcribeTask.exec(cs, this.epOther, this.ep);
|
||||||
if (this.listenTask) this.listenTask.exec(cs, this.ep);
|
if (this.listenTask) this.listenTask.exec(cs, this.epOther);
|
||||||
|
|
||||||
/* if we can release the media back to the SBC, do so now */
|
/* if we can release the media back to the SBC, do so now */
|
||||||
if (this.canReleaseMedia) this._releaseMedia(cs, sd);
|
if (this.canReleaseMedia) this._releaseMedia(cs, sd);
|
||||||
|
|||||||
@@ -295,9 +295,9 @@ class Dialogflow extends Task {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// if a final transcription, start a typing sound
|
// if a final transcription, start a typing sound
|
||||||
if (this.thinkingSound > 0 && !transcription.isEmpty && transcription.isFinal &&
|
if (this.thinkingMusic && !transcription.isEmpty && transcription.isFinal &&
|
||||||
transcription.confidence > 0.8) {
|
transcription.confidence > 0.8) {
|
||||||
ep.play(this.data.thinkingSound).catch((err) => this.logger.info(err, 'Error playing typing sound'));
|
ep.play(this.data.thinkingMusic).catch((err) => this.logger.info(err, 'Error playing typing sound'));
|
||||||
}
|
}
|
||||||
|
|
||||||
// interrupt playback on speaking if bargein = true
|
// interrupt playback on speaking if bargein = true
|
||||||
@@ -405,8 +405,8 @@ class Dialogflow extends Task {
|
|||||||
this.dtmfEntry = dtmfEntry;
|
this.dtmfEntry = dtmfEntry;
|
||||||
this.digitBuffer = null;
|
this.digitBuffer = null;
|
||||||
// if a final transcription, start a typing sound
|
// if a final transcription, start a typing sound
|
||||||
if (this.thinkingSound > 0) {
|
if (this.thinkingMusic) {
|
||||||
ep.play(this.thinkingSound).catch((err) => this.logger.info(err, 'Error playing typing sound'));
|
ep.play(this.thinkingMusic).catch((err) => this.logger.info(err, 'Error playing typing sound'));
|
||||||
}
|
}
|
||||||
|
|
||||||
// kill the current dialogflow, which will result in us getting an immediate intent
|
// kill the current dialogflow, which will result in us getting an immediate intent
|
||||||
@@ -453,7 +453,10 @@ class Dialogflow extends Task {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async _performHook(cs, hook, results = {}) {
|
async _performHook(cs, hook, results = {}) {
|
||||||
const json = await this.cs.requestor.request(hook, {...results, ...cs.callInfo.toJSON()});
|
const b3 = this.getTracingPropagation();
|
||||||
|
const httpHeaders = b3 && {b3};
|
||||||
|
const json = await this.cs.requestor.request('verb:hook', hook,
|
||||||
|
{...results, ...cs.callInfo.toJSON()}, httpHeaders);
|
||||||
if (json && Array.isArray(json)) {
|
if (json && Array.isArray(json)) {
|
||||||
const makeTask = require('../make_task');
|
const makeTask = require('../make_task');
|
||||||
const tasks = normalizeJambones(this.logger, json).map((tdata) => makeTask(this.logger, tdata));
|
const tasks = normalizeJambones(this.logger, json).map((tdata) => makeTask(this.logger, tdata));
|
||||||
|
|||||||
@@ -302,6 +302,8 @@ class TaskEnqueue extends Task {
|
|||||||
|
|
||||||
async _playHook(cs, dlg, hook, allowed = [TaskName.Play, TaskName.Say, TaskName.Pause, TaskName.Leave]) {
|
async _playHook(cs, dlg, hook, allowed = [TaskName.Play, TaskName.Say, TaskName.Pause, TaskName.Leave]) {
|
||||||
const {lengthOfList, getListPosition} = cs.srf.locals.dbHelpers;
|
const {lengthOfList, getListPosition} = cs.srf.locals.dbHelpers;
|
||||||
|
const b3 = this.getTracingPropagation();
|
||||||
|
const httpHeaders = b3 && {b3};
|
||||||
|
|
||||||
assert(!this._playSession);
|
assert(!this._playSession);
|
||||||
if (this.killed) return [];
|
if (this.killed) return [];
|
||||||
@@ -317,7 +319,7 @@ class TaskEnqueue extends Task {
|
|||||||
} catch (err) {
|
} catch (err) {
|
||||||
this.logger.error({err}, `TaskEnqueue:_playHook error retrieving list info for queue ${this.queueName}`);
|
this.logger.error({err}, `TaskEnqueue:_playHook error retrieving list info for queue ${this.queueName}`);
|
||||||
}
|
}
|
||||||
const json = await cs.application.requestor.request(hook, params);
|
const json = await cs.application.requestor.request('verb:hook', hook, params, httpHeaders);
|
||||||
const tasks = normalizeJambones(this.logger, json).map((tdata) => makeTask(this.logger, tdata));
|
const tasks = normalizeJambones(this.logger, json).map((tdata) => makeTask(this.logger, tdata));
|
||||||
|
|
||||||
const allowedTasks = tasks.filter((t) => allowed.includes(t.name));
|
const allowedTasks = tasks.filter((t) => allowed.includes(t.name));
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ const {
|
|||||||
|
|
||||||
const makeTask = require('./make_task');
|
const makeTask = require('./make_task');
|
||||||
const assert = require('assert');
|
const assert = require('assert');
|
||||||
|
//const GATHER_STABILITY_THRESHOLD = Number(process.env.JAMBONZ_GATHER_STABILITY_THRESHOLD || 0.7);
|
||||||
|
|
||||||
class TaskGather extends Task {
|
class TaskGather extends Task {
|
||||||
constructor(logger, opts, parentTask) {
|
constructor(logger, opts, parentTask) {
|
||||||
@@ -16,18 +17,36 @@ class TaskGather extends Task {
|
|||||||
this.preconditions = TaskPreconditions.Endpoint;
|
this.preconditions = TaskPreconditions.Endpoint;
|
||||||
|
|
||||||
[
|
[
|
||||||
'finishOnKey', 'hints', 'input', 'numDigits',
|
'finishOnKey', 'hints', 'input', 'numDigits', 'minDigits', 'maxDigits',
|
||||||
'partialResultHook',
|
'interDigitTimeout', 'partialResultHook', 'bargein', 'dtmfBargein',
|
||||||
'speechTimeout', 'timeout', 'say', 'play'
|
'speechTimeout', 'timeout', 'say', 'play'
|
||||||
].forEach((k) => this[k] = this.data[k]);
|
].forEach((k) => this[k] = this.data[k]);
|
||||||
|
|
||||||
this.timeout = (this.timeout || 5) * 1000;
|
/* when collecting dtmf, bargein on dtmf is true unless explicitly set to false */
|
||||||
this.interim = this.partialResultCallback;
|
if (this.dtmfBargein !== false && this.input.includes('digits')) this.dtmfBargein = true;
|
||||||
|
|
||||||
|
/* timeout of zero means no timeout */
|
||||||
|
this.timeout = this.timeout === 0 ? 0 : (this.timeout || 15) * 1000;
|
||||||
|
this.interim = this.partialResultHook || this.bargein;
|
||||||
|
this.listenDuringPrompt = this.data.listenDuringPrompt === false ? false : true;
|
||||||
|
this.minBargeinWordCount = this.data.minBargeinWordCount || 0;
|
||||||
if (this.data.recognizer) {
|
if (this.data.recognizer) {
|
||||||
const recognizer = this.data.recognizer;
|
const recognizer = this.data.recognizer;
|
||||||
this.vendor = recognizer.vendor;
|
this.vendor = recognizer.vendor;
|
||||||
this.language = recognizer.language;
|
this.language = recognizer.language;
|
||||||
this.hints = recognizer.hints || [];
|
this.hints = recognizer.hints || [];
|
||||||
|
this.hintsBoost = recognizer.hintsBoost;
|
||||||
|
this.profanityFilter = recognizer.profanityFilter;
|
||||||
|
this.punctuation = !!recognizer.punctuation;
|
||||||
|
this.enhancedModel = !!recognizer.enhancedModel;
|
||||||
|
this.model = recognizer.model || 'command_and_search';
|
||||||
|
this.words = !!recognizer.words;
|
||||||
|
this.singleUtterance = recognizer.singleUtterance || true;
|
||||||
|
this.diarization = !!recognizer.diarization;
|
||||||
|
this.diarizationMinSpeakers = recognizer.diarizationMinSpeakers || 0;
|
||||||
|
this.diarizationMaxSpeakers = recognizer.diarizationMaxSpeakers || 0;
|
||||||
|
this.interactionType = recognizer.interactionType || 'unspecified';
|
||||||
|
this.naicsCode = recognizer.naicsCode || 0;
|
||||||
this.altLanguages = recognizer.altLanguages || [];
|
this.altLanguages = recognizer.altLanguages || [];
|
||||||
|
|
||||||
/* vad: if provided, we dont connect to recognizer until voice activity is detected */
|
/* vad: if provided, we dont connect to recognizer until voice activity is detected */
|
||||||
@@ -44,13 +63,19 @@ class TaskGather extends Task {
|
|||||||
this.profanityOption = recognizer.profanityOption || 'raw';
|
this.profanityOption = recognizer.profanityOption || 'raw';
|
||||||
this.requestSnr = recognizer.requestSnr || false;
|
this.requestSnr = recognizer.requestSnr || false;
|
||||||
this.initialSpeechTimeoutMs = recognizer.initialSpeechTimeoutMs || 0;
|
this.initialSpeechTimeoutMs = recognizer.initialSpeechTimeoutMs || 0;
|
||||||
|
this.azureServiceEndpoint = recognizer.azureServiceEndpoint;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.digitBuffer = '';
|
this.digitBuffer = '';
|
||||||
this._earlyMedia = this.data.earlyMedia === true;
|
this._earlyMedia = this.data.earlyMedia === true;
|
||||||
|
|
||||||
if (this.say) this.sayTask = makeTask(this.logger, {say: this.say}, this);
|
if (this.say) {
|
||||||
if (this.play) this.playTask = makeTask(this.logger, {play: this.play}, this);
|
this.sayTask = makeTask(this.logger, {say: this.say}, this);
|
||||||
|
}
|
||||||
|
if (this.play) {
|
||||||
|
this.playTask = makeTask(this.logger, {play: this.play}, this);
|
||||||
|
}
|
||||||
|
if (!this.sayTask && !this.playTask) this.listenDuringPrompt = false;
|
||||||
|
|
||||||
this.parentTask = parentTask;
|
this.parentTask = parentTask;
|
||||||
}
|
}
|
||||||
@@ -64,7 +89,23 @@ class TaskGather extends Task {
|
|||||||
(this.playTask && this.playTask.earlyMedia);
|
(this.playTask && this.playTask.earlyMedia);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get summary() {
|
||||||
|
let s = `${this.name}{`;
|
||||||
|
if (this.input.length === 2) s += 'inputs=[speech,digits],';
|
||||||
|
else if (this.input.includes('digits')) s += 'inputs=digits';
|
||||||
|
else s += 'inputs=speech,';
|
||||||
|
|
||||||
|
if (this.input.includes('speech')) {
|
||||||
|
s += `vendor=${this.vendor || 'default'},language=${this.language || 'default'}`;
|
||||||
|
}
|
||||||
|
if (this.sayTask) s += ',with nested say task';
|
||||||
|
if (this.playTask) s += ',with nested play task';
|
||||||
|
s += '}';
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
async exec(cs, ep) {
|
async exec(cs, ep) {
|
||||||
|
this.logger.debug('Gather:exec');
|
||||||
await super.exec(cs);
|
await super.exec(cs);
|
||||||
const {updateSpeechCredentialLastUsed} = require('../utils/db-utils')(this.logger, cs.srf);
|
const {updateSpeechCredentialLastUsed} = require('../utils/db-utils')(this.logger, cs.srf);
|
||||||
|
|
||||||
@@ -84,29 +125,53 @@ class TaskGather extends Task {
|
|||||||
throw new Error(`no speech-to-text service credentials for ${this.vendor} have been configured`);
|
throw new Error(`no speech-to-text service credentials for ${this.vendor} have been configured`);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const startListening = (cs, ep) => {
|
||||||
|
this._startTimer();
|
||||||
|
if (this.input.includes('speech') && !this.listenDuringPrompt) {
|
||||||
|
this._initSpeech(cs, ep)
|
||||||
|
.then(() => {
|
||||||
|
this._startTranscribing(ep);
|
||||||
|
return updateSpeechCredentialLastUsed(this.sttCredentials.speech_credential_sid);
|
||||||
|
})
|
||||||
|
.catch(() => {});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
try {
|
try {
|
||||||
if (this.sayTask) {
|
if (this.sayTask) {
|
||||||
|
const {span, ctx} = this.startChildSpan(`nested:${this.sayTask.summary}`);
|
||||||
|
this.sayTask.span = span;
|
||||||
|
this.sayTask.ctx = ctx;
|
||||||
this.sayTask.exec(cs, ep); // kicked off, _not_ waiting for it to complete
|
this.sayTask.exec(cs, ep); // kicked off, _not_ waiting for it to complete
|
||||||
this.sayTask.on('playDone', (err) => {
|
this.sayTask.on('playDone', (err) => {
|
||||||
if (!this.killed) this._startTimer();
|
span.end();
|
||||||
|
if (err) this.logger.error({err}, 'Gather:exec Error playing tts');
|
||||||
|
this.logger.debug('Gather: nested say task completed');
|
||||||
|
if (!this.killed) startListening(cs, ep);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
else if (this.playTask) {
|
else if (this.playTask) {
|
||||||
|
const {span, ctx} = this.startChildSpan(`nested:${this.playTask.summary}`);
|
||||||
|
this.playTask.span = span;
|
||||||
|
this.playTask.ctx = ctx;
|
||||||
this.playTask.exec(cs, ep); // kicked off, _not_ waiting for it to complete
|
this.playTask.exec(cs, ep); // kicked off, _not_ waiting for it to complete
|
||||||
this.playTask.on('playDone', (err) => {
|
this.playTask.on('playDone', (err) => {
|
||||||
if (!this.killed) this._startTimer();
|
span.end();
|
||||||
|
if (err) this.logger.error({err}, 'Gather:exec Error playing url');
|
||||||
|
this.logger.debug('Gather: nested play task completed');
|
||||||
|
if (!this.killed) startListening(cs, ep);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
else this._startTimer();
|
else startListening(cs, ep);
|
||||||
|
|
||||||
if (this.input.includes('speech')) {
|
if (this.input.includes('speech') && this.listenDuringPrompt) {
|
||||||
await this._initSpeech(cs, ep);
|
await this._initSpeech(cs, ep);
|
||||||
this._startTranscribing(ep);
|
this._startTranscribing(ep);
|
||||||
updateSpeechCredentialLastUsed(this.sttCredentials.speech_credential_sid)
|
updateSpeechCredentialLastUsed(this.sttCredentials.speech_credential_sid)
|
||||||
.catch(() => {/*already logged error */});
|
.catch(() => {/*already logged error */});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.input.includes('digits')) {
|
if (this.input.includes('digits') || this.dtmfBargein) {
|
||||||
ep.on('dtmf', this._onDtmf.bind(this, cs, ep));
|
ep.on('dtmf', this._onDtmf.bind(this, cs, ep));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -116,55 +181,99 @@ class TaskGather extends Task {
|
|||||||
}
|
}
|
||||||
ep.removeCustomEventListener(GoogleTranscriptionEvents.Transcription);
|
ep.removeCustomEventListener(GoogleTranscriptionEvents.Transcription);
|
||||||
ep.removeCustomEventListener(GoogleTranscriptionEvents.EndOfUtterance);
|
ep.removeCustomEventListener(GoogleTranscriptionEvents.EndOfUtterance);
|
||||||
|
ep.removeCustomEventListener(GoogleTranscriptionEvents.VadDetected);
|
||||||
ep.removeCustomEventListener(AwsTranscriptionEvents.Transcription);
|
ep.removeCustomEventListener(AwsTranscriptionEvents.Transcription);
|
||||||
|
ep.removeCustomEventListener(AwsTranscriptionEvents.VadDetected);
|
||||||
ep.removeCustomEventListener(AzureTranscriptionEvents.Transcription);
|
ep.removeCustomEventListener(AzureTranscriptionEvents.Transcription);
|
||||||
ep.removeCustomEventListener(AzureTranscriptionEvents.NoSpeechDetected);
|
ep.removeCustomEventListener(AzureTranscriptionEvents.NoSpeechDetected);
|
||||||
|
ep.removeCustomEventListener(AzureTranscriptionEvents.VadDetected);
|
||||||
}
|
}
|
||||||
|
|
||||||
kill(cs) {
|
kill(cs) {
|
||||||
super.kill(cs);
|
super.kill(cs);
|
||||||
this._killAudio(cs);
|
this._killAudio(cs);
|
||||||
this.ep.removeAllListeners('dtmf');
|
this.ep.removeAllListeners('dtmf');
|
||||||
|
clearTimeout(this.interDigitTimer);
|
||||||
|
this.playTask?.span.end();
|
||||||
|
this.sayTask?.span.end();
|
||||||
this._resolve('killed');
|
this._resolve('killed');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
updateTimeout(timeout) {
|
||||||
|
this.logger.info(`TaskGather:updateTimeout - updating timeout to ${timeout}`);
|
||||||
|
this.timeout = timeout;
|
||||||
|
this._startTimer();
|
||||||
|
}
|
||||||
|
|
||||||
_onDtmf(cs, ep, evt) {
|
_onDtmf(cs, ep, evt) {
|
||||||
this.logger.debug(evt, 'TaskGather:_onDtmf');
|
this.logger.debug(evt, 'TaskGather:_onDtmf');
|
||||||
if (evt.dtmf === this.finishOnKey) this._resolve('dtmf-terminator-key');
|
clearTimeout(this.interDigitTimer);
|
||||||
|
let resolved = false;
|
||||||
|
if (this.dtmfBargein) this._killAudio(cs);
|
||||||
|
if (evt.dtmf === this.finishOnKey && this.input.includes('digits')) {
|
||||||
|
resolved = true;
|
||||||
|
this._resolve('dtmf-terminator-key');
|
||||||
|
}
|
||||||
else {
|
else {
|
||||||
this.digitBuffer += evt.dtmf;
|
this.digitBuffer += evt.dtmf;
|
||||||
if (this.digitBuffer.length === this.numDigits) this._resolve('dtmf-num-digits');
|
const len = this.digitBuffer.length;
|
||||||
|
if (len === this.numDigits || len === this.maxDigits) {
|
||||||
|
resolved = true;
|
||||||
|
this._resolve('dtmf-num-digits');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!resolved && this.interDigitTimeout > 0 && this.digitBuffer.length >= this.minDigits) {
|
||||||
|
/* start interDigitTimer */
|
||||||
|
const ms = this.interDigitTimeout * 1000;
|
||||||
|
this.logger.debug(`starting interdigit timer of ${ms}`);
|
||||||
|
this.interDigitTimer = setTimeout(() => this._resolve('dtmf-interdigit-timeout'), ms);
|
||||||
}
|
}
|
||||||
this._killAudio(cs);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async _initSpeech(cs, ep) {
|
async _initSpeech(cs, ep) {
|
||||||
const opts = {};
|
const opts = {};
|
||||||
|
|
||||||
if (this.vad.enable) {
|
if (this.vad?.enable) {
|
||||||
opts.START_RECOGNIZING_ON_VAD = 1;
|
opts.START_RECOGNIZING_ON_VAD = 1;
|
||||||
if (this.vad.voiceMs) opts.RECOGNIZER_VAD_VOICE_MS = this.vad.voiceMs;
|
if (this.vad.voiceMs) opts.RECOGNIZER_VAD_VOICE_MS = this.vad.voiceMs;
|
||||||
|
else opts.RECOGNIZER_VAD_VOICE_MS = 125;
|
||||||
if (this.vad.mode >= 0 && this.vad.mode <= 3) opts.RECOGNIZER_VAD_MODE = this.vad.mode;
|
if (this.vad.mode >= 0 && this.vad.mode <= 3) opts.RECOGNIZER_VAD_MODE = this.vad.mode;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ('google' === this.vendor) {
|
if ('google' === this.vendor) {
|
||||||
if (this.sttCredentials) opts.GOOGLE_APPLICATION_CREDENTIALS = JSON.stringify(this.sttCredentials.credentials);
|
if (this.sttCredentials) opts.GOOGLE_APPLICATION_CREDENTIALS = JSON.stringify(this.sttCredentials.credentials);
|
||||||
Object.assign(opts, {
|
[
|
||||||
GOOGLE_SPEECH_USE_ENHANCED: true,
|
['enhancedModel', 'GOOGLE_SPEECH_USE_ENHANCED'],
|
||||||
GOOGLE_SPEECH_SINGLE_UTTERANCE: true,
|
['separateRecognitionPerChannel', 'GOOGLE_SPEECH_SEPARATE_RECOGNITION_PER_CHANNEL'],
|
||||||
GOOGLE_SPEECH_MODEL: 'command_and_search'
|
['profanityFilter', 'GOOGLE_SPEECH_PROFANITY_FILTER'],
|
||||||
|
['punctuation', 'GOOGLE_SPEECH_ENABLE_AUTOMATIC_PUNCTUATION'],
|
||||||
|
['words', 'GOOGLE_SPEECH_ENABLE_WORD_TIME_OFFSETS'],
|
||||||
|
['singleUtterance', 'GOOGLE_SPEECH_SINGLE_UTTERANCE'],
|
||||||
|
['diarization', 'GOOGLE_SPEECH_PROFANITY_FILTER']
|
||||||
|
].forEach((arr) => {
|
||||||
|
if (this[arr[0]]) opts[arr[1]] = true;
|
||||||
});
|
});
|
||||||
if (this.hints && this.hints.length > 1) {
|
if (this.hints.length > 1) {
|
||||||
opts.GOOGLE_SPEECH_HINTS = this.hints.map((h) => h.trim()).join(',');
|
opts.GOOGLE_SPEECH_HINTS = this.hints.join(',');
|
||||||
|
if (typeof this.hintsBoost === 'number') {
|
||||||
|
opts.GOOGLE_SPEECH_HINTS_BOOST = this.hintsBoost;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (this.altLanguages && this.altLanguages.length > 1) {
|
if (this.altLanguages.length > 1) opts.GOOGLE_SPEECH_ALTERNATIVE_LANGUAGE_CODES = this.altLanguages.join(',');
|
||||||
opts.GOOGLE_SPEECH_ALTERNATIVE_LANGUAGE_CODES = this.altLanguages.join(',');
|
if ('unspecified' !== this.interactionType) {
|
||||||
|
opts.GOOGLE_SPEECH_METADATA_INTERACTION_TYPE = this.interactionType;
|
||||||
}
|
}
|
||||||
if (this.profanityFilter === true) {
|
opts.GOOGLE_SPEECH_MODEL = this.model;
|
||||||
Object.assign(opts, {'GOOGLE_SPEECH_PROFANITY_FILTER': true});
|
if (this.diarization && this.diarizationMinSpeakers > 0) {
|
||||||
|
opts.GOOGLE_SPEECH_SPEAKER_DIARIZATION_MIN_SPEAKER_COUNT = this.diarizationMinSpeakers;
|
||||||
}
|
}
|
||||||
|
if (this.diarization && this.diarizationMaxSpeakers > 0) {
|
||||||
|
opts.GOOGLE_SPEECH_SPEAKER_DIARIZATION_MAX_SPEAKER_COUNT = this.diarizationMaxSpeakers;
|
||||||
|
}
|
||||||
|
if (this.naicsCode > 0) opts.GOOGLE_SPEECH_METADATA_INDUSTRY_NAICS_CODE = this.naicsCode;
|
||||||
ep.addCustomEventListener(GoogleTranscriptionEvents.Transcription, this._onTranscription.bind(this, cs, ep));
|
ep.addCustomEventListener(GoogleTranscriptionEvents.Transcription, this._onTranscription.bind(this, cs, ep));
|
||||||
ep.addCustomEventListener(GoogleTranscriptionEvents.EndOfUtterance, this._onEndOfUtterance.bind(this, cs, ep));
|
ep.addCustomEventListener(GoogleTranscriptionEvents.EndOfUtterance, this._onEndOfUtterance.bind(this, cs, ep));
|
||||||
|
ep.addCustomEventListener(GoogleTranscriptionEvents.VadDetected, this._onVadDetected.bind(this, cs, ep));
|
||||||
}
|
}
|
||||||
else if (['aws', 'polly'].includes(this.vendor)) {
|
else if (['aws', 'polly'].includes(this.vendor)) {
|
||||||
if (this.vocabularyName) opts.AWS_VOCABULARY_NAME = this.vocabularyName;
|
if (this.vocabularyName) opts.AWS_VOCABULARY_NAME = this.vocabularyName;
|
||||||
@@ -180,6 +289,7 @@ class TaskGather extends Task {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
ep.addCustomEventListener(AwsTranscriptionEvents.Transcription, this._onTranscription.bind(this, cs, ep));
|
ep.addCustomEventListener(AwsTranscriptionEvents.Transcription, this._onTranscription.bind(this, cs, ep));
|
||||||
|
ep.addCustomEventListener(AwsTranscriptionEvents.VadDetected, this._onVadDetected.bind(this, cs, ep));
|
||||||
}
|
}
|
||||||
else if ('microsoft' === this.vendor) {
|
else if ('microsoft' === this.vendor) {
|
||||||
if (this.sttCredentials) {
|
if (this.sttCredentials) {
|
||||||
@@ -191,23 +301,34 @@ class TaskGather extends Task {
|
|||||||
if (this.hints && this.hints.length > 1) {
|
if (this.hints && this.hints.length > 1) {
|
||||||
opts.AZURE_SPEECH_HINTS = this.hints.map((h) => h.trim()).join(',');
|
opts.AZURE_SPEECH_HINTS = this.hints.map((h) => h.trim()).join(',');
|
||||||
}
|
}
|
||||||
//if (this.requestSnr) opts.AZURE_REQUEST_SNR = 1;
|
if (this.altLanguages && this.altLanguages.length > 0) {
|
||||||
//if (this.profanityOption !== 'raw') opts.AZURE_PROFANITY_OPTION = this.profanityOption;
|
opts.AZURE_SPEECH_ALTERNATIVE_LANGUAGE_CODES = this.altLanguages.join(',');
|
||||||
|
}
|
||||||
|
if (this.requestSnr) opts.AZURE_REQUEST_SNR = 1;
|
||||||
|
if (this.profanityOption && this.profanityOption !== 'raw') opts.AZURE_PROFANITY_OPTION = this.profanityOption;
|
||||||
|
if (this.azureServiceEndpoint) opts.AZURE_SERVICE_ENDPOINT = this.azureServiceEndpoint;
|
||||||
if (this.initialSpeechTimeoutMs > 0) opts.AZURE_INITIAL_SPEECH_TIMEOUT_MS = this.initialSpeechTimeoutMs;
|
if (this.initialSpeechTimeoutMs > 0) opts.AZURE_INITIAL_SPEECH_TIMEOUT_MS = this.initialSpeechTimeoutMs;
|
||||||
|
else if (this.timeout === 0) opts.AZURE_INITIAL_SPEECH_TIMEOUT_MS = 120000; // lengthy
|
||||||
opts.AZURE_USE_OUTPUT_FORMAT_DETAILED = 1;
|
opts.AZURE_USE_OUTPUT_FORMAT_DETAILED = 1;
|
||||||
|
|
||||||
ep.addCustomEventListener(AzureTranscriptionEvents.Transcription, this._onTranscription.bind(this, cs, ep));
|
ep.addCustomEventListener(AzureTranscriptionEvents.Transcription, this._onTranscription.bind(this, cs, ep));
|
||||||
ep.addCustomEventListener(AzureTranscriptionEvents.NoSpeechDetected, this._onNoSpeechDetected.bind(this, cs, ep));
|
ep.addCustomEventListener(AzureTranscriptionEvents.NoSpeechDetected, this._onNoSpeechDetected.bind(this, cs, ep));
|
||||||
|
ep.addCustomEventListener(AzureTranscriptionEvents.VadDetected, this._onVadDetected.bind(this, cs, ep));
|
||||||
}
|
}
|
||||||
await ep.set(opts)
|
await ep.set(opts)
|
||||||
.catch((err) => this.logger.info(err, 'Error setting channel variables'));
|
.catch((err) => this.logger.info(err, 'Error setting channel variables'));
|
||||||
}
|
}
|
||||||
|
|
||||||
_startTranscribing(ep) {
|
_startTranscribing(ep) {
|
||||||
|
this.logger.debug({
|
||||||
|
vendor: this.vendor,
|
||||||
|
locale: this.language,
|
||||||
|
interim: this.interim
|
||||||
|
}, 'Gather:_startTranscribing');
|
||||||
ep.startTranscription({
|
ep.startTranscription({
|
||||||
vendor: this.vendor,
|
vendor: this.vendor,
|
||||||
locale: this.language,
|
locale: this.language,
|
||||||
interim: this.partialResultCallback ? true : false,
|
interim: this.interim,
|
||||||
}).catch((err) => {
|
}).catch((err) => {
|
||||||
const {writeAlerts, AlertType} = this.cs.srf.locals;
|
const {writeAlerts, AlertType} = this.cs.srf.locals;
|
||||||
this.logger.error(err, 'TaskGather:_startTranscribing error');
|
this.logger.error(err, 'TaskGather:_startTranscribing error');
|
||||||
@@ -221,9 +342,16 @@ class TaskGather extends Task {
|
|||||||
}
|
}
|
||||||
|
|
||||||
_startTimer() {
|
_startTimer() {
|
||||||
|
if (0 === this.timeout) return;
|
||||||
|
if (this._timeoutTimer) {
|
||||||
|
clearTimeout(this._timeoutTimer);
|
||||||
|
this._timeoutTimer = null;
|
||||||
|
}
|
||||||
assert(!this._timeoutTimer);
|
assert(!this._timeoutTimer);
|
||||||
this.logger.debug(`Gather:_startTimer: timeout ${this.timeout}`);
|
this.logger.debug(`Gather:_startTimer: timeout ${this.timeout}`);
|
||||||
this._timeoutTimer = setTimeout(() => this._resolve('timeout'), this.timeout);
|
this._timeoutTimer = setTimeout(() => {
|
||||||
|
this._resolve(this.digitBuffer.length >= this.minDigits ? 'dtmf-num-digits' : 'timeout');
|
||||||
|
}, this.timeout);
|
||||||
}
|
}
|
||||||
|
|
||||||
_clearTimer() {
|
_clearTimer() {
|
||||||
@@ -234,6 +362,15 @@ class TaskGather extends Task {
|
|||||||
}
|
}
|
||||||
|
|
||||||
_killAudio(cs) {
|
_killAudio(cs) {
|
||||||
|
if (!this.sayTask && !this.playTask && this.bargein) {
|
||||||
|
if (this.ep?.connected && !this.playComplete) {
|
||||||
|
this.logger.debug('Gather:_killAudio: killing playback of any audio');
|
||||||
|
this.playComplete = true;
|
||||||
|
this.ep.api('uuid_break', this.ep.uuid)
|
||||||
|
.catch((err) => this.logger.info(err, 'Error killing audio'));
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
if (this.sayTask && !this.sayTask.killed) {
|
if (this.sayTask && !this.sayTask.killed) {
|
||||||
this.sayTask.removeAllListeners('playDone');
|
this.sayTask.removeAllListeners('playDone');
|
||||||
this.sayTask.kill(cs);
|
this.sayTask.kill(cs);
|
||||||
@@ -249,58 +386,134 @@ class TaskGather extends Task {
|
|||||||
_onTranscription(cs, ep, evt) {
|
_onTranscription(cs, ep, evt) {
|
||||||
if ('aws' === this.vendor && Array.isArray(evt) && evt.length > 0) evt = evt[0];
|
if ('aws' === this.vendor && Array.isArray(evt) && evt.length > 0) evt = evt[0];
|
||||||
if ('microsoft' === this.vendor) {
|
if ('microsoft' === this.vendor) {
|
||||||
const nbest = evt.NBest;
|
const final = evt.RecognitionStatus === 'Success';
|
||||||
const newEvent = {
|
if (final) {
|
||||||
is_final: evt.RecognitionStatus === 'Success',
|
// don't sort based on confidence: https://github.com/Azure-Samples/cognitive-services-speech-sdk/issues/1463
|
||||||
alternatives: [
|
//const nbest = evt.NBest.sort((a, b) => b.Confidence - a.Confidence);
|
||||||
{
|
const nbest = evt.NBest;
|
||||||
confidence: nbest[0].Confidence,
|
const language_code = evt.PrimaryLanguage?.Language || this.language;
|
||||||
transcript: nbest[0].Display
|
evt = {
|
||||||
}
|
is_final: true,
|
||||||
]
|
language_code,
|
||||||
};
|
alternatives: [
|
||||||
evt = newEvent;
|
{
|
||||||
|
confidence: nbest[0].Confidence,
|
||||||
|
transcript: nbest[0].Display
|
||||||
|
}
|
||||||
|
]
|
||||||
|
};
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
evt = {
|
||||||
|
is_final: false,
|
||||||
|
alternatives: [
|
||||||
|
{
|
||||||
|
transcript: evt.Text
|
||||||
|
}
|
||||||
|
]
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
this.logger.debug(evt, 'TaskGather:_onTranscription');
|
if (evt.is_final) {
|
||||||
if (evt.is_final) this._resolve('speech', evt);
|
if (evt.alternatives[0].transcript === '' && !this.callSession.callGone && !this.killed) {
|
||||||
else if (this.partialResultHook) {
|
this.logger.info({evt}, 'TaskGather:_onTranscription - got empty transcript, listen again');
|
||||||
this.cs.requestor.request(this.partialResultHook, Object.assign({speech: evt}, this.cs.callInfo))
|
return this._startTranscribing(ep);
|
||||||
.catch((err) => this.logger.info(err, 'GatherTask:_onTranscription error'));
|
}
|
||||||
|
this._resolve('speech', evt);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
/* google has a measure of stability:
|
||||||
|
https://cloud.google.com/speech-to-text/docs/basics#streaming_responses
|
||||||
|
others do not.
|
||||||
|
*/
|
||||||
|
//const isStableEnough = typeof evt.stability === 'undefined' || evt.stability > GATHER_STABILITY_THRESHOLD;
|
||||||
|
|
||||||
|
if (this.bargein && /* isStableEnough && */
|
||||||
|
evt.alternatives[0].transcript.split(' ').length >= this.minBargeinWordCount) {
|
||||||
|
if (!this.playComplete) {
|
||||||
|
this.logger.debug({transcript: evt.alternatives[0].transcript}, 'killing audio due to speech');
|
||||||
|
this.emit('vad');
|
||||||
|
}
|
||||||
|
this._killAudio(cs);
|
||||||
|
}
|
||||||
|
if (this.partialResultHook) {
|
||||||
|
const b3 = this.getTracingPropagation();
|
||||||
|
const httpHeaders = b3 && {b3};
|
||||||
|
this.cs.requestor.request(this.partialResultHook, Object.assign({speech: evt},
|
||||||
|
this.cs.callInfo, httpHeaders));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
_onEndOfUtterance(cs, ep) {
|
_onEndOfUtterance(cs, ep) {
|
||||||
this.logger.info('TaskGather:_onEndOfUtterance');
|
this.logger.debug('TaskGather:_onEndOfUtterance');
|
||||||
|
if (this.bargein && this.minBargeinWordCount === 0) {
|
||||||
|
this._killAudio(cs);
|
||||||
|
}
|
||||||
|
|
||||||
if (!this.resolved && !this.killed) {
|
if (!this.resolved && !this.killed) {
|
||||||
this._startTranscribing(ep);
|
this._startTranscribing(ep);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_onVadDetected(cs, ep) {
|
||||||
|
if (this.bargein && this.minBargeinWordCount === 0) {
|
||||||
|
this.logger.debug('TaskGather:_onVadDetected');
|
||||||
|
this._killAudio(cs);
|
||||||
|
this.emit('vad');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
_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) {
|
||||||
if (this.resolved) return;
|
|
||||||
this.resolved = true;
|
|
||||||
this.logger.debug(`TaskGather:resolve with reason ${reason}`);
|
this.logger.debug(`TaskGather:resolve with reason ${reason}`);
|
||||||
|
if (this.resolved) return;
|
||||||
|
|
||||||
|
this.resolved = true;
|
||||||
|
clearTimeout(this.interDigitTimer);
|
||||||
|
|
||||||
|
this.span.setAttributes({'stt.resolve': reason, 'stt.result': JSON.stringify(evt)});
|
||||||
if (this.ep && this.ep.connected) {
|
if (this.ep && this.ep.connected) {
|
||||||
this.ep.stopTranscription({vendor: this.vendor})
|
this.ep.stopTranscription({vendor: this.vendor})
|
||||||
.catch((err) => this.logger.error({err}, 'Error stopping transcription'));
|
.catch((err) => this.logger.error({err}, 'Error stopping transcription'));
|
||||||
}
|
}
|
||||||
|
|
||||||
this._clearTimer();
|
this._clearTimer();
|
||||||
if (reason.startsWith('dtmf')) {
|
|
||||||
await this.performAction({digits: this.digitBuffer, reason: 'dtmfDetected'});
|
if (this.callSession && this.callSession.callGone) {
|
||||||
}
|
this.logger.debug('TaskGather:_resolve - call is gone, not invoking web callback');
|
||||||
else if (reason.startsWith('speech')) {
|
this.notifyTaskDone();
|
||||||
if (this.parentTask) this.parentTask.emit('transcription', evt);
|
return;
|
||||||
else await this.performAction({speech: evt, reason: 'speechDetected'});
|
|
||||||
}
|
|
||||||
else if (reason.startsWith('timeout')) {
|
|
||||||
if (this.parentTask) this.parentTask.emit('timeout', evt);
|
|
||||||
else await this.performAction({reason: 'timeout'});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (reason.startsWith('dtmf')) {
|
||||||
|
if (this.parentTask) this.parentTask.emit('dtmf', evt);
|
||||||
|
else {
|
||||||
|
this.emit('dtmf', evt);
|
||||||
|
await this.performAction({digits: this.digitBuffer, reason: 'dtmfDetected'});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (reason.startsWith('speech')) {
|
||||||
|
if (this.parentTask) this.parentTask.emit('transcription', evt);
|
||||||
|
else {
|
||||||
|
this.emit('transcription', evt);
|
||||||
|
await this.performAction({speech: evt, reason: 'speechDetected'});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (reason.startsWith('timeout')) {
|
||||||
|
if (this.parentTask) this.parentTask.emit('timeout', evt);
|
||||||
|
else {
|
||||||
|
this.emit('timeout', evt);
|
||||||
|
await this.performAction({reason: 'timeout'});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (err) { /*already logged error*/ }
|
||||||
this.notifyTaskDone();
|
this.notifyTaskDone();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -289,7 +289,9 @@ class Lex extends Task {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async _performHook(cs, hook, results) {
|
async _performHook(cs, hook, results) {
|
||||||
const json = await this.cs.requestor.request(hook, results);
|
const b3 = this.getTracingPropagation();
|
||||||
|
const httpHeaders = b3 && {b3};
|
||||||
|
const json = await this.cs.requestor.request('verb:hook', hook, results, httpHeaders);
|
||||||
if (json && Array.isArray(json)) {
|
if (json && Array.isArray(json)) {
|
||||||
const makeTask = require('./make_task');
|
const makeTask = require('./make_task');
|
||||||
const tasks = normalizeJambones(this.logger, json).map((tdata) => makeTask(this.logger, tdata));
|
const tasks = normalizeJambones(this.logger, json).map((tdata) => makeTask(this.logger, tdata));
|
||||||
|
|||||||
@@ -38,7 +38,12 @@ class TaskListen extends Task {
|
|||||||
if (this.playBeep) await this._playBeep(ep);
|
if (this.playBeep) await this._playBeep(ep);
|
||||||
if (this.transcribeTask) {
|
if (this.transcribeTask) {
|
||||||
this.logger.debug('TaskListen:exec - starting nested transcribe task');
|
this.logger.debug('TaskListen:exec - starting nested transcribe task');
|
||||||
this.transcribeTask.exec(cs, ep);
|
const {span, ctx} = this.startChildSpan(`nested:${this.transcribeTask.summary}`);
|
||||||
|
this.transcribeTask.span = span;
|
||||||
|
this.transcribeTask.ctx = ctx;
|
||||||
|
this.transcribeTask.exec(cs, ep)
|
||||||
|
.then((result) => span.end())
|
||||||
|
.catch((err) => span.end());
|
||||||
}
|
}
|
||||||
await this._startListening(cs, ep);
|
await this._startListening(cs, ep);
|
||||||
await this.awaitTaskDone();
|
await this.awaitTaskDone();
|
||||||
|
|||||||
@@ -20,9 +20,9 @@ function makeTask(logger, obj, parent) {
|
|||||||
case TaskName.SipRefer:
|
case TaskName.SipRefer:
|
||||||
const TaskSipRefer = require('./sip_refer');
|
const TaskSipRefer = require('./sip_refer');
|
||||||
return new TaskSipRefer(logger, data, parent);
|
return new TaskSipRefer(logger, data, parent);
|
||||||
case TaskName.Cognigy:
|
case TaskName.Config:
|
||||||
const TaskCognigy = require('./cognigy');
|
const TaskConfig = require('./config');
|
||||||
return new TaskCognigy(logger, data, parent);
|
return new TaskConfig(logger, data, parent);
|
||||||
case TaskName.Conference:
|
case TaskName.Conference:
|
||||||
const TaskConference = require('./conference');
|
const TaskConference = require('./conference');
|
||||||
return new TaskConference(logger, data, parent);
|
return new TaskConference(logger, data, parent);
|
||||||
|
|||||||
@@ -13,6 +13,10 @@ class TaskPlay extends Task {
|
|||||||
|
|
||||||
get name() { return TaskName.Play; }
|
get name() { return TaskName.Play; }
|
||||||
|
|
||||||
|
get summary() {
|
||||||
|
return `${this.name}:{url=${this.url}}`;
|
||||||
|
}
|
||||||
|
|
||||||
async exec(cs, ep) {
|
async exec(cs, ep) {
|
||||||
await super.exec(cs);
|
await super.exec(cs);
|
||||||
this.ep = ep;
|
this.ep = ep;
|
||||||
|
|||||||
@@ -31,8 +31,15 @@ class Rasa extends Task {
|
|||||||
|
|
||||||
/* start the first gather */
|
/* start the first gather */
|
||||||
this.gatherTask = this._makeGatherTask(this.prompt);
|
this.gatherTask = this._makeGatherTask(this.prompt);
|
||||||
|
const {span, ctx} = this.startChildSpan(`nested:${this.gatherTask.summary}`);
|
||||||
|
this.gatherTask.span = span;
|
||||||
|
this.gatherTask.ctx = ctx;
|
||||||
this.gatherTask.exec(cs, ep, this)
|
this.gatherTask.exec(cs, ep, this)
|
||||||
.catch((err) => this.logger.info({err}, 'Rasa gather task returned error'));
|
.then(() => span.end())
|
||||||
|
.catch((err) => {
|
||||||
|
span.end();
|
||||||
|
this.logger.info({err}, 'Rasa gather task returned error');
|
||||||
|
});
|
||||||
|
|
||||||
await this.awaitTaskDone();
|
await this.awaitTaskDone();
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
@@ -118,8 +125,15 @@ class Rasa extends Task {
|
|||||||
if (botUtterance) {
|
if (botUtterance) {
|
||||||
this.logger.debug({botUtterance}, 'Rasa:_onTranscription: got user utterance');
|
this.logger.debug({botUtterance}, 'Rasa:_onTranscription: got user utterance');
|
||||||
this.gatherTask = this._makeGatherTask(botUtterance);
|
this.gatherTask = this._makeGatherTask(botUtterance);
|
||||||
|
const {span, ctx} = this.startChildSpan(`nested:${this.gatherTask.summary}`);
|
||||||
|
this.gatherTask.span = span;
|
||||||
|
this.gatherTask.ctx = ctx;
|
||||||
this.gatherTask.exec(cs, ep, this)
|
this.gatherTask.exec(cs, ep, this)
|
||||||
.catch((err) => this.logger.info({err}, 'Rasa gather task returned error'));
|
.then(() => span.end())
|
||||||
|
.catch((err) => {
|
||||||
|
span.end();
|
||||||
|
this.logger.info({err}, 'Rasa gather task returned error');
|
||||||
|
});
|
||||||
if (this.eventHook) {
|
if (this.eventHook) {
|
||||||
this.performHook(cs, this.eventHook, {event: 'botMessage', message: response})
|
this.performHook(cs, this.eventHook, {event: 'botMessage', message: response})
|
||||||
.then((redirected) => {
|
.then((redirected) => {
|
||||||
|
|||||||
@@ -48,7 +48,23 @@ class TaskRestDial extends Task {
|
|||||||
cs.setDialog(dlg);
|
cs.setDialog(dlg);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const tasks = await cs.requestor.request(this.call_hook, cs.callInfo);
|
const b3 = this.getTracingPropagation();
|
||||||
|
const httpHeaders = b3 && {b3};
|
||||||
|
const params = {
|
||||||
|
...cs.callInfo,
|
||||||
|
defaults: {
|
||||||
|
synthesizer: {
|
||||||
|
vendor: cs.speechSynthesisVendor,
|
||||||
|
language: cs.speechSynthesisLanguage,
|
||||||
|
voice: cs.speechSynthesisVoice
|
||||||
|
},
|
||||||
|
recognizer: {
|
||||||
|
vendor: cs.speechRecognizerVendor,
|
||||||
|
language: cs.speechRecognizerLanguage
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
const tasks = await cs.requestor.request('session:new', this.call_hook, params, httpHeaders);
|
||||||
if (tasks && Array.isArray(tasks)) {
|
if (tasks && Array.isArray(tasks)) {
|
||||||
this.logger.debug({tasks: tasks}, `TaskRestDial: replacing application with ${tasks.length} tasks`);
|
this.logger.debug({tasks: tasks}, `TaskRestDial: replacing application with ${tasks.length} tasks`);
|
||||||
cs.replaceApplication(normalizeJambones(this.logger, tasks).map((tdata) => makeTask(this.logger, tdata)));
|
cs.replaceApplication(normalizeJambones(this.logger, tasks).map((tdata) => makeTask(this.logger, tdata)));
|
||||||
|
|||||||
@@ -14,6 +14,14 @@ class TaskSay extends Task {
|
|||||||
|
|
||||||
get name() { return TaskName.Say; }
|
get name() { return TaskName.Say; }
|
||||||
|
|
||||||
|
get summary() {
|
||||||
|
for (let i = 0; i < this.text.length; i++) {
|
||||||
|
if (this.text[i].startsWith('silence_stream')) continue;
|
||||||
|
return `${this.name}{text=${this.text[i].slice(0, 15)}${this.text[i].length > 15 ? '...' : ''}}`;
|
||||||
|
}
|
||||||
|
return `${this.name}{${this.text[0]}}`;
|
||||||
|
}
|
||||||
|
|
||||||
async exec(cs, ep) {
|
async exec(cs, ep) {
|
||||||
await super.exec(cs);
|
await super.exec(cs);
|
||||||
|
|
||||||
@@ -21,15 +29,20 @@ class TaskSay extends Task {
|
|||||||
const {updateSpeechCredentialLastUsed} = require('../utils/db-utils')(this.logger, srf);
|
const {updateSpeechCredentialLastUsed} = require('../utils/db-utils')(this.logger, srf);
|
||||||
const {writeAlerts, AlertType, stats} = srf.locals;
|
const {writeAlerts, AlertType, stats} = srf.locals;
|
||||||
const {synthAudio} = srf.locals.dbHelpers;
|
const {synthAudio} = srf.locals.dbHelpers;
|
||||||
const hasVerbLevelTts = this.synthesizer.vendor && this.synthesizer.vendor !== 'default';
|
const vendor = this.synthesizer.vendor && this.synthesizer.vendor !== 'default' ?
|
||||||
const vendor = hasVerbLevelTts ? this.synthesizer.vendor : cs.speechSynthesisVendor ;
|
this.synthesizer.vendor :
|
||||||
const language = hasVerbLevelTts ? this.synthesizer.language : cs.speechSynthesisLanguage ;
|
cs.speechSynthesisVendor;
|
||||||
const voice = hasVerbLevelTts ? this.synthesizer.voice : cs.speechSynthesisVoice ;
|
const language = this.synthesizer.language && this.synthesizer.language !== 'default' ?
|
||||||
|
this.synthesizer.language :
|
||||||
|
cs.speechSynthesisLanguage ;
|
||||||
|
const voice = this.synthesizer.voice && this.synthesizer.voice !== 'default' ?
|
||||||
|
this.synthesizer.voice :
|
||||||
|
cs.speechSynthesisVoice;
|
||||||
const engine = this.synthesizer.engine || 'standard';
|
const engine = this.synthesizer.engine || 'standard';
|
||||||
const salt = cs.callSid;
|
const salt = cs.callSid;
|
||||||
const credentials = cs.getSpeechCredentials(vendor, 'tts');
|
const credentials = cs.getSpeechCredentials(vendor, 'tts');
|
||||||
|
|
||||||
this.logger.info({language, voice}, `Task:say - using vendor: ${vendor}`);
|
this.logger.info({vendor, language, voice}, 'TaskSay:exec');
|
||||||
this.ep = ep;
|
this.ep = ep;
|
||||||
try {
|
try {
|
||||||
if (!credentials) {
|
if (!credentials) {
|
||||||
@@ -38,49 +51,75 @@ 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;
|
||||||
const filepath = (await Promise.all(this.text.map(async(text) => {
|
|
||||||
const {filePath, servedFromCache} = await synthAudio(stats, {
|
/* produce an audio segment from the provided text */
|
||||||
text,
|
const generateAudio = async(text) => {
|
||||||
vendor,
|
if (this.killed) return;
|
||||||
language,
|
if (text.startsWith('silence_stream://')) return text;
|
||||||
voice,
|
|
||||||
engine,
|
/* otel: trace time for tts */
|
||||||
salt,
|
const {span} = this.startChildSpan('tts-generation', {
|
||||||
credentials
|
'tts.vendor': vendor,
|
||||||
}).catch((err) => {
|
'tts.language': language,
|
||||||
this.logger.info(err, 'Error synthesizing tts');
|
'tts.voice': voice
|
||||||
|
});
|
||||||
|
try {
|
||||||
|
const {filePath, servedFromCache} = await synthAudio(stats, {
|
||||||
|
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 */});
|
|
||||||
}
|
}
|
||||||
return filePath;
|
};
|
||||||
}))).filter((fp) => fp && fp.length);
|
|
||||||
|
|
||||||
|
const arr = this.text.map((t) => generateAudio(t));
|
||||||
|
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]);
|
||||||
}
|
}
|
||||||
else await ep.play(filepath[segment]);
|
else {
|
||||||
} while (!this.killed && ++segment < filepath.length);
|
this.logger.debug(`Say:exec sending command to play file ${filepath[segment]}`);
|
||||||
|
await ep.play(filepath[segment]);
|
||||||
|
this.logger.debug(`Say:exec completed play file ${filepath[segment]}`);
|
||||||
|
}
|
||||||
|
segment++;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this.logger.info(err, 'TaskSay:exec error');
|
this.logger.info(err, 'TaskSay:exec error');
|
||||||
|
|||||||
@@ -19,7 +19,11 @@ class TaskSipDecline extends Task {
|
|||||||
res.send(this.data.status, this.data.reason, {
|
res.send(this.data.status, this.data.reason, {
|
||||||
headers: this.headers
|
headers: this.headers
|
||||||
});
|
});
|
||||||
cs.emit('callStatusChange', {callStatus: CallStatus.Failed, sipStatus: this.data.status});
|
cs.emit('callStatusChange', {
|
||||||
|
callStatus: CallStatus.Failed,
|
||||||
|
sipStatus: this.data.status,
|
||||||
|
sipReason: this.data.reason
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -26,6 +26,12 @@ class TaskSipRefer extends Task {
|
|||||||
try {
|
try {
|
||||||
this.notifyHandler = this._handleNotify.bind(this, cs, dlg);
|
this.notifyHandler = this._handleNotify.bind(this, cs, dlg);
|
||||||
dlg.on('notify', this.notifyHandler);
|
dlg.on('notify', this.notifyHandler);
|
||||||
|
/* otel: trace time for tts */
|
||||||
|
this.referSpan = this.startSpan('send-refer', {
|
||||||
|
'refer.refer_to': referTo,
|
||||||
|
'refer.referred_by': referredBy
|
||||||
|
});
|
||||||
|
|
||||||
const response = await dlg.request({
|
const response = await dlg.request({
|
||||||
method: 'REFER',
|
method: 'REFER',
|
||||||
headers: {
|
headers: {
|
||||||
@@ -35,22 +41,27 @@ class TaskSipRefer extends Task {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
this.referStatus = response.status;
|
this.referStatus = response.status;
|
||||||
|
this.referSpan.setAttributes({'refer.status_code': response.status});
|
||||||
this.logger.info(`TaskSipRefer:exec - received ${this.referStatus} to REFER`);
|
this.logger.info(`TaskSipRefer:exec - received ${this.referStatus} to REFER`);
|
||||||
|
|
||||||
/* if we fail, fall through to next verb. If success, we should get BYE from far end */
|
/* if we fail, fall through to next verb. If success, we should get BYE from far end */
|
||||||
if (this.referStatus === 202) {
|
if (this.referStatus === 202) {
|
||||||
await this.awaitTaskDone();
|
await this.awaitTaskDone();
|
||||||
}
|
}
|
||||||
else await this.performAction({refer_status: this.referStatus});
|
else {
|
||||||
|
await this.performAction({refer_status: this.referStatus});
|
||||||
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this.logger.info({err}, 'TaskSipRefer:exec - error sending REFER');
|
this.logger.info({err}, 'TaskSipRefer:exec - error sending REFER');
|
||||||
}
|
}
|
||||||
|
this.referSpan?.end();
|
||||||
}
|
}
|
||||||
|
|
||||||
async kill(cs) {
|
async kill(cs) {
|
||||||
super.kill(cs);
|
super.kill(cs);
|
||||||
const {dlg} = cs;
|
const {dlg} = cs;
|
||||||
dlg.off('notify', this.notifyHandler);
|
dlg.off('notify', this.notifyHandler);
|
||||||
|
this.notifyTaskDone();
|
||||||
}
|
}
|
||||||
|
|
||||||
async _handleNotify(cs, dlg, req, res) {
|
async _handleNotify(cs, dlg, req, res) {
|
||||||
@@ -65,9 +76,13 @@ class TaskSipRefer extends Task {
|
|||||||
const status = arr[1];
|
const status = arr[1];
|
||||||
this.logger.debug(`TaskSipRefer:_handleNotify: call got status ${status}`);
|
this.logger.debug(`TaskSipRefer:_handleNotify: call got status ${status}`);
|
||||||
if (this.eventHook) {
|
if (this.eventHook) {
|
||||||
await cs.requestor.request(this.eventHook, {event: 'transfer-status', call_status: status});
|
const b3 = this.getTracingPropagation();
|
||||||
|
const httpHeaders = b3 && {b3};
|
||||||
|
await cs.requestor.request('verb:hook', this.eventHook,
|
||||||
|
{event: 'transfer-status', call_status: status}, httpHeaders);
|
||||||
}
|
}
|
||||||
if (status >= 200) {
|
if (status >= 200) {
|
||||||
|
this.referSpan.setAttributes({'refer.finalNotify': status});
|
||||||
await this.performAction({refer_status: 202, final_referred_call_status: status});
|
await this.performAction({refer_status: 202, final_referred_call_status: status});
|
||||||
this.notifyTaskDone();
|
this.notifyTaskDone();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,20 +21,30 @@
|
|||||||
"referTo"
|
"referTo"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"cognigy": {
|
"config": {
|
||||||
"properties": {
|
"properties": {
|
||||||
"url": "string",
|
"synthesizer": "#synthesizer",
|
||||||
"token": "string",
|
|
||||||
"recognizer": "#recognizer",
|
"recognizer": "#recognizer",
|
||||||
"tts": "#synthesizer",
|
"bargeIn": "#bargeIn"
|
||||||
"prompt": "string",
|
},
|
||||||
|
"required": []
|
||||||
|
},
|
||||||
|
"bargeIn": {
|
||||||
|
"properties": {
|
||||||
|
"enable": "boolean",
|
||||||
|
"sticky": "boolean",
|
||||||
"actionHook": "object|string",
|
"actionHook": "object|string",
|
||||||
"eventHook": "object|string",
|
"input": "array",
|
||||||
"data": "object"
|
"finishOnKey": "string",
|
||||||
|
"numDigits": "number",
|
||||||
|
"minDigits": "number",
|
||||||
|
"maxDigits": "number",
|
||||||
|
"interDigitTimeout": "number",
|
||||||
|
"dtmfBargein": "boolean",
|
||||||
|
"minBargeinWordCount": "number"
|
||||||
},
|
},
|
||||||
"required": [
|
"required": [
|
||||||
"url",
|
"enable"
|
||||||
"token"
|
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"dequeue": {
|
"dequeue": {
|
||||||
@@ -98,8 +108,15 @@
|
|||||||
"finishOnKey": "string",
|
"finishOnKey": "string",
|
||||||
"input": "array",
|
"input": "array",
|
||||||
"numDigits": "number",
|
"numDigits": "number",
|
||||||
|
"minDigits": "number",
|
||||||
|
"maxDigits": "number",
|
||||||
|
"interDigitTimeout": "number",
|
||||||
"partialResultHook": "object|string",
|
"partialResultHook": "object|string",
|
||||||
"speechTimeout": "number",
|
"speechTimeout": "number",
|
||||||
|
"listenDuringPrompt": "boolean",
|
||||||
|
"dtmfBargein": "boolean",
|
||||||
|
"bargein": "boolean",
|
||||||
|
"minBargeinWordCount": "number",
|
||||||
"timeout": "number",
|
"timeout": "number",
|
||||||
"recognizer": "#recognizer",
|
"recognizer": "#recognizer",
|
||||||
"play": "#play",
|
"play": "#play",
|
||||||
@@ -133,6 +150,7 @@
|
|||||||
"answerOnBridge": "boolean",
|
"answerOnBridge": "boolean",
|
||||||
"callerId": "string",
|
"callerId": "string",
|
||||||
"confirmHook": "object|string",
|
"confirmHook": "object|string",
|
||||||
|
"referHook": "object|string",
|
||||||
"dialMusic": "string",
|
"dialMusic": "string",
|
||||||
"dtmfCapture": "object",
|
"dtmfCapture": "object",
|
||||||
"dtmfHook": "object|string",
|
"dtmfHook": "object|string",
|
||||||
@@ -338,6 +356,7 @@
|
|||||||
"type": "string",
|
"type": "string",
|
||||||
"enum": ["GET", "POST"]
|
"enum": ["GET", "POST"]
|
||||||
},
|
},
|
||||||
|
"headers": "object",
|
||||||
"name": "string",
|
"name": "string",
|
||||||
"number": "string",
|
"number": "string",
|
||||||
"sipUri": "string",
|
"sipUri": "string",
|
||||||
@@ -391,6 +410,7 @@
|
|||||||
"language": "string",
|
"language": "string",
|
||||||
"vad": "#vad",
|
"vad": "#vad",
|
||||||
"hints": "array",
|
"hints": "array",
|
||||||
|
"hintsBoost": "number",
|
||||||
"altLanguages": "array",
|
"altLanguages": "array",
|
||||||
"profanityFilter": "boolean",
|
"profanityFilter": "boolean",
|
||||||
"interim": "boolean",
|
"interim": "boolean",
|
||||||
@@ -428,6 +448,7 @@
|
|||||||
"tag"
|
"tag"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
"model": "string",
|
||||||
"outputFormat": {
|
"outputFormat": {
|
||||||
"type": "string",
|
"type": "string",
|
||||||
"enum": [
|
"enum": [
|
||||||
@@ -444,7 +465,8 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"requestSnr": "boolean",
|
"requestSnr": "boolean",
|
||||||
"initialSpeechTimeoutMs": "number"
|
"initialSpeechTimeoutMs": "number",
|
||||||
|
"azureServiceEndpoint": "string"
|
||||||
},
|
},
|
||||||
"required": [
|
"required": [
|
||||||
"vendor"
|
"vendor"
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ const debug = require('debug')('jambonz:feature-server');
|
|||||||
const assert = require('assert');
|
const assert = require('assert');
|
||||||
const {TaskPreconditions} = require('../utils/constants');
|
const {TaskPreconditions} = require('../utils/constants');
|
||||||
const normalizeJambones = require('../utils/normalize-jambones');
|
const normalizeJambones = require('../utils/normalize-jambones');
|
||||||
|
const {trace} = require('@opentelemetry/api');
|
||||||
const specs = new Map();
|
const specs = new Map();
|
||||||
const _specData = require('./specs');
|
const _specData = require('./specs');
|
||||||
for (const key in _specData) {specs.set(key, _specData[key]);}
|
for (const key in _specData) {specs.set(key, _specData[key]);}
|
||||||
@@ -42,6 +43,10 @@ class Task extends Emitter {
|
|||||||
return this.cs;
|
return this.cs;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get summary() {
|
||||||
|
return this.name;
|
||||||
|
}
|
||||||
|
|
||||||
toJSON() {
|
toJSON() {
|
||||||
return this.data;
|
return this.data;
|
||||||
}
|
}
|
||||||
@@ -67,6 +72,34 @@ class Task extends Emitter {
|
|||||||
setImmediate(() => this.parentTask = null);
|
setImmediate(() => this.parentTask = null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
startSpan(name, attributes) {
|
||||||
|
const {srf} = require('../..');
|
||||||
|
const {tracer} = srf.locals.otel;
|
||||||
|
const span = tracer.startSpan(name, undefined, this.ctx);
|
||||||
|
if (attributes) span.setAttributes(attributes);
|
||||||
|
trace.setSpan(this.ctx, span);
|
||||||
|
return span;
|
||||||
|
}
|
||||||
|
|
||||||
|
startChildSpan(name, attributes) {
|
||||||
|
const {srf} = require('../..');
|
||||||
|
const {tracer} = srf.locals.otel;
|
||||||
|
const span = tracer.startSpan(name, undefined, this.ctx);
|
||||||
|
if (attributes) span.setAttributes(attributes);
|
||||||
|
const ctx = trace.setSpan(this.ctx, span);
|
||||||
|
return {span, ctx};
|
||||||
|
}
|
||||||
|
|
||||||
|
getTracingPropagation(encoding, span) {
|
||||||
|
// TODO: support encodings beyond b3 https://github.com/openzipkin/b3-propagation
|
||||||
|
if (span) {
|
||||||
|
return `${span.spanContext().traceId}-${span.spanContext().spanId}-1`;
|
||||||
|
}
|
||||||
|
if (this.span) {
|
||||||
|
return `${this.span.spanContext().traceId}-${this.span.spanContext().spanId}-1`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* when a subclass Task has completed its work, it should call this method
|
* when a subclass Task has completed its work, it should call this method
|
||||||
*/
|
*/
|
||||||
@@ -104,32 +137,62 @@ 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();
|
||||||
const json = await this.cs.requestor.request(this.actionHook, params);
|
const span = this.startSpan('verb:hook', {'hook.url': this.actionHook});
|
||||||
if (expectResponse && json && Array.isArray(json)) {
|
const b3 = this.getTracingPropagation('b3', span);
|
||||||
const makeTask = require('./make_task');
|
const httpHeaders = b3 && {b3};
|
||||||
const tasks = normalizeJambones(this.logger, json).map((tdata) => makeTask(this.logger, tdata));
|
span.setAttributes({'http.body': JSON.stringify(params)});
|
||||||
if (tasks && tasks.length > 0) {
|
try {
|
||||||
this.logger.info({tasks: tasks}, `${this.name} replacing application with ${tasks.length} tasks`);
|
const json = await this.cs.requestor.request('verb:hook', this.actionHook, params, httpHeaders);
|
||||||
this.callSession.replaceApplication(tasks);
|
span.setAttributes({'http.statusCode': 200});
|
||||||
|
span.end();
|
||||||
|
if (expectResponse && json && Array.isArray(json)) {
|
||||||
|
const makeTask = require('./make_task');
|
||||||
|
const tasks = normalizeJambones(this.logger, json).map((tdata) => makeTask(this.logger, tdata));
|
||||||
|
if (tasks && tasks.length > 0) {
|
||||||
|
this.logger.info({tasks: tasks}, `${this.name} replacing application with ${tasks.length} tasks`);
|
||||||
|
this.callSession.replaceApplication(tasks);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
} catch (err) {
|
||||||
|
span.setAttributes({'http.statusCode': err.statusCode});
|
||||||
|
span.end();
|
||||||
|
throw err;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async performHook(cs, hook, results) {
|
async performHook(cs, hook, results) {
|
||||||
const json = await cs.requestor.request(hook, results);
|
const span = this.startSpan('verb:hook', {'hook.url': hook});
|
||||||
if (json && Array.isArray(json)) {
|
const b3 = this.getTracingPropagation('b3', span);
|
||||||
const makeTask = require('./make_task');
|
const httpHeaders = b3 && {b3};
|
||||||
const tasks = normalizeJambones(this.logger, json).map((tdata) => makeTask(this.logger, tdata));
|
span.setAttributes({'http.body': JSON.stringify(results)});
|
||||||
if (tasks && tasks.length > 0) {
|
try {
|
||||||
this.redirect(cs, tasks);
|
const json = await cs.requestor.request('verb:hook', hook, results, httpHeaders);
|
||||||
return true;
|
span.setAttributes({'http.statusCode': 200});
|
||||||
|
span.end();
|
||||||
|
if (json && Array.isArray(json)) {
|
||||||
|
const makeTask = require('./make_task');
|
||||||
|
const tasks = normalizeJambones(this.logger, json).map((tdata) => makeTask(this.logger, tdata));
|
||||||
|
if (tasks && tasks.length > 0) {
|
||||||
|
this.redirect(cs, tasks);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
return false;
|
||||||
|
} catch (err) {
|
||||||
|
span.setAttributes({'http.statusCode': err.statusCode});
|
||||||
|
span.end();
|
||||||
|
throw err;
|
||||||
}
|
}
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
redirect(cs, tasks) {
|
redirect(cs, tasks) {
|
||||||
|
|||||||
@@ -28,10 +28,13 @@ class TaskTranscribe extends Task {
|
|||||||
|
|
||||||
/* google-specific options */
|
/* google-specific options */
|
||||||
this.hints = recognizer.hints || [];
|
this.hints = recognizer.hints || [];
|
||||||
|
this.hintsBoost = recognizer.hintsBoost;
|
||||||
this.profanityFilter = recognizer.profanityFilter;
|
this.profanityFilter = recognizer.profanityFilter;
|
||||||
this.punctuation = !!recognizer.punctuation;
|
this.punctuation = !!recognizer.punctuation;
|
||||||
this.enhancedModel = !!recognizer.enhancedModel;
|
this.enhancedModel = !!recognizer.enhancedModel;
|
||||||
|
this.model = recognizer.model || 'phone_call';
|
||||||
this.words = !!recognizer.words;
|
this.words = !!recognizer.words;
|
||||||
|
this.singleUtterance = recognizer.singleUtterance || false;
|
||||||
this.diarization = !!recognizer.diarization;
|
this.diarization = !!recognizer.diarization;
|
||||||
this.diarizationMinSpeakers = recognizer.diarizationMinSpeakers || 0;
|
this.diarizationMinSpeakers = recognizer.diarizationMinSpeakers || 0;
|
||||||
this.diarizationMaxSpeakers = recognizer.diarizationMaxSpeakers || 0;
|
this.diarizationMaxSpeakers = recognizer.diarizationMaxSpeakers || 0;
|
||||||
@@ -50,15 +53,17 @@ class TaskTranscribe extends Task {
|
|||||||
this.profanityOption = recognizer.profanityOption || 'raw';
|
this.profanityOption = recognizer.profanityOption || 'raw';
|
||||||
this.requestSnr = recognizer.requestSnr || false;
|
this.requestSnr = recognizer.requestSnr || false;
|
||||||
this.initialSpeechTimeoutMs = recognizer.initialSpeechTimeoutMs || 0;
|
this.initialSpeechTimeoutMs = recognizer.initialSpeechTimeoutMs || 0;
|
||||||
|
this.azureServiceEndpoint = recognizer.azureServiceEndpoint;
|
||||||
}
|
}
|
||||||
|
|
||||||
get name() { return TaskName.Transcribe; }
|
get name() { return TaskName.Transcribe; }
|
||||||
|
|
||||||
async exec(cs, ep, parentTask) {
|
async exec(cs, ep, ep2) {
|
||||||
super.exec(cs);
|
super.exec(cs);
|
||||||
const {updateSpeechCredentialLastUsed} = require('../utils/db-utils')(this.logger, cs.srf);
|
const {updateSpeechCredentialLastUsed} = require('../utils/db-utils')(this.logger, cs.srf);
|
||||||
|
|
||||||
this.ep = ep;
|
this.ep = ep;
|
||||||
|
this.ep2 = ep2;
|
||||||
if ('default' === this.vendor || !this.vendor) this.vendor = cs.speechRecognizerVendor;
|
if ('default' === this.vendor || !this.vendor) this.vendor = cs.speechRecognizerVendor;
|
||||||
if ('default' === this.language || !this.language) this.language = cs.speechRecognizerLanguage;
|
if ('default' === this.language || !this.language) this.language = cs.speechRecognizerLanguage;
|
||||||
this.sttCredentials = cs.getSpeechCredentials(this.vendor, 'stt');
|
this.sttCredentials = cs.getSpeechCredentials(this.vendor, 'stt');
|
||||||
@@ -74,7 +79,9 @@ class TaskTranscribe extends Task {
|
|||||||
}).catch((err) => this.logger.info({err}, 'Error generating alert for no stt'));
|
}).catch((err) => this.logger.info({err}, 'Error generating alert for no stt'));
|
||||||
throw new Error('no provisioned speech credentials for TTS');
|
throw new Error('no provisioned speech credentials for TTS');
|
||||||
}
|
}
|
||||||
await this._startTranscribing(cs, ep);
|
await this._startTranscribing(cs, ep, 1);
|
||||||
|
if (this.separateRecognitionPerChannel && ep2) await this._startTranscribing(cs, ep2, 2);
|
||||||
|
|
||||||
updateSpeechCredentialLastUsed(this.sttCredentials.speech_credential_sid)
|
updateSpeechCredentialLastUsed(this.sttCredentials.speech_credential_sid)
|
||||||
.catch(() => {/*already logged error */});
|
.catch(() => {/*already logged error */});
|
||||||
|
|
||||||
@@ -102,11 +109,15 @@ class TaskTranscribe extends Task {
|
|||||||
// hangup after 1 sec if we don't get a final transcription
|
// hangup after 1 sec if we don't get a final transcription
|
||||||
this._timer = setTimeout(() => this.notifyTaskDone(), 1000);
|
this._timer = setTimeout(() => this.notifyTaskDone(), 1000);
|
||||||
}
|
}
|
||||||
|
if (this.separateRecognitionPerChannel && this.ep2 && this.ep2.connected) {
|
||||||
|
this.ep2.stopTranscription({vendor: this.vendor})
|
||||||
|
.catch((err) => this.logger.info(err, 'Error TaskTranscribe:kill'));
|
||||||
|
}
|
||||||
else this.notifyTaskDone();
|
else this.notifyTaskDone();
|
||||||
await this.awaitTaskDone();
|
await this.awaitTaskDone();
|
||||||
}
|
}
|
||||||
|
|
||||||
async _startTranscribing(cs, ep) {
|
async _startTranscribing(cs, ep, channel) {
|
||||||
const opts = {};
|
const opts = {};
|
||||||
|
|
||||||
if (this.vad.enable) {
|
if (this.vad.enable) {
|
||||||
@@ -115,42 +126,43 @@ class TaskTranscribe extends Task {
|
|||||||
if (this.vad.mode >= 0 && this.vad.mode <= 3) opts.RECOGNIZER_VAD_MODE = this.vad.mode;
|
if (this.vad.mode >= 0 && this.vad.mode <= 3) opts.RECOGNIZER_VAD_MODE = this.vad.mode;
|
||||||
}
|
}
|
||||||
|
|
||||||
ep.addCustomEventListener(GoogleTranscriptionEvents.Transcription, this._onTranscription.bind(this, cs, ep));
|
ep.addCustomEventListener(GoogleTranscriptionEvents.Transcription,
|
||||||
ep.addCustomEventListener(GoogleTranscriptionEvents.NoAudioDetected, this._onNoAudio.bind(this, cs, ep));
|
this._onTranscription.bind(this, cs, ep, channel));
|
||||||
|
ep.addCustomEventListener(GoogleTranscriptionEvents.NoAudioDetected, this._onNoAudio.bind(this, cs, ep, channel));
|
||||||
ep.addCustomEventListener(GoogleTranscriptionEvents.MaxDurationExceeded,
|
ep.addCustomEventListener(GoogleTranscriptionEvents.MaxDurationExceeded,
|
||||||
this._onMaxDurationExceeded.bind(this, cs, ep));
|
this._onMaxDurationExceeded.bind(this, cs, ep, channel));
|
||||||
ep.addCustomEventListener(AwsTranscriptionEvents.Transcription, this._onTranscription.bind(this, cs, ep));
|
ep.addCustomEventListener(AwsTranscriptionEvents.Transcription, this._onTranscription.bind(this, cs, ep, channel));
|
||||||
ep.addCustomEventListener(AwsTranscriptionEvents.NoAudioDetected, this._onNoAudio.bind(this, cs, ep));
|
ep.addCustomEventListener(AwsTranscriptionEvents.NoAudioDetected, this._onNoAudio.bind(this, cs, ep, channel));
|
||||||
ep.addCustomEventListener(AwsTranscriptionEvents.MaxDurationExceeded,
|
ep.addCustomEventListener(AwsTranscriptionEvents.MaxDurationExceeded,
|
||||||
this._onMaxDurationExceeded.bind(this, cs, ep));
|
this._onMaxDurationExceeded.bind(this, cs, ep, channel));
|
||||||
ep.addCustomEventListener(AzureTranscriptionEvents.Transcription, this._onTranscription.bind(this, cs, ep));
|
ep.addCustomEventListener(AzureTranscriptionEvents.Transcription,
|
||||||
ep.addCustomEventListener(AzureTranscriptionEvents.NoSpeechDetected, this._onNoAudio.bind(this, cs, ep));
|
this._onTranscription.bind(this, cs, ep, channel));
|
||||||
|
ep.addCustomEventListener(AzureTranscriptionEvents.NoSpeechDetected, this._onNoAudio.bind(this, cs, ep, channel));
|
||||||
|
|
||||||
if (this.vendor === 'google') {
|
if (this.vendor === 'google') {
|
||||||
if (this.sttCredentials) opts.GOOGLE_APPLICATION_CREDENTIALS = JSON.stringify(this.sttCredentials.credentials);
|
if (this.sttCredentials) opts.GOOGLE_APPLICATION_CREDENTIALS = JSON.stringify(this.sttCredentials.credentials);
|
||||||
[
|
[
|
||||||
['enhancedModel', 'GOOGLE_SPEECH_USE_ENHANCED'],
|
['enhancedModel', 'GOOGLE_SPEECH_USE_ENHANCED'],
|
||||||
['separateRecognitionPerChannel', 'GOOGLE_SPEECH_SEPARATE_RECOGNITION_PER_CHANNEL'],
|
//['separateRecognitionPerChannel', 'GOOGLE_SPEECH_SEPARATE_RECOGNITION_PER_CHANNEL'],
|
||||||
['profanityFilter', 'GOOGLE_SPEECH_PROFANITY_FILTER'],
|
['profanityFilter', 'GOOGLE_SPEECH_PROFANITY_FILTER'],
|
||||||
['punctuation', 'GOOGLE_SPEECH_ENABLE_AUTOMATIC_PUNCTUATION'],
|
['punctuation', 'GOOGLE_SPEECH_ENABLE_AUTOMATIC_PUNCTUATION'],
|
||||||
['words', 'GOOGLE_SPEECH_ENABLE_WORD_TIME_OFFSETS'],
|
['words', 'GOOGLE_SPEECH_ENABLE_WORD_TIME_OFFSETS'],
|
||||||
|
['singleUtterance', 'GOOGLE_SPEECH_SINGLE_UTTERANCE'],
|
||||||
['diarization', 'GOOGLE_SPEECH_PROFANITY_FILTER']
|
['diarization', 'GOOGLE_SPEECH_PROFANITY_FILTER']
|
||||||
].forEach((arr) => {
|
].forEach((arr) => {
|
||||||
if (this[arr[0]]) opts[arr[1]] = true;
|
if (this[arr[0]]) opts[arr[1]] = true;
|
||||||
});
|
});
|
||||||
if (this.hints.length > 1) opts.GOOGLE_SPEECH_HINTS = this.hints.join(',');
|
if (this.hints.length > 1) {
|
||||||
|
opts.GOOGLE_SPEECH_HINTS = this.hints.join(',');
|
||||||
|
if (typeof this.hintsBoost === 'number') {
|
||||||
|
opts.GOOGLE_SPEECH_HINTS_BOOST = this.hintsBoost;
|
||||||
|
}
|
||||||
|
}
|
||||||
if (this.altLanguages.length > 1) opts.GOOGLE_SPEECH_ALTERNATIVE_LANGUAGE_CODES = this.altLanguages.join(',');
|
if (this.altLanguages.length > 1) opts.GOOGLE_SPEECH_ALTERNATIVE_LANGUAGE_CODES = this.altLanguages.join(',');
|
||||||
if ('unspecified' !== this.interactionType) {
|
if ('unspecified' !== this.interactionType) {
|
||||||
opts.GOOGLE_SPEECH_METADATA_INTERACTION_TYPE = this.interactionType;
|
opts.GOOGLE_SPEECH_METADATA_INTERACTION_TYPE = this.interactionType;
|
||||||
|
|
||||||
// additionally set model if appropriate
|
|
||||||
if ('phone_call' === this.interactionType) opts.GOOGLE_SPEECH_MODEL = 'phone_call';
|
|
||||||
else if (['voice_search', 'voice_command'].includes(this.interactionType)) {
|
|
||||||
opts.GOOGLE_SPEECH_MODEL = 'command_and_search';
|
|
||||||
}
|
|
||||||
else opts.GOOGLE_SPEECH_MODEL = 'phone_call';
|
|
||||||
}
|
}
|
||||||
else opts.GOOGLE_SPEECH_MODEL = 'phone_call';
|
opts.GOOGLE_SPEECH_MODEL = this.model;
|
||||||
if (this.diarization && this.diarizationMinSpeakers > 0) {
|
if (this.diarization && this.diarizationMinSpeakers > 0) {
|
||||||
opts.GOOGLE_SPEECH_SPEAKER_DIARIZATION_MIN_SPEAKER_COUNT = this.diarizationMinSpeakers;
|
opts.GOOGLE_SPEECH_SPEAKER_DIARIZATION_MIN_SPEAKER_COUNT = this.diarizationMinSpeakers;
|
||||||
}
|
}
|
||||||
@@ -201,10 +213,12 @@ class TaskTranscribe extends Task {
|
|||||||
if (this.hints && this.hints.length > 1) {
|
if (this.hints && this.hints.length > 1) {
|
||||||
opts.AZURE_SPEECH_HINTS = this.hints.map((h) => h.trim()).join(',');
|
opts.AZURE_SPEECH_HINTS = this.hints.map((h) => h.trim()).join(',');
|
||||||
}
|
}
|
||||||
|
if (this.altLanguages.length > 1) opts.AZURE_SPEECH_ALTERNATIVE_LANGUAGE_CODES = this.altLanguages.join(',');
|
||||||
if (this.requestSnr) opts.AZURE_REQUEST_SNR = 1;
|
if (this.requestSnr) opts.AZURE_REQUEST_SNR = 1;
|
||||||
if (this.profanityOption !== 'raw') opts.AZURE_PROFANITY_OPTION = this.profanityOption;
|
if (this.profanityOption !== 'raw') opts.AZURE_PROFANITY_OPTION = this.profanityOption;
|
||||||
if (this.initialSpeechTimeoutMs > 0) opts.AZURE_INITIAL_SPEECH_TIMEOUT_MS = this.initialSpeechTimeoutMs;
|
if (this.initialSpeechTimeoutMs > 0) opts.AZURE_INITIAL_SPEECH_TIMEOUT_MS = this.initialSpeechTimeoutMs;
|
||||||
if (this.outputFormat !== 'simple') opts.AZURE_USE_OUTPUT_FORMAT_DETAILED = 1;
|
if (this.outputFormat !== 'simple') opts.AZURE_USE_OUTPUT_FORMAT_DETAILED = 1;
|
||||||
|
if (this.azureServiceEndpoint) opts.AZURE_SERVICE_ENDPOINT = this.azureServiceEndpoint;
|
||||||
|
|
||||||
await ep.set(opts)
|
await ep.set(opts)
|
||||||
.catch((err) => this.logger.info(err, 'TaskTranscribe:_startTranscribing with azure'));
|
.catch((err) => this.logger.info(err, 'TaskTranscribe:_startTranscribing with azure'));
|
||||||
@@ -217,15 +231,16 @@ class TaskTranscribe extends Task {
|
|||||||
vendor: this.vendor,
|
vendor: this.vendor,
|
||||||
interim: this.interim ? true : false,
|
interim: this.interim ? true : false,
|
||||||
locale: this.language,
|
locale: this.language,
|
||||||
channels: this.separateRecognitionPerChannel ? 2 : 1
|
channels: /*this.separateRecognitionPerChannel ? 2 : */ 1
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
_onTranscription(cs, ep, evt) {
|
_onTranscription(cs, ep, channel, evt) {
|
||||||
this.logger.debug(evt, 'TaskTranscribe:_onTranscription');
|
this.logger.debug({evt, channel}, 'TaskTranscribe:_onTranscription');
|
||||||
if ('aws' === this.vendor && Array.isArray(evt) && evt.length > 0) evt = evt[0];
|
if ('aws' === this.vendor && Array.isArray(evt) && evt.length > 0) evt = evt[0];
|
||||||
if ('microsoft' === this.vendor) {
|
if ('microsoft' === this.vendor) {
|
||||||
const nbest = evt.NBest;
|
const nbest = evt.NBest;
|
||||||
|
const language_code = evt.PrimaryLanguage?.Language || this.language;
|
||||||
const alternatives = nbest ? nbest.map((n) => {
|
const alternatives = nbest ? nbest.map((n) => {
|
||||||
return {
|
return {
|
||||||
confidence: n.Confidence,
|
confidence: n.Confidence,
|
||||||
@@ -240,13 +255,25 @@ class TaskTranscribe extends Task {
|
|||||||
|
|
||||||
const newEvent = {
|
const newEvent = {
|
||||||
is_final: evt.RecognitionStatus === 'Success',
|
is_final: evt.RecognitionStatus === 'Success',
|
||||||
|
channel,
|
||||||
|
language_code,
|
||||||
alternatives
|
alternatives
|
||||||
};
|
};
|
||||||
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);
|
||||||
|
}
|
||||||
|
|
||||||
|
evt.channel_tag = channel;
|
||||||
|
|
||||||
if (this.transcriptionHook) {
|
if (this.transcriptionHook) {
|
||||||
this.cs.requestor.request(this.transcriptionHook, Object.assign({speech: evt}, this.cs.callInfo))
|
const b3 = this.getTracingPropagation();
|
||||||
|
const httpHeaders = b3 && {b3};
|
||||||
|
this.cs.requestor.request('verb:hook', this.transcriptionHook,
|
||||||
|
Object.assign({speech: evt}, this.cs.callInfo), httpHeaders)
|
||||||
.catch((err) => this.logger.info(err, 'TranscribeTask:_onTranscription error'));
|
.catch((err) => this.logger.info(err, 'TranscribeTask:_onTranscription error'));
|
||||||
}
|
}
|
||||||
if (this.parentTask) {
|
if (this.parentTask) {
|
||||||
@@ -259,13 +286,13 @@ class TaskTranscribe extends Task {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
_onNoAudio(cs, ep) {
|
_onNoAudio(cs, ep, channel) {
|
||||||
this.logger.debug('TaskTranscribe:_onNoAudio restarting transcription');
|
this.logger.debug(`TaskTranscribe:_onNoAudio restarting transcription on channel ${channel}`);
|
||||||
this._transcribe(ep);
|
this._transcribe(ep);
|
||||||
}
|
}
|
||||||
|
|
||||||
_onMaxDurationExceeded(cs, ep) {
|
_onMaxDurationExceeded(cs, ep, channel) {
|
||||||
this.logger.debug('TaskTranscribe:_onMaxDurationExceeded restarting transcription');
|
this.logger.debug(`TaskTranscribe:_onMaxDurationExceeded restarting transcription on channel ${channel}`);
|
||||||
this._transcribe(ep);
|
this._transcribe(ep);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
75
lib/utils/base-requestor.js
Normal file
75
lib/utils/base-requestor.js
Normal file
@@ -0,0 +1,75 @@
|
|||||||
|
const assert = require('assert');
|
||||||
|
const Emitter = require('events');
|
||||||
|
const crypto = require('crypto');
|
||||||
|
const timeSeries = require('@jambonz/time-series');
|
||||||
|
let alerter ;
|
||||||
|
|
||||||
|
class BaseRequestor extends Emitter {
|
||||||
|
constructor(logger, account_sid, hook, secret) {
|
||||||
|
super();
|
||||||
|
assert(typeof hook === 'object');
|
||||||
|
|
||||||
|
this.logger = logger;
|
||||||
|
this.url = hook.url;
|
||||||
|
|
||||||
|
this.username = hook.username;
|
||||||
|
this.password = hook.password;
|
||||||
|
this.secret = secret;
|
||||||
|
this.account_sid = account_sid;
|
||||||
|
|
||||||
|
const {stats} = require('../../').srf.locals;
|
||||||
|
this.stats = stats;
|
||||||
|
|
||||||
|
if (!alerter) {
|
||||||
|
alerter = timeSeries(logger, {
|
||||||
|
host: process.env.JAMBONES_TIME_SERIES_HOST,
|
||||||
|
commitSize: 50,
|
||||||
|
commitInterval: 'test' === process.env.NODE_ENV ? 7 : 20
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
get Alerter() {
|
||||||
|
return alerter;
|
||||||
|
}
|
||||||
|
|
||||||
|
close() {
|
||||||
|
/* subclass responsibility */
|
||||||
|
}
|
||||||
|
|
||||||
|
_computeSignature(payload, timestamp, secret) {
|
||||||
|
assert(secret);
|
||||||
|
const data = `${timestamp}.${JSON.stringify(payload)}`;
|
||||||
|
return crypto
|
||||||
|
.createHmac('sha256', secret)
|
||||||
|
.update(data, 'utf8')
|
||||||
|
.digest('hex');
|
||||||
|
}
|
||||||
|
|
||||||
|
_generateSigHeader(payload, secret) {
|
||||||
|
const timestamp = Math.floor(Date.now() / 1000);
|
||||||
|
const signature = this._computeSignature(payload, timestamp, secret);
|
||||||
|
const scheme = 'v1';
|
||||||
|
return {
|
||||||
|
'Jambonz-Signature': `t=${timestamp},${scheme}=${signature}`
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
_isAbsoluteUrl(u) {
|
||||||
|
return typeof u === 'string' &&
|
||||||
|
u.startsWith('https://') || u.startsWith('http://') ||
|
||||||
|
u.startsWith('ws://') || u.startsWith('wss://');
|
||||||
|
}
|
||||||
|
_isRelativeUrl(u) {
|
||||||
|
return typeof u === 'string' && u.startsWith('/');
|
||||||
|
}
|
||||||
|
_roundTrip(startAt) {
|
||||||
|
const diff = process.hrtime(startAt);
|
||||||
|
const time = diff[0] * 1e3 + diff[1] * 1e-6;
|
||||||
|
return time.toFixed(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = BaseRequestor;
|
||||||
78
lib/utils/call-tracer.js
Normal file
78
lib/utils/call-tracer.js
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
const {context, trace} = require('@opentelemetry/api');
|
||||||
|
const {Dialog} = require('drachtio-srf');
|
||||||
|
class RootSpan {
|
||||||
|
constructor(callType, req) {
|
||||||
|
let tracer, callSid, linkedSpanId;
|
||||||
|
|
||||||
|
if (req instanceof Dialog) {
|
||||||
|
const dlg = req;
|
||||||
|
tracer = dlg.srf.locals.otel.tracer;
|
||||||
|
callSid = dlg.callSid;
|
||||||
|
linkedSpanId = dlg.linkedSpanId;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
tracer = req.srf.locals.otel.tracer;
|
||||||
|
callSid = req.locals.callSid;
|
||||||
|
}
|
||||||
|
this._span = tracer.startSpan(callType || 'incoming-call');
|
||||||
|
if (req instanceof Dialog) {
|
||||||
|
const dlg = req;
|
||||||
|
this._span.setAttributes({
|
||||||
|
linkedSpanId,
|
||||||
|
callId: dlg.sip.callId
|
||||||
|
});
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
this._span.setAttributes({
|
||||||
|
callSid,
|
||||||
|
accountSid: req.get('X-Account-Sid'),
|
||||||
|
applicationSid: req.locals.application_sid,
|
||||||
|
callId: req.get('Call-ID'),
|
||||||
|
externalCallId: req.get('X-CID')
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
this._ctx = trace.setSpan(context.active(), this._span);
|
||||||
|
this.tracer = tracer;
|
||||||
|
}
|
||||||
|
|
||||||
|
get context() {
|
||||||
|
return this._ctx;
|
||||||
|
}
|
||||||
|
|
||||||
|
get traceId() {
|
||||||
|
return this._span.spanContext().traceId;
|
||||||
|
}
|
||||||
|
|
||||||
|
get spanId() {
|
||||||
|
return this._span.spanContext().spanId;
|
||||||
|
}
|
||||||
|
|
||||||
|
get traceFlags() {
|
||||||
|
return this._span.spanContext().traceFlags;
|
||||||
|
}
|
||||||
|
|
||||||
|
getTracingPropagation(encoding) {
|
||||||
|
// TODO: support encodings beyond b3 https://github.com/openzipkin/b3-propagation
|
||||||
|
if (this._span && this.traceId !== '00000000000000000000000000000000') {
|
||||||
|
return `${this.traceId}-${this.spanId}-1`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setAttributes(attrs) {
|
||||||
|
this._span.setAttributes(attrs);
|
||||||
|
}
|
||||||
|
|
||||||
|
end() {
|
||||||
|
this._span.end();
|
||||||
|
}
|
||||||
|
|
||||||
|
startChildSpan(name, attributes) {
|
||||||
|
const span = this.tracer.startSpan(name, attributes, this._ctx);
|
||||||
|
const ctx = trace.setSpan(context.active(), span);
|
||||||
|
return {span, ctx};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = RootSpan;
|
||||||
|
|
||||||
@@ -2,6 +2,7 @@
|
|||||||
"TaskName": {
|
"TaskName": {
|
||||||
"Cognigy": "cognigy",
|
"Cognigy": "cognigy",
|
||||||
"Conference": "conference",
|
"Conference": "conference",
|
||||||
|
"Config": "config",
|
||||||
"Dequeue": "dequeue",
|
"Dequeue": "dequeue",
|
||||||
"Dial": "dial",
|
"Dial": "dial",
|
||||||
"Dialogflow": "dialogflow",
|
"Dialogflow": "dialogflow",
|
||||||
@@ -58,19 +59,22 @@
|
|||||||
"Transcription": "google_transcribe::transcription",
|
"Transcription": "google_transcribe::transcription",
|
||||||
"EndOfUtterance": "google_transcribe::end_of_utterance",
|
"EndOfUtterance": "google_transcribe::end_of_utterance",
|
||||||
"NoAudioDetected": "google_transcribe::no_audio_detected",
|
"NoAudioDetected": "google_transcribe::no_audio_detected",
|
||||||
"MaxDurationExceeded": "google_transcribe::max_duration_exceeded"
|
"MaxDurationExceeded": "google_transcribe::max_duration_exceeded",
|
||||||
|
"VadDetected": "google_transcribe::vad_detected"
|
||||||
},
|
},
|
||||||
"AwsTranscriptionEvents": {
|
"AwsTranscriptionEvents": {
|
||||||
"Transcription": "aws_transcribe::transcription",
|
"Transcription": "aws_transcribe::transcription",
|
||||||
"EndOfTranscript": "aws_transcribe::end_of_transcript",
|
"EndOfTranscript": "aws_transcribe::end_of_transcript",
|
||||||
"NoAudioDetected": "aws_transcribe::no_audio_detected",
|
"NoAudioDetected": "aws_transcribe::no_audio_detected",
|
||||||
"MaxDurationExceeded": "aws_transcribe::max_duration_exceeded"
|
"MaxDurationExceeded": "aws_transcribe::max_duration_exceeded",
|
||||||
|
"VadDetected": "aws_transcribe::vad_detected"
|
||||||
},
|
},
|
||||||
"AzureTranscriptionEvents": {
|
"AzureTranscriptionEvents": {
|
||||||
"Transcription": "azure_transcribe::transcription",
|
"Transcription": "azure_transcribe::transcription",
|
||||||
"StartOfUtterance": "azure_transcribe::start_of_utterance",
|
"StartOfUtterance": "azure_transcribe::start_of_utterance",
|
||||||
"EndOfUtterance": "azure_transcribe::end_of_utterance",
|
"EndOfUtterance": "azure_transcribe::end_of_utterance",
|
||||||
"NoSpeechDetected": "azure_transcribe::no_speech_detected"
|
"NoSpeechDetected": "azure_transcribe::no_speech_detected",
|
||||||
|
"VadDetected": "azure_transcribe::vad_detected"
|
||||||
},
|
},
|
||||||
"ListenEvents": {
|
"ListenEvents": {
|
||||||
"Connect": "mod_audio_fork::connect",
|
"Connect": "mod_audio_fork::connect",
|
||||||
@@ -105,6 +109,16 @@
|
|||||||
"Hangup": "hangup",
|
"Hangup": "hangup",
|
||||||
"Replaced": "replaced"
|
"Replaced": "replaced"
|
||||||
},
|
},
|
||||||
|
"HookMsgTypes": [
|
||||||
|
"session:new",
|
||||||
|
"session:reconnect",
|
||||||
|
"session:redirect",
|
||||||
|
"call:status",
|
||||||
|
"queue:status",
|
||||||
|
"dial:confirm",
|
||||||
|
"verb:hook",
|
||||||
|
"jambonz:error"
|
||||||
|
],
|
||||||
"MAX_SIMRINGS": 10,
|
"MAX_SIMRINGS": 10,
|
||||||
"BONG_TONE": "tone_stream://v=-7;%(100,0,941.0,1477.0);v=-7;>=2;+=.1;%(1400,0,350,440)",
|
"BONG_TONE": "tone_stream://v=-7;%(100,0,941.0,1477.0);v=-7;>=2;+=.1;%(1400,0,350,440)",
|
||||||
"FS_UUID_SET_NAME": "fsUUIDs"
|
"FS_UUID_SET_NAME": "fsUUIDs"
|
||||||
|
|||||||
52
lib/utils/cron-jobs.js
Normal file
52
lib/utils/cron-jobs.js
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
const {execSync} = require('child_process');
|
||||||
|
const now = Date.now();
|
||||||
|
const fsInventory = process.env.JAMBONES_FREESWITCH
|
||||||
|
.split(',')
|
||||||
|
.map((fs) => {
|
||||||
|
const arr = /^([^:]*):([^:]*):([^:]*)(?::([^:]*))?/.exec(fs);
|
||||||
|
const opts = {address: arr[1], port: arr[2], secret: arr[3]};
|
||||||
|
if (arr.length > 4) opts.advertisedAddress = arr[4];
|
||||||
|
if (process.env.NODE_ENV === 'test') opts.listenAddress = '0.0.0.0';
|
||||||
|
return opts;
|
||||||
|
});
|
||||||
|
|
||||||
|
const clearChannels = () => {
|
||||||
|
const {logger} = require('../..');
|
||||||
|
const pwd = fsInventory[0].secret;
|
||||||
|
const maxDurationMins = process.env.JAMBONES_FREESWITCH_MAX_CALL_DURATION_MINS || 180;
|
||||||
|
|
||||||
|
const calls = execSync(`/usr/local/freeswitch/bin/fs_cli -p ${pwd} -x "show calls"`, {encoding: 'utf8'})
|
||||||
|
.split('\n')
|
||||||
|
.filter((line) => line.match(/^[0-9A-Fa-f]{8}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{8}/))
|
||||||
|
.map((line) => {
|
||||||
|
const arr = line.split(',');
|
||||||
|
const dt = new Date(arr[2]);
|
||||||
|
const duration = (now - dt.getTime()) / 1000;
|
||||||
|
return {
|
||||||
|
uuid: arr[0],
|
||||||
|
time: arr[2],
|
||||||
|
duration
|
||||||
|
};
|
||||||
|
})
|
||||||
|
.filter((c) => c.duration > 60 * maxDurationMins);
|
||||||
|
|
||||||
|
if (calls.length > 0) {
|
||||||
|
logger.debug(`clearChannels: clearing ${calls.length} old calls longer than ${maxDurationMins} mins`);
|
||||||
|
for (const call of calls) {
|
||||||
|
const cmd = `/usr/local/freeswitch/bin/fs_cli -p ${pwd} -x "uuid_kill ${call.uuid}"`;
|
||||||
|
const out = execSync(cmd, {encoding: 'utf8'});
|
||||||
|
logger.debug({out}, 'clearChannels: command output');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return calls.length;
|
||||||
|
};
|
||||||
|
|
||||||
|
const clearFiles = () => {
|
||||||
|
const {logger} = require('../..');
|
||||||
|
const out = execSync('find /tmp -name "*.mp3" -mtime +2 -exec rm {} \\;');
|
||||||
|
logger.debug({out}, 'clearFiles: command output');
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
module.exports = {clearChannels, clearFiles};
|
||||||
|
|
||||||
@@ -48,6 +48,7 @@ module.exports = (logger, srf) => {
|
|||||||
const pp = pool.promise();
|
const pp = pool.promise();
|
||||||
|
|
||||||
const lookupAccountDetails = async(account_sid) => {
|
const lookupAccountDetails = async(account_sid) => {
|
||||||
|
|
||||||
const [r] = await pp.query({sql: sqlAccountDetails, nestTables: true}, account_sid);
|
const [r] = await pp.query({sql: sqlAccountDetails, nestTables: true}, account_sid);
|
||||||
if (0 === r.length) throw new Error(`invalid accountSid: ${account_sid}`);
|
if (0 === r.length) throw new Error(`invalid accountSid: ${account_sid}`);
|
||||||
const [r2] = await pp.query(sqlSpeechCredentials, account_sid);
|
const [r2] = await pp.query(sqlSpeechCredentials, account_sid);
|
||||||
|
|||||||
115
lib/utils/http-requestor.js
Normal file
115
lib/utils/http-requestor.js
Normal file
@@ -0,0 +1,115 @@
|
|||||||
|
const bent = require('bent');
|
||||||
|
const parseUrl = require('parse-url');
|
||||||
|
const assert = require('assert');
|
||||||
|
const BaseRequestor = require('./base-requestor');
|
||||||
|
const {HookMsgTypes} = require('./constants.json');
|
||||||
|
const snakeCaseKeys = require('./snakecase-keys');
|
||||||
|
|
||||||
|
const toBase64 = (str) => Buffer.from(str || '', 'utf8').toString('base64');
|
||||||
|
|
||||||
|
function basicAuth(username, password) {
|
||||||
|
if (!username || !password) return {};
|
||||||
|
const creds = `${username}:${password || ''}`;
|
||||||
|
const header = `Basic ${toBase64(creds)}`;
|
||||||
|
return {Authorization: header};
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class HttpRequestor extends BaseRequestor {
|
||||||
|
constructor(logger, account_sid, hook, secret) {
|
||||||
|
super(logger, account_sid, hook, secret);
|
||||||
|
|
||||||
|
this.method = hook.method || 'POST';
|
||||||
|
this.authHeader = basicAuth(hook.username, hook.password);
|
||||||
|
|
||||||
|
const u = parseUrl(this.url);
|
||||||
|
const myPort = u.port ? `:${u.port}` : '';
|
||||||
|
const baseUrl = this._baseUrl = `${u.protocol}://${u.resource}${myPort}`;
|
||||||
|
|
||||||
|
this.get = bent(baseUrl, 'GET', 'buffer', 200, 201);
|
||||||
|
this.post = bent(baseUrl, 'POST', 'buffer', 200, 201);
|
||||||
|
|
||||||
|
|
||||||
|
assert(this._isAbsoluteUrl(this.url));
|
||||||
|
assert(['GET', 'POST'].includes(this.method));
|
||||||
|
}
|
||||||
|
|
||||||
|
get baseUrl() {
|
||||||
|
return this._baseUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Make an HTTP request.
|
||||||
|
* All requests use json bodies.
|
||||||
|
* All requests expect a 200 statusCode on success
|
||||||
|
* @param {object|string} hook - may be a absolute or relative url, or an object
|
||||||
|
* @param {string} [hook.url] - an absolute or relative url
|
||||||
|
* @param {string} [hook.method] - 'GET' or 'POST'
|
||||||
|
* @param {string} [hook.username] - if basic auth is protecting the endpoint
|
||||||
|
* @param {string} [hook.password] - if basic auth is protecting the endpoint
|
||||||
|
* @param {object} [params] - request parameters
|
||||||
|
*/
|
||||||
|
async request(type, hook, params, httpHeaders = {}) {
|
||||||
|
/* jambonz:error only sent over ws */
|
||||||
|
if (type === 'jambonz:error') return;
|
||||||
|
|
||||||
|
assert(HookMsgTypes.includes(type));
|
||||||
|
|
||||||
|
const payload = params ? snakeCaseKeys(params, ['customerData', 'sip']) : null;
|
||||||
|
const url = hook.url || hook;
|
||||||
|
const method = hook.method || 'POST';
|
||||||
|
|
||||||
|
assert.ok(url, 'HttpRequestor:request url was not provided');
|
||||||
|
assert.ok, (['GET', 'POST'].includes(method), `HttpRequestor:request method must be 'GET' or 'POST' not ${method}`);
|
||||||
|
const {url: urlInfo = hook, method: methodInfo = 'POST'} = hook; // mask user/pass
|
||||||
|
this.logger.debug({url: urlInfo, method: methodInfo, payload}, `HttpRequestor:request ${method} ${url}`);
|
||||||
|
const startAt = process.hrtime();
|
||||||
|
|
||||||
|
let buf;
|
||||||
|
try {
|
||||||
|
const sigHeader = this._generateSigHeader(payload, this.secret);
|
||||||
|
const headers = {...sigHeader, ...this.authHeader, ...httpHeaders};
|
||||||
|
this.logger.debug({url, headers}, 'send webhook');
|
||||||
|
buf = this._isRelativeUrl(url) ?
|
||||||
|
await this.post(url, payload, headers) :
|
||||||
|
await bent(method, 'buffer', 200, 201, 202)(url, payload, headers);
|
||||||
|
} catch (err) {
|
||||||
|
if (err.statusCode) {
|
||||||
|
this.logger.info({baseUrl: this.baseUrl, url},
|
||||||
|
`web callback returned unexpected status code ${err.statusCode}`);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
this.logger.error({err, baseUrl: this.baseUrl, url},
|
||||||
|
'web callback returned unexpected error');
|
||||||
|
}
|
||||||
|
let opts = {account_sid: this.account_sid};
|
||||||
|
if (err.code === 'ECONNREFUSED') {
|
||||||
|
opts = {...opts, alert_type: this.Alerter.AlertType.WEBHOOK_CONNECTION_FAILURE, url};
|
||||||
|
}
|
||||||
|
else if (err.name === 'StatusError') {
|
||||||
|
opts = {...opts, alert_type: this.Alerter.AlertType.WEBHOOK_STATUS_FAILURE, url, status: err.statusCode};
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
opts = {...opts, alert_type: this.Alerter.AlertType.WEBHOOK_CONNECTION_FAILURE, url, detail: err.message};
|
||||||
|
}
|
||||||
|
this.Alerter.writeAlerts(opts).catch((err) => this.logger.info({err, opts}, 'Error writing alert'));
|
||||||
|
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
const rtt = this._roundTrip(startAt);
|
||||||
|
if (buf) this.stats.histogram('app.hook.response_time', rtt, ['hook_type:app']);
|
||||||
|
|
||||||
|
if (buf && buf.toString().length > 0) {
|
||||||
|
try {
|
||||||
|
const json = JSON.parse(buf.toString());
|
||||||
|
this.logger.info({response: json}, `HttpRequestor:request ${method} ${url} succeeded in ${rtt}ms`);
|
||||||
|
return json;
|
||||||
|
}
|
||||||
|
catch (err) {
|
||||||
|
//this.logger.debug({err, url, method}, `HttpRequestor:request returned non-JSON content: '${buf.toString()}'`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = HttpRequestor;
|
||||||
@@ -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');
|
||||||
|
|
||||||
@@ -32,6 +31,7 @@ function initMS(logger, wrapper, ms) {
|
|||||||
function installSrfLocals(srf, logger) {
|
function installSrfLocals(srf, logger) {
|
||||||
logger.debug('installing srf locals');
|
logger.debug('installing srf locals');
|
||||||
assert(!srf.locals.dbHelpers);
|
assert(!srf.locals.dbHelpers);
|
||||||
|
const {tracer} = srf.locals.otel;
|
||||||
const {getSBC, lifecycleEmitter} = require('./sbc-pinger')(logger);
|
const {getSBC, lifecycleEmitter} = require('./sbc-pinger')(logger);
|
||||||
const StatsCollector = require('@jambonz/stats-collector');
|
const StatsCollector = require('@jambonz/stats-collector');
|
||||||
const stats = srf.locals.stats = new StatsCollector(logger);
|
const stats = srf.locals.stats = new StatsCollector(logger);
|
||||||
@@ -49,7 +49,11 @@ function installSrfLocals(srf, logger) {
|
|||||||
assert.ok(arr, `Invalid syntax JAMBONES_FREESWITCH: ${process.env.JAMBONES_FREESWITCH}`);
|
assert.ok(arr, `Invalid syntax JAMBONES_FREESWITCH: ${process.env.JAMBONES_FREESWITCH}`);
|
||||||
const opts = {address: arr[1], port: arr[2], secret: arr[3]};
|
const opts = {address: arr[1], port: arr[2], secret: arr[3]};
|
||||||
if (arr.length > 4) opts.advertisedAddress = arr[4];
|
if (arr.length > 4) opts.advertisedAddress = arr[4];
|
||||||
|
/* NB: originally for testing only, but for now all jambonz deployments
|
||||||
|
have freeswitch installed locally alongside this app
|
||||||
|
*/
|
||||||
if (process.env.NODE_ENV === 'test') opts.listenAddress = '0.0.0.0';
|
if (process.env.NODE_ENV === 'test') opts.listenAddress = '0.0.0.0';
|
||||||
|
else if (process.env.JAMBONES_ESL_LISTEN_ADDRESS) opts.listenAddress = process.env.JAMBONES_ESL_LISTEN_ADDRESS;
|
||||||
return opts;
|
return opts;
|
||||||
});
|
});
|
||||||
logger.info({fsInventory}, 'freeswitch inventory');
|
logger.info({fsInventory}, 'freeswitch inventory');
|
||||||
@@ -62,7 +66,7 @@ function installSrfLocals(srf, logger) {
|
|||||||
initMS(logger, val, ms);
|
initMS(logger, val, ms);
|
||||||
}
|
}
|
||||||
catch (err) {
|
catch (err) {
|
||||||
logger.info(`failed connecting to freeswitch at ${fs.address}, will retry shortly`);
|
logger.info({err}, `failed connecting to freeswitch at ${fs.address}, will retry shortly: ${err.message}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// retry to connect to any that were initially offline
|
// retry to connect to any that were initially offline
|
||||||
@@ -74,7 +78,7 @@ function installSrfLocals(srf, logger) {
|
|||||||
const ms = await mrf.connect(val.opts);
|
const ms = await mrf.connect(val.opts);
|
||||||
initMS(logger, val, ms);
|
initMS(logger, val, ms);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
logger.info(`failed connecting to freeswitch at ${val.opts.address}, will retry shortly`);
|
logger.info({err}, `failed connecting to freeswitch at ${val.opts.address}, will retry shortly`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -127,7 +131,7 @@ function installSrfLocals(srf, logger) {
|
|||||||
password: process.env.JAMBONES_MYSQL_PASSWORD,
|
password: process.env.JAMBONES_MYSQL_PASSWORD,
|
||||||
database: process.env.JAMBONES_MYSQL_DATABASE,
|
database: process.env.JAMBONES_MYSQL_DATABASE,
|
||||||
connectionLimit: process.env.JAMBONES_MYSQL_CONNECTION_LIMIT || 10
|
connectionLimit: process.env.JAMBONES_MYSQL_CONNECTION_LIMIT || 10
|
||||||
}, logger);
|
}, logger, tracer);
|
||||||
const {
|
const {
|
||||||
client,
|
client,
|
||||||
updateCallStatus,
|
updateCallStatus,
|
||||||
@@ -152,7 +156,7 @@ function installSrfLocals(srf, logger) {
|
|||||||
} = require('@jambonz/realtimedb-helpers')({
|
} = require('@jambonz/realtimedb-helpers')({
|
||||||
host: process.env.JAMBONES_REDIS_HOST,
|
host: process.env.JAMBONES_REDIS_HOST,
|
||||||
port: process.env.JAMBONES_REDIS_PORT || 6379
|
port: process.env.JAMBONES_REDIS_PORT || 6379
|
||||||
}, logger);
|
}, logger, tracer);
|
||||||
const {
|
const {
|
||||||
writeAlerts,
|
writeAlerts,
|
||||||
AlertType
|
AlertType
|
||||||
@@ -162,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,
|
||||||
@@ -196,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;
|
||||||
@@ -208,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;
|
||||||
|
|||||||
@@ -4,15 +4,18 @@ const SipError = require('drachtio-srf').SipError;
|
|||||||
const {TaskPreconditions, CallDirection} = require('../utils/constants');
|
const {TaskPreconditions, CallDirection} = require('../utils/constants');
|
||||||
const CallInfo = require('../session/call-info');
|
const CallInfo = require('../session/call-info');
|
||||||
const assert = require('assert');
|
const assert = require('assert');
|
||||||
|
const normalizeJambones = require('../utils/normalize-jambones');
|
||||||
|
const makeTask = require('../tasks/make_task');
|
||||||
const ConfirmCallSession = require('../session/confirm-call-session');
|
const ConfirmCallSession = require('../session/confirm-call-session');
|
||||||
const AdultingCallSession = require('../session/adulting-call-session');
|
const AdultingCallSession = require('../session/adulting-call-session');
|
||||||
const deepcopy = require('deepcopy');
|
const deepcopy = require('deepcopy');
|
||||||
const moment = require('moment');
|
const moment = require('moment');
|
||||||
const stripCodecs = require('./strip-ancillary-codecs');
|
const stripCodecs = require('./strip-ancillary-codecs');
|
||||||
|
const RootSpan = require('./call-tracer');
|
||||||
const { v4: uuidv4 } = require('uuid');
|
const { v4: uuidv4 } = require('uuid');
|
||||||
|
|
||||||
class SingleDialer extends Emitter {
|
class SingleDialer extends Emitter {
|
||||||
constructor({logger, sbcAddress, target, opts, application, callInfo, accountInfo}) {
|
constructor({logger, sbcAddress, target, opts, application, callInfo, accountInfo, rootSpan, startSpan}) {
|
||||||
super();
|
super();
|
||||||
assert(target.type);
|
assert(target.type);
|
||||||
|
|
||||||
@@ -22,6 +25,8 @@ class SingleDialer extends Emitter {
|
|||||||
this.opts = opts;
|
this.opts = opts;
|
||||||
this.application = application;
|
this.application = application;
|
||||||
this.confirmHook = target.confirmHook;
|
this.confirmHook = target.confirmHook;
|
||||||
|
this.rootSpan = rootSpan;
|
||||||
|
this.startSpan = startSpan;
|
||||||
|
|
||||||
this.bindings = logger.bindings();
|
this.bindings = logger.bindings();
|
||||||
|
|
||||||
@@ -60,12 +65,18 @@ class SingleDialer extends Emitter {
|
|||||||
opts.headers = opts.headers || {};
|
opts.headers = opts.headers || {};
|
||||||
opts.headers = {
|
opts.headers = {
|
||||||
...opts.headers,
|
...opts.headers,
|
||||||
|
...(this.target.headers || {}),
|
||||||
'X-Jambonz-Routing': this.target.type,
|
'X-Jambonz-Routing': this.target.type,
|
||||||
'X-Jambonz-FS-UUID': srf.locals.fsUUID,
|
|
||||||
'X-Call-Sid': this.callSid
|
'X-Call-Sid': this.callSid
|
||||||
};
|
};
|
||||||
|
if (srf.locals.fsUUID) {
|
||||||
|
opts.headers = {
|
||||||
|
...opts.headers,
|
||||||
|
'X-Jambonz-FS-UUID': srf.locals.fsUUID,
|
||||||
|
};
|
||||||
|
}
|
||||||
this.ms = ms;
|
this.ms = ms;
|
||||||
let uri, to;
|
let uri, to, inviteSpan;
|
||||||
try {
|
try {
|
||||||
switch (this.target.type) {
|
switch (this.target.type) {
|
||||||
case 'phone':
|
case 'phone':
|
||||||
@@ -131,13 +142,24 @@ class SingleDialer extends Emitter {
|
|||||||
localSdp: this.ep.local.sdp
|
localSdp: this.ep.local.sdp
|
||||||
});
|
});
|
||||||
if (this.target.auth) opts.auth = this.target.auth;
|
if (this.target.auth) opts.auth = this.target.auth;
|
||||||
|
inviteSpan = this.startSpan('invite', {
|
||||||
|
'invite.uri': uri,
|
||||||
|
'invite.dest_type': this.target.type
|
||||||
|
});
|
||||||
|
|
||||||
this.dlg = await srf.createUAC(uri, {...opts, followRedirects: true, keepUriOnRedirect: true}, {
|
this.dlg = await srf.createUAC(uri, {...opts, followRedirects: true, keepUriOnRedirect: true}, {
|
||||||
cbRequest: (err, req) => {
|
cbRequest: (err, req) => {
|
||||||
if (err) {
|
if (err) {
|
||||||
this.logger.error(err, 'SingleDialer:exec Error creating call');
|
this.logger.error(err, 'SingleDialer:exec Error creating call');
|
||||||
this.emit('callCreateFail', err);
|
this.emit('callCreateFail', err);
|
||||||
|
inviteSpan.setAttributes({
|
||||||
|
'invite.status_code': 500,
|
||||||
|
'invite.err': err.message
|
||||||
|
});
|
||||||
|
inviteSpan.end();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
inviteSpan.setAttributes({'invite.call_id': req.get('Call-ID')});
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* INVITE has been sent out
|
* INVITE has been sent out
|
||||||
@@ -150,7 +172,8 @@ class SingleDialer extends Emitter {
|
|||||||
parentCallInfo: this.parentCallInfo,
|
parentCallInfo: this.parentCallInfo,
|
||||||
req,
|
req,
|
||||||
to,
|
to,
|
||||||
callSid: this.callSid
|
callSid: this.callSid,
|
||||||
|
traceId: this.rootSpan.traceId
|
||||||
});
|
});
|
||||||
this.logger = srf.locals.parentLogger.child({
|
this.logger = srf.locals.parentLogger.child({
|
||||||
callSid: this.callSid,
|
callSid: this.callSid,
|
||||||
@@ -158,10 +181,14 @@ class SingleDialer extends Emitter {
|
|||||||
callId: this.callInfo.callId
|
callId: this.callInfo.callId
|
||||||
});
|
});
|
||||||
this.inviteInProgress = req;
|
this.inviteInProgress = req;
|
||||||
this.emit('callStatusChange', {callStatus: CallStatus.Trying, sipStatus: 100});
|
this.emit('callStatusChange', {
|
||||||
|
callStatus: CallStatus.Trying,
|
||||||
|
sipStatus: 100,
|
||||||
|
sipReason: 'Trying'
|
||||||
|
});
|
||||||
},
|
},
|
||||||
cbProvisional: (prov) => {
|
cbProvisional: (prov) => {
|
||||||
const status = {sipStatus: prov.status};
|
const status = {sipStatus: prov.status, sipReason: prov.reason};
|
||||||
if ([180, 183].includes(prov.status) && prov.body) {
|
if ([180, 183].includes(prov.status) && prov.body) {
|
||||||
if (status.callStatus !== CallStatus.EarlyMedia) {
|
if (status.callStatus !== CallStatus.EarlyMedia) {
|
||||||
status.callStatus = CallStatus.EarlyMedia;
|
status.callStatus = CallStatus.EarlyMedia;
|
||||||
@@ -176,15 +203,27 @@ class SingleDialer extends Emitter {
|
|||||||
await connectStream(this.dlg.remote.sdp);
|
await connectStream(this.dlg.remote.sdp);
|
||||||
this.dlg.callSid = this.callSid;
|
this.dlg.callSid = this.callSid;
|
||||||
this.inviteInProgress = null;
|
this.inviteInProgress = null;
|
||||||
this.emit('callStatusChange', {sipStatus: 200, callStatus: CallStatus.InProgress});
|
this.emit('callStatusChange', {
|
||||||
|
sipStatus: 200,
|
||||||
|
sipReason: 'OK',
|
||||||
|
callStatus: CallStatus.InProgress
|
||||||
|
});
|
||||||
this.logger.debug(`SingleDialer:exec call connected: ${this.callSid}`);
|
this.logger.debug(`SingleDialer:exec call connected: ${this.callSid}`);
|
||||||
const connectTime = this.dlg.connectTime = moment();
|
const connectTime = this.dlg.connectTime = moment();
|
||||||
|
inviteSpan.setAttributes({'invite.status_code': 200});
|
||||||
|
inviteSpan.end();
|
||||||
|
|
||||||
|
|
||||||
/* race condition: we were killed just as call was answered */
|
/* race condition: we were killed just as call was answered */
|
||||||
if (this.killed) {
|
if (this.killed) {
|
||||||
this.logger.info(`SingleDialer:exec race condition - we were killed as call connected: ${this.callSid}`);
|
this.logger.info(`SingleDialer:exec race condition - we were killed as call connected: ${this.callSid}`);
|
||||||
const duration = moment().diff(connectTime, 'seconds');
|
const duration = moment().diff(connectTime, 'seconds');
|
||||||
this.emit('callStatusChange', {callStatus: CallStatus.Completed, duration});
|
this.emit('callStatusChange', {
|
||||||
|
callStatus: CallStatus.Completed,
|
||||||
|
sipStatus: 487,
|
||||||
|
sipReason: 'Request Terminated',
|
||||||
|
duration
|
||||||
|
});
|
||||||
if (this.ep) this.ep.destroy();
|
if (this.ep) this.ep.destroy();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -211,6 +250,9 @@ class SingleDialer extends Emitter {
|
|||||||
} catch (err) {
|
} catch (err) {
|
||||||
this.logger.error(err, 'Error handling reinvite');
|
this.logger.error(err, 'Error handling reinvite');
|
||||||
}
|
}
|
||||||
|
})
|
||||||
|
.on('refer', (req, res) => {
|
||||||
|
this.emit('refer', this.callInfo, req, res);
|
||||||
});
|
});
|
||||||
|
|
||||||
if (this.confirmHook) this._executeApp(this.confirmHook);
|
if (this.confirmHook) this._executeApp(this.confirmHook);
|
||||||
@@ -220,13 +262,21 @@ class SingleDialer extends Emitter {
|
|||||||
const status = {callStatus: CallStatus.Failed};
|
const status = {callStatus: CallStatus.Failed};
|
||||||
if (err instanceof SipError) {
|
if (err instanceof SipError) {
|
||||||
status.sipStatus = err.status;
|
status.sipStatus = err.status;
|
||||||
|
status.sipReason = err.reason;
|
||||||
if (err.status === 487) status.callStatus = CallStatus.NoAnswer;
|
if (err.status === 487) status.callStatus = CallStatus.NoAnswer;
|
||||||
else if ([486, 600].includes(err.status)) status.callStatus = CallStatus.Busy;
|
else if ([486, 600].includes(err.status)) status.callStatus = CallStatus.Busy;
|
||||||
this.logger.info(`SingleDialer:exec outdial failure ${err.status}`);
|
this.logger.info(`SingleDialer:exec outdial failure ${err.status}`);
|
||||||
|
inviteSpan.setAttributes({'invite.status_code': err.status});
|
||||||
|
inviteSpan.end();
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
this.logger.error(err, 'SingleDialer:exec');
|
this.logger.error(err, 'SingleDialer:exec');
|
||||||
status.sipStatus = 500;
|
status.sipStatus = 500;
|
||||||
|
inviteSpan.setAttributes({
|
||||||
|
'invite.status_code': 500,
|
||||||
|
'invite.err': err.message
|
||||||
|
});
|
||||||
|
inviteSpan.end();
|
||||||
}
|
}
|
||||||
this.emit('callStatusChange', status);
|
this.emit('callStatusChange', status);
|
||||||
if (this.ep) this.ep.destroy();
|
if (this.ep) this.ep.destroy();
|
||||||
@@ -261,8 +311,8 @@ class SingleDialer extends Emitter {
|
|||||||
async _executeApp(confirmHook) {
|
async _executeApp(confirmHook) {
|
||||||
try {
|
try {
|
||||||
// retrieve set of tasks
|
// retrieve set of tasks
|
||||||
const tasks = await this.requestor.request(confirmHook, this.callInfo.toJSON());
|
const json = await this.requestor.request('dial:confirm', confirmHook, this.callInfo.toJSON());
|
||||||
|
const tasks = normalizeJambones(this.logger, json).map((tdata) => makeTask(this.logger, tdata));
|
||||||
// verify it contains only allowed verbs
|
// verify it contains only allowed verbs
|
||||||
const allowedTasks = tasks.filter((task) => {
|
const allowedTasks = tasks.filter((task) => {
|
||||||
return [
|
return [
|
||||||
@@ -282,7 +332,9 @@ class SingleDialer extends Emitter {
|
|||||||
dlg: this.dlg,
|
dlg: this.dlg,
|
||||||
ep: this.ep,
|
ep: this.ep,
|
||||||
callInfo: this.callInfo,
|
callInfo: this.callInfo,
|
||||||
tasks
|
accountInfo: this.accountInfo,
|
||||||
|
tasks,
|
||||||
|
rootSpan: this.rootSpan
|
||||||
});
|
});
|
||||||
await cs.exec();
|
await cs.exec();
|
||||||
|
|
||||||
@@ -296,7 +348,6 @@ class SingleDialer extends Emitter {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async doAdulting({logger, tasks, application}) {
|
async doAdulting({logger, tasks, application}) {
|
||||||
this.logger = logger;
|
|
||||||
this.adulting = true;
|
this.adulting = true;
|
||||||
this.emit('adulting');
|
this.emit('adulting');
|
||||||
if (this.ep) {
|
if (this.ep) {
|
||||||
@@ -307,15 +358,21 @@ class SingleDialer extends Emitter {
|
|||||||
else {
|
else {
|
||||||
await this.reAnchorMedia();
|
await this.reAnchorMedia();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.dlg.callSid = this.callSid;
|
||||||
|
this.dlg.linkedSpanId = this.rootSpan.traceId;
|
||||||
|
const rootSpan = new RootSpan('outbound-call', this.dlg);
|
||||||
|
const newLogger = logger.child({traceId: rootSpan.traceId});
|
||||||
const cs = new AdultingCallSession({
|
const cs = new AdultingCallSession({
|
||||||
logger: this.logger,
|
logger: newLogger,
|
||||||
singleDialer: this,
|
singleDialer: this,
|
||||||
application,
|
application,
|
||||||
callInfo: this.callInfo,
|
callInfo: this.callInfo,
|
||||||
accountInfo: this.accountInfo,
|
accountInfo: this.accountInfo,
|
||||||
tasks
|
tasks,
|
||||||
|
rootSpan
|
||||||
});
|
});
|
||||||
cs.exec();
|
cs.exec().catch((err) => newLogger.error({err}, 'doAdulting: error executing session'));
|
||||||
return cs;
|
return cs;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -342,16 +399,16 @@ class SingleDialer extends Emitter {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
_notifyCallStatusChange({callStatus, sipStatus, duration}) {
|
_notifyCallStatusChange({callStatus, sipStatus, sipReason, duration}) {
|
||||||
assert((typeof duration === 'number' && callStatus === CallStatus.Completed) ||
|
assert((typeof duration === 'number' && callStatus === CallStatus.Completed) ||
|
||||||
(!duration && callStatus !== CallStatus.Completed),
|
(!duration && callStatus !== CallStatus.Completed),
|
||||||
'duration MUST be supplied when call completed AND ONLY when call completed');
|
'duration MUST be supplied when call completed AND ONLY when call completed');
|
||||||
|
|
||||||
if (this.callInfo) {
|
if (this.callInfo) {
|
||||||
this.callInfo.updateCallStatus(callStatus, sipStatus);
|
this.callInfo.updateCallStatus(callStatus, sipStatus, sipReason);
|
||||||
if (typeof duration === 'number') this.callInfo.duration = duration;
|
if (typeof duration === 'number') this.callInfo.duration = duration;
|
||||||
try {
|
try {
|
||||||
this.requestor.request(this.application.call_status_hook, this.callInfo.toJSON());
|
this.requestor.request('call:status', this.application.call_status_hook, this.callInfo.toJSON());
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this.logger.info(err, `SingleDialer:_notifyCallStatusChange error sending ${callStatus} ${sipStatus}`);
|
this.logger.info(err, `SingleDialer:_notifyCallStatusChange error sending ${callStatus} ${sipStatus}`);
|
||||||
}
|
}
|
||||||
@@ -364,9 +421,13 @@ class SingleDialer extends Emitter {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function placeOutdial({logger, srf, ms, sbcAddress, target, opts, application, callInfo, accountInfo}) {
|
function placeOutdial({
|
||||||
|
logger, srf, ms, sbcAddress, target, opts, application, callInfo, accountInfo, rootSpan, startSpan
|
||||||
|
}) {
|
||||||
const myOpts = deepcopy(opts);
|
const myOpts = deepcopy(opts);
|
||||||
const sd = new SingleDialer({logger, sbcAddress, target, myOpts, application, callInfo, accountInfo});
|
const sd = new SingleDialer({
|
||||||
|
logger, sbcAddress, target, myOpts, application, callInfo, accountInfo, rootSpan, startSpan
|
||||||
|
});
|
||||||
sd.exec(srf, ms, myOpts);
|
sd.exec(srf, ms, myOpts);
|
||||||
return sd;
|
return sd;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -17,6 +17,10 @@ module.exports = (logger) => {
|
|||||||
assert.ok(sbcs.length, 'JAMBONES_SBCS env var is empty or misconfigured');
|
assert.ok(sbcs.length, 'JAMBONES_SBCS env var is empty or misconfigured');
|
||||||
logger.info({sbcs}, 'SBC inventory');
|
logger.info({sbcs}, 'SBC inventory');
|
||||||
}
|
}
|
||||||
|
else if (process.env.K8S && process.env.K8S_SBC_SIP_SERVICE_NAME) {
|
||||||
|
sbcs = [`${process.env.K8S_SBC_SIP_SERVICE_NAME}:5060`];
|
||||||
|
logger.info({sbcs}, 'SBC inventory');
|
||||||
|
}
|
||||||
|
|
||||||
// listen for SNS lifecycle changes
|
// listen for SNS lifecycle changes
|
||||||
let lifecycleEmitter = new Emitter();
|
let lifecycleEmitter = new Emitter();
|
||||||
|
|||||||
@@ -1,3 +1,3 @@
|
|||||||
module.exports = function(tasks) {
|
module.exports = function(tasks) {
|
||||||
return `[${tasks.map((t) => t.name).join(',')}]`;
|
return `[${tasks.map((t) => t.summary).join(',')}]`;
|
||||||
};
|
};
|
||||||
|
|||||||
331
lib/utils/ws-requestor.js
Normal file
331
lib/utils/ws-requestor.js
Normal file
@@ -0,0 +1,331 @@
|
|||||||
|
const assert = require('assert');
|
||||||
|
const BaseRequestor = require('./base-requestor');
|
||||||
|
const short = require('short-uuid');
|
||||||
|
const {HookMsgTypes} = require('./constants.json');
|
||||||
|
const Websocket = require('ws');
|
||||||
|
const snakeCaseKeys = require('./snakecase-keys');
|
||||||
|
const HttpRequestor = require('./http-requestor');
|
||||||
|
const MAX_RECONNECTS = 5;
|
||||||
|
const RESPONSE_TIMEOUT_MS = process.env.JAMBONES_WS_API_MSG_RESPONSE_TIMEOUT || 5000;
|
||||||
|
|
||||||
|
class WsRequestor extends BaseRequestor {
|
||||||
|
constructor(logger, account_sid, hook, secret) {
|
||||||
|
super(logger, account_sid, hook, secret);
|
||||||
|
this.connections = 0;
|
||||||
|
this.messagesInFlight = new Map();
|
||||||
|
this.maliciousClient = false;
|
||||||
|
this.closedGracefully = false;
|
||||||
|
this.backoffMs = 500;
|
||||||
|
this.connectInProgress = false;
|
||||||
|
this.queuedMsg = [];
|
||||||
|
this.id = short.generate();
|
||||||
|
|
||||||
|
assert(this._isAbsoluteUrl(this.url));
|
||||||
|
|
||||||
|
this.on('socket-closed', this._onSocketClosed.bind(this));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send a JSON payload over the websocket. If this is the first request,
|
||||||
|
* open the websocket.
|
||||||
|
* All requests expect an ack message in response
|
||||||
|
* @param {object|string} hook - may be a absolute or relative url, or an object
|
||||||
|
* @param {string} [hook.url] - an absolute or relative url
|
||||||
|
* @param {string} [hook.method] - 'GET' or 'POST'
|
||||||
|
* @param {string} [hook.username] - if basic auth is protecting the endpoint
|
||||||
|
* @param {string} [hook.password] - if basic auth is protecting the endpoint
|
||||||
|
* @param {object} [params] - request parameters
|
||||||
|
*/
|
||||||
|
async request(type, hook, params, httpHeaders = {}) {
|
||||||
|
assert(HookMsgTypes.includes(type));
|
||||||
|
const url = hook.url || hook;
|
||||||
|
|
||||||
|
if (this.maliciousClient) {
|
||||||
|
this.logger.info({url: this.url}, 'WsRequestor:request - discarding msg to malicious client');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (this.closedGracefully) {
|
||||||
|
this.logger.debug(`WsRequestor:request - discarding ${type} because we closed the socket`);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (type === 'session:new') this.call_sid = params.callSid;
|
||||||
|
|
||||||
|
/* if we have an absolute url, and it is http then do a standard webhook */
|
||||||
|
if (this._isAbsoluteUrl(url) && url.startsWith('http')) {
|
||||||
|
this.logger.debug({hook}, 'WsRequestor: sending a webhook (HTTP)');
|
||||||
|
const requestor = new HttpRequestor(this.logger, this.account_sid, hook, this.secret);
|
||||||
|
return requestor.request(type, hook, params, httpHeaders);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* connect if necessary */
|
||||||
|
if (!this.ws) {
|
||||||
|
if (this.connectInProgress) {
|
||||||
|
this.logger.debug(
|
||||||
|
`WsRequestor:request(${this.id}) - queueing ${type} message since we are connecting`);
|
||||||
|
this.queuedMsg.push({type, hook, params, httpHeaders});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.connectInProgress = true;
|
||||||
|
this.logger.debug(`WsRequestor:request(${this.id}) - connecting since we do not have a connection`);
|
||||||
|
if (this.connections >= MAX_RECONNECTS) {
|
||||||
|
throw new Error(`max attempts connecting to ${this.url}`);
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
const startAt = process.hrtime();
|
||||||
|
await this._connect();
|
||||||
|
const rtt = this._roundTrip(startAt);
|
||||||
|
this.stats.histogram('app.hook.connect_time', rtt, ['hook_type:app']);
|
||||||
|
} catch (err) {
|
||||||
|
this.logger.info({url, err}, 'WsRequestor:request - failed connecting');
|
||||||
|
this.connectInProgress = false;
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
assert(this.ws);
|
||||||
|
|
||||||
|
/* prepare and send message */
|
||||||
|
let payload = params ? snakeCaseKeys(params, ['customerData', 'sip']) : null;
|
||||||
|
if (type === 'session:new') this._sessionData = payload;
|
||||||
|
if (type === 'session:reconnect') payload = this._sessionData;
|
||||||
|
assert.ok(url, 'WsRequestor:request url was not provided');
|
||||||
|
|
||||||
|
const msgid = short.generate();
|
||||||
|
const b3 = httpHeaders?.b3 ? {b3: httpHeaders.b3} : {};
|
||||||
|
const obj = {
|
||||||
|
type,
|
||||||
|
msgid,
|
||||||
|
call_sid: this.call_sid,
|
||||||
|
hook: type === 'verb:hook' ? url : undefined,
|
||||||
|
data: {...payload},
|
||||||
|
...b3
|
||||||
|
};
|
||||||
|
|
||||||
|
const sendQueuedMsgs = () => {
|
||||||
|
if (this.queuedMsg.length > 0) {
|
||||||
|
for (const {type, hook, params, httpHeaders} of this.queuedMsg) {
|
||||||
|
this.logger.debug(`WsRequestor:request - preparing queued ${type} for sending`);
|
||||||
|
setImmediate(this.request.bind(this, type, hook, params, httpHeaders));
|
||||||
|
}
|
||||||
|
this.queuedMsg.length = 0;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
//this.logger.debug({obj}, `websocket: sending (${url})`);
|
||||||
|
|
||||||
|
/* simple notifications */
|
||||||
|
if (['call:status', 'jambonz:error', 'session:reconnect'].includes(type)) {
|
||||||
|
this.ws.send(JSON.stringify(obj), () => {
|
||||||
|
this.logger.debug({obj}, `WsRequestor:request websocket: sent (${url})`);
|
||||||
|
sendQueuedMsgs();
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* messages that require an ack */
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
/* give the far end a reasonable amount of time to ack our message */
|
||||||
|
const timer = setTimeout(() => {
|
||||||
|
const {failure} = this.messagesInFlight.get(msgid);
|
||||||
|
failure && failure(`timeout from far end for msgid ${msgid}`);
|
||||||
|
this.messagesInFlight.delete(msgid);
|
||||||
|
}, RESPONSE_TIMEOUT_MS);
|
||||||
|
|
||||||
|
/* save the message info for reply */
|
||||||
|
const startAt = process.hrtime();
|
||||||
|
this.messagesInFlight.set(msgid, {
|
||||||
|
timer,
|
||||||
|
success: (response) => {
|
||||||
|
clearTimeout(timer);
|
||||||
|
const rtt = this._roundTrip(startAt);
|
||||||
|
this.logger.info({response}, `WsRequestor:request ${url} succeeded in ${rtt}ms`);
|
||||||
|
this.stats.histogram('app.hook.ws_response_time', rtt, ['hook_type:app']);
|
||||||
|
resolve(response);
|
||||||
|
},
|
||||||
|
failure: (err) => {
|
||||||
|
clearTimeout(timer);
|
||||||
|
reject(err);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
/* send the message */
|
||||||
|
this.ws.send(JSON.stringify(obj), () => {
|
||||||
|
this.logger.debug({obj}, `WsRequestor:request websocket: sent (${url})`);
|
||||||
|
sendQueuedMsgs();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
close() {
|
||||||
|
this.closedGracefully = true;
|
||||||
|
this.logger.info('WsRequestor:close closing socket');
|
||||||
|
try {
|
||||||
|
if (this.ws) {
|
||||||
|
this.ws.close();
|
||||||
|
this.ws.removeAllListeners();
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const [msgid, obj] of this.messagesInFlight) {
|
||||||
|
const {timer} = obj;
|
||||||
|
clearTimeout(timer);
|
||||||
|
obj.failure(`abandoning msgid ${msgid} since we have closed the socket`);
|
||||||
|
}
|
||||||
|
this.messagesInFlight.clear();
|
||||||
|
|
||||||
|
} catch (err) {
|
||||||
|
this.logger.info({err}, 'WsRequestor: Error closing socket');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_connect() {
|
||||||
|
assert(!this.ws);
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
const handshakeTimeout = process.env.JAMBONES_WS_HANDSHAKE_TIMEOUT_MS ?
|
||||||
|
parseInt(process.env.JAMBONES_WS_HANDSHAKE_TIMEOUT_MS) :
|
||||||
|
1500;
|
||||||
|
let opts = {
|
||||||
|
followRedirects: true,
|
||||||
|
maxRedirects: 2,
|
||||||
|
handshakeTimeout,
|
||||||
|
maxPayload: 8096,
|
||||||
|
};
|
||||||
|
if (this.username && this.password) opts = {...opts, auth: `${this.username}:${this.password}`};
|
||||||
|
|
||||||
|
this
|
||||||
|
.once('ready', (ws) => {
|
||||||
|
this.removeAllListeners('not-ready');
|
||||||
|
if (this.connections > 1) this.request('session:reconnect', this.url);
|
||||||
|
resolve();
|
||||||
|
})
|
||||||
|
.once('not-ready', (err) => {
|
||||||
|
this.removeAllListeners('ready');
|
||||||
|
reject(err);
|
||||||
|
});
|
||||||
|
const ws = new Websocket(this.url, ['ws.jambonz.org'], opts);
|
||||||
|
this._setHandlers(ws);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
_setHandlers(ws) {
|
||||||
|
this.logger.debug('WsRequestor:_setHandlers');
|
||||||
|
ws
|
||||||
|
.once('open', this._onOpen.bind(this, ws))
|
||||||
|
.once('close', this._onClose.bind(this))
|
||||||
|
.on('message', this._onMessage.bind(this))
|
||||||
|
.once('unexpected-response', this._onUnexpectedResponse.bind(this, ws))
|
||||||
|
.on('error', this._onError.bind(this));
|
||||||
|
}
|
||||||
|
|
||||||
|
_onError(err) {
|
||||||
|
if (this.connections > 0) {
|
||||||
|
this.logger.info({url: this.url, err}, 'WsRequestor:_onError');
|
||||||
|
}
|
||||||
|
else this.emit('not-ready', err);
|
||||||
|
}
|
||||||
|
|
||||||
|
_onOpen(ws) {
|
||||||
|
this.logger.info({url: this.url}, `WsRequestor(${this.id}) - successfully connected`);
|
||||||
|
if (this.ws) this.logger.info({old_ws: this.ws._socket.address()}, 'WsRequestor:_onOpen');
|
||||||
|
assert(!this.ws);
|
||||||
|
this.ws = ws;
|
||||||
|
this.connectInProgress = false;
|
||||||
|
this.connections++;
|
||||||
|
this.emit('ready', ws);
|
||||||
|
}
|
||||||
|
|
||||||
|
_onClose(code) {
|
||||||
|
this.logger.info(`WsRequestor(${this.id}) - closed from far end ${code}`);
|
||||||
|
if (this.connections > 0 && code !== 1000) {
|
||||||
|
this.logger.info({url: this.url}, 'WsRequestor - socket closed unexpectedly from remote side');
|
||||||
|
this.emit('socket-closed');
|
||||||
|
}
|
||||||
|
else if (code === 1000) this.closedGracefully = true;
|
||||||
|
this.ws?.removeAllListeners();
|
||||||
|
this.ws = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
_onUnexpectedResponse(ws, req, res) {
|
||||||
|
assert(!this.ws);
|
||||||
|
this.logger.info({
|
||||||
|
headers: res.headers,
|
||||||
|
statusCode: res.statusCode,
|
||||||
|
statusMessage: res.statusMessage
|
||||||
|
}, 'WsRequestor - unexpected response');
|
||||||
|
this.emit('connection-failure');
|
||||||
|
this.emit('not-ready', new Error(`${res.statusCode} ${res.statusMessage}`));
|
||||||
|
}
|
||||||
|
|
||||||
|
_onSocketClosed() {
|
||||||
|
this.ws = null;
|
||||||
|
this.emit('connection-dropped');
|
||||||
|
if (this.connections > 0 && this.connections < MAX_RECONNECTS && !this.closedGracefully) {
|
||||||
|
this.logger.debug(`WsRequestor:_onSocketClosed waiting ${this.backoffMs} to reconnect`);
|
||||||
|
setTimeout(() => {
|
||||||
|
this.logger.debug(
|
||||||
|
{haveWs: !!this.ws, connectInProgress: this.connectInProgress},
|
||||||
|
'WsRequestor:_onSocketClosed time to reconnect');
|
||||||
|
if (!this.ws && !this.connectInProgress) {
|
||||||
|
this.connectInProgress = true;
|
||||||
|
this._connect().catch((err) => this.connectInProgress = false);
|
||||||
|
}
|
||||||
|
}, this.backoffMs);
|
||||||
|
this.backoffMs = this.backoffMs < 2000 ? this.backoffMs * 2 : (this.backoffMs + 2000);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_onMessage(content, isBinary) {
|
||||||
|
if (this.isBinary) {
|
||||||
|
this.logger.info({url: this.url}, 'WsRequestor:_onMessage - discarding binary message');
|
||||||
|
this.maliciousClient = true;
|
||||||
|
this.ws.close();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* messages must be JSON format */
|
||||||
|
try {
|
||||||
|
const obj = JSON.parse(content);
|
||||||
|
const {type, msgid, command, call_sid = this.call_sid, queueCommand = false, data} = obj;
|
||||||
|
|
||||||
|
this.logger.debug({obj}, 'WsRequestor:request websocket: received');
|
||||||
|
assert.ok(type, 'type property not supplied');
|
||||||
|
|
||||||
|
switch (type) {
|
||||||
|
case 'ack':
|
||||||
|
assert.ok(msgid, 'msgid not supplied');
|
||||||
|
this._recvAck(msgid, data);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'command':
|
||||||
|
assert.ok(command, 'command property not supplied');
|
||||||
|
assert.ok(data, 'data property not supplied');
|
||||||
|
this._recvCommand(msgid, command, call_sid, queueCommand, data);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
assert.ok(false, `invalid type property: ${type}`);
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
this.logger.info({err, content}, 'WsRequestor:_onMessage - invalid incoming message');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_recvAck(msgid, data) {
|
||||||
|
const obj = this.messagesInFlight.get(msgid);
|
||||||
|
if (!obj) {
|
||||||
|
this.logger.info({url: this.url}, `WsRequestor:_recvAck - ack to unknown msgid ${msgid}, discarding`);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.logger.debug({url: this.url}, `WsRequestor:_recvAck - received response to ${msgid}`);
|
||||||
|
this.messagesInFlight.delete(msgid);
|
||||||
|
const {success} = obj;
|
||||||
|
success && success(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
_recvCommand(msgid, command, call_sid, queueCommand, data) {
|
||||||
|
// TODO: validate command
|
||||||
|
this.logger.info({msgid, command, call_sid, queueCommand, data}, 'received command');
|
||||||
|
this.emit('command', {msgid, command, call_sid, queueCommand, data});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = WsRequestor;
|
||||||
2960
package-lock.json
generated
2960
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
28
package.json
28
package.json
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "jambonz-feature-server",
|
"name": "jambonz-feature-server",
|
||||||
"version": "v0.7.3",
|
"version": "v0.7.5",
|
||||||
"main": "app.js",
|
"main": "app.js",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">= 10.16.0"
|
"node": ">= 10.16.0"
|
||||||
@@ -21,7 +21,7 @@
|
|||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"start": "node app",
|
"start": "node app",
|
||||||
"test": "NODE_ENV=test JAMBONES_HOSTING=1 DRACHTIO_HOST=127.0.0.1 DRACHTIO_PORT=9060 DRACHTIO_SECRET=cymru JAMBONES_MYSQL_HOST=127.0.0.1 JAMBONES_MYSQL_PORT=3360 JAMBONES_MYSQL_USER=jambones_test JAMBONES_MYSQL_PASSWORD=jambones_test JAMBONES_MYSQL_DATABASE=jambones_test JAMBONES_REDIS_HOST=127.0.0.1 JAMBONES_REDIS_PORT=16379 JAMBONES_LOGLEVEL=debug ENABLE_METRICS=0 HTTP_PORT=3000 JAMBONES_SBCS=172.38.0.10 JAMBONES_FREESWITCH=127.0.0.1:8022:ClueCon:docker-host JAMBONES_TIME_SERIES_HOST=127.0.0.1 JAMBONES_NETWORK_CIDR=172.38.0.0/16 node test/ ",
|
"test": "NODE_ENV=test JAMBONES_HOSTING=1 DRACHTIO_HOST=127.0.0.1 DRACHTIO_PORT=9060 DRACHTIO_SECRET=cymru JAMBONES_MYSQL_HOST=127.0.0.1 JAMBONES_MYSQL_PORT=3360 JAMBONES_MYSQL_USER=jambones_test JAMBONES_MYSQL_PASSWORD=jambones_test JAMBONES_MYSQL_DATABASE=jambones_test JAMBONES_REDIS_HOST=127.0.0.1 JAMBONES_REDIS_PORT=16379 JAMBONES_LOGLEVEL=info ENABLE_METRICS=0 HTTP_PORT=3000 JAMBONES_SBCS=172.38.0.10 JAMBONES_FREESWITCH=127.0.0.1:8022:ClueCon:docker-host JAMBONES_TIME_SERIES_HOST=127.0.0.1 JAMBONES_NETWORK_CIDR=172.38.0.0/16 node test/ ",
|
||||||
"coverage": "./node_modules/.bin/nyc --reporter html --report-dir ./coverage npm run test",
|
"coverage": "./node_modules/.bin/nyc --reporter html --report-dir ./coverage npm run test",
|
||||||
"jslint": "eslint app.js lib"
|
"jslint": "eslint app.js lib"
|
||||||
},
|
},
|
||||||
@@ -30,10 +30,22 @@
|
|||||||
"@jambonz/db-helpers": "^0.6.16",
|
"@jambonz/db-helpers": "^0.6.16",
|
||||||
"@jambonz/http-health-check": "^0.0.1",
|
"@jambonz/http-health-check": "^0.0.1",
|
||||||
"@jambonz/mw-registrar": "^0.2.1",
|
"@jambonz/mw-registrar": "^0.2.1",
|
||||||
"@jambonz/realtimedb-helpers": "^0.4.24",
|
"@jambonz/realtimedb-helpers": "^0.4.27",
|
||||||
"@jambonz/stats-collector": "^0.1.6",
|
"@jambonz/stats-collector": "^0.1.6",
|
||||||
"@jambonz/time-series": "^0.1.6",
|
"@jambonz/time-series": "^0.1.6",
|
||||||
"aws-sdk": "^2.1060.0",
|
"@opentelemetry/api": "^1.1.0",
|
||||||
|
"@opentelemetry/exporter-jaeger": "^1.1.0",
|
||||||
|
"@opentelemetry/exporter-trace-otlp-http": "^0.27.0",
|
||||||
|
"@opentelemetry/exporter-zipkin": "^1.1.0",
|
||||||
|
"@opentelemetry/instrumentation": "^0.27.0",
|
||||||
|
"@opentelemetry/instrumentation-express": "^0.28.0",
|
||||||
|
"@opentelemetry/instrumentation-http": "^0.27.0",
|
||||||
|
"@opentelemetry/instrumentation-pino": "^0.28.1",
|
||||||
|
"@opentelemetry/resources": "^1.1.0",
|
||||||
|
"@opentelemetry/sdk-trace-base": "^1.1.0",
|
||||||
|
"@opentelemetry/sdk-trace-node": "^1.1.0",
|
||||||
|
"@opentelemetry/semantic-conventions": "^1.1.0",
|
||||||
|
"aws-sdk": "^2.1073.0",
|
||||||
"bent": "^7.3.12",
|
"bent": "^7.3.12",
|
||||||
"cidr-matcher": "^2.1.1",
|
"cidr-matcher": "^2.1.1",
|
||||||
"debug": "^4.3.2",
|
"debug": "^4.3.2",
|
||||||
@@ -43,12 +55,14 @@
|
|||||||
"express": "^4.17.1",
|
"express": "^4.17.1",
|
||||||
"helmet": "^5.0.2",
|
"helmet": "^5.0.2",
|
||||||
"ip": "^1.1.5",
|
"ip": "^1.1.5",
|
||||||
"moment": "^2.29.1",
|
"moment": "^2.29.2",
|
||||||
"parse-url": "^5.0.7",
|
"parse-url": "^5.0.7",
|
||||||
"pino": "^6.13.4",
|
"pino": "^6.13.4",
|
||||||
|
"short-uuid": "^4.2.0",
|
||||||
"to-snake-case": "^1.0.0",
|
"to-snake-case": "^1.0.0",
|
||||||
"uuid": "^8.3.2",
|
"uuid": "^8.3.2",
|
||||||
"verify-aws-sns-signature": "^0.0.6",
|
"verify-aws-sns-signature": "^0.0.6",
|
||||||
|
"ws": "^8.5.0",
|
||||||
"xml2js": "^0.4.23"
|
"xml2js": "^0.4.23"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
@@ -58,5 +72,9 @@
|
|||||||
"eslint-plugin-promise": "^4.3.1",
|
"eslint-plugin-promise": "^4.3.1",
|
||||||
"nyc": "^15.1.0",
|
"nyc": "^15.1.0",
|
||||||
"tape": "^5.2.2"
|
"tape": "^5.2.2"
|
||||||
|
},
|
||||||
|
"optionalDependencies": {
|
||||||
|
"bufferutil": "^4.0.6",
|
||||||
|
"utf-8-validate": "^5.0.8"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
61
tracer.js
Normal file
61
tracer.js
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
const opentelemetry = require('@opentelemetry/api');
|
||||||
|
const { registerInstrumentations } = require('@opentelemetry/instrumentation');
|
||||||
|
const { NodeTracerProvider } = require('@opentelemetry/sdk-trace-node');
|
||||||
|
const { Resource } = require('@opentelemetry/resources');
|
||||||
|
const { SemanticResourceAttributes } = require('@opentelemetry/semantic-conventions');
|
||||||
|
const { BatchSpanProcessor } = require('@opentelemetry/sdk-trace-base');
|
||||||
|
const { JaegerExporter } = require('@opentelemetry/exporter-jaeger');
|
||||||
|
const { ZipkinExporter } = require('@opentelemetry/exporter-zipkin');
|
||||||
|
const { OTLPTraceExporter } = require ('@opentelemetry/exporter-trace-otlp-http');
|
||||||
|
//const { HttpInstrumentation } = require('@opentelemetry/instrumentation-http');
|
||||||
|
//const { ExpressInstrumentation } = require('@opentelemetry/instrumentation-express');
|
||||||
|
//const { PinoInstrumentation } = require('@opentelemetry/instrumentation-pino');
|
||||||
|
|
||||||
|
module.exports = (serviceName) => {
|
||||||
|
if (process.env.JAMBONES_OTEL_ENABLED) {
|
||||||
|
const {version} = require('./package.json');
|
||||||
|
const provider = new NodeTracerProvider({
|
||||||
|
resource: new Resource({
|
||||||
|
[SemanticResourceAttributes.SERVICE_NAME]: serviceName,
|
||||||
|
[SemanticResourceAttributes.SERVICE_VERSION]: version,
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
|
||||||
|
let exporter;
|
||||||
|
if (process.env.OTEL_EXPORTER_JAEGER_AGENT_HOST) {
|
||||||
|
exporter = new JaegerExporter();
|
||||||
|
}
|
||||||
|
else if (process.env.OTEL_EXPORTER_ZIPKIN_URL) {
|
||||||
|
exporter = new ZipkinExporter({url:process.env.OTEL_EXPORTER_ZIPKIN_URL});
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
exporter = new OTLPTraceExporter({
|
||||||
|
url: process.OTEL_EXPORTER_COLLECTOR_URL
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
provider.addSpanProcessor(new BatchSpanProcessor(exporter, {
|
||||||
|
// The maximum queue size. After the size is reached spans are dropped.
|
||||||
|
maxQueueSize: 100,
|
||||||
|
// The maximum batch size of every export. It must be smaller or equal to maxQueueSize.
|
||||||
|
maxExportBatchSize: 10,
|
||||||
|
// The interval between two consecutive exports
|
||||||
|
scheduledDelayMillis: 500,
|
||||||
|
// How long the export can run before it is cancelled
|
||||||
|
exportTimeoutMillis: 30000,
|
||||||
|
}));
|
||||||
|
|
||||||
|
// Initialize the OpenTelemetry APIs to use the NodeTracerProvider bindings
|
||||||
|
provider.register();
|
||||||
|
registerInstrumentations({
|
||||||
|
instrumentations: [
|
||||||
|
//new HttpInstrumentation(),
|
||||||
|
//new ExpressInstrumentation(),
|
||||||
|
//new PinoInstrumentation()
|
||||||
|
],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return opentelemetry.trace.getTracer(serviceName);
|
||||||
|
};
|
||||||
|
|
||||||
Reference in New Issue
Block a user