Files
sbc-sip-sidecar/lib/sip-trunk-options-ping.js
Sam Machin a22ab38aab Disable Options and Registration on certain failures (#90)
* disable options ping on defined errors

* correct sid

* disable reg on error

* fix path

* fixes from testing

* fixes from testing

* lint

* update dbhelpers dep

* remove whitespace in CONFIG strings

* lint

* this seems like a better way of convering and matching status codes
2025-04-02 08:22:52 -04:00

120 lines
5.4 KiB
JavaScript

const { addSipGatewayToBlacklist, removeSipGatewayFromBlacklist, isSipGatewayBlacklisted } = require('./utils');
const {OPTIONS_RESPONSE_REMOVE} = require('./config');
const send_options_gateways = [];
const send_options_bots = [];
class OptionsBot {
constructor(logger, gateway) {
this.logger = logger;
this.sip_gateway_sid = gateway.sip_gateway_sid;
this.voip_carrier_sid = gateway.voip_carrier_sid;
this.ipv4 = gateway.ipv4;
this.port = gateway.port;
this.protocol = gateway.protocol;
this.expiry = (process.env.SEND_OPTIONS_PING_INTERVAL || 60);
this.blacklist_expiry = (process.env.OPTIONS_PING_TTL || 300);
const useSipsScheme = gateway.protocol.includes('tls') && gateway.use_sips_scheme;
const isIPv4 = /[0-9]{1,3}.[0-9]{1,3}.[0-9]{1,3}.[0-9]{1,3}/.test(gateway.ipv4);
const transport = gateway.protocol.includes('/') ? gateway.protocol.substring(0, gateway.protocol.indexOf('/')) :
gateway.protocol;
this.proxy = `sip:${this.ipv4}${isIPv4 ? `:${this.port}` : ''};transport=${transport}`;
this.uri = `sip${useSipsScheme ?
's' : ''}:${gateway.ipv4}${gateway.port && !useSipsScheme ? `:${gateway.port}` : ''}`;
}
async options(srf) {
const { lookupCarrierBySid } = srf.locals.dbHelpers;
const { writeAlerts, logger, realtimeDbHelpers } = srf.locals;
try {
const req = await srf.request({
uri: this.uri,
method: 'OPTIONS',
proxy: this.proxy
});
req.on('response', async(res) => {
if (res.status !== 200) {
this.logger.info(`Received Options response ${res.status} for ${this.uri}`);
// Check if the SIP gateway is not already blacklisted
if (!await isSipGatewayBlacklisted(realtimeDbHelpers.client, logger, this.sip_gateway_sid)) {
// Add the gateway to the blacklist since it failed the OPTIONS ping
await addSipGatewayToBlacklist(realtimeDbHelpers.client, logger, this.sip_gateway_sid,
this.blacklist_expiry);
const carrier = await lookupCarrierBySid(this.voip_carrier_sid);
if (carrier) {
writeAlerts({
account_sid: carrier.account_sid,
service_provider_sid: carrier.service_provider_sid,
// eslint-disable-next-line max-len
message: `Options ping ${this.ipv4}${this.port ? `:${this.port}` : ''};transport=${this.protocol} unsuccessfully, received: ${res.status}`
});
}
}
if (OPTIONS_RESPONSE_REMOVE.includes(res.status)) {
const { updateSipGatewayBySid } = srf.locals.dbHelpers;
await updateSipGatewayBySid(this.sip_gateway_sid, {send_options_ping: false});
const carrier = await lookupCarrierBySid(this.voip_carrier_sid);
if (carrier) {
// eslint-disable-next-line max-len
this.logger.info(`Disabling Options ping for ${this.ipv4} in carrier ${carrier.name} (sid:${carrier.voip_carrier_sid})`);
writeAlerts({
account_sid: carrier.account_sid,
service_provider_sid: carrier.service_provider_sid,
// eslint-disable-next-line max-len
message: `Disabling Options ping for ${this.ipv4} in carrier ${carrier.name} (sid:${carrier.voip_carrier_sid})`
});
}
}
} else {
// If the gateway is blacklisted, remove it from the blacklist as we have successfully pinged it
if (await isSipGatewayBlacklisted(realtimeDbHelpers.client, logger, this.sip_gateway_sid)) {
await removeSipGatewayFromBlacklist(realtimeDbHelpers.client, logger, this.sip_gateway_sid);
}
}
});
} catch (err) {
this.logger.error({ err }, `Error Options ping to ${this.uri}`);
if (!await isSipGatewayBlacklisted(realtimeDbHelpers.client, logger, this.sip_gateway_sid)) {
await addSipGatewayToBlacklist(realtimeDbHelpers.client, logger, this.sip_gateway_sid, this.blacklist_expiry);
const carrier = await lookupCarrierBySid(this.voip_carrier_sid);
if (carrier) {
writeAlerts({
account_sid: carrier.account_sid,
service_provider_sid: carrier.service_provider_sid,
// eslint-disable-next-line max-len
message: `Options ping ${this.ipv4}${this.port ? `:${this.port}` : ''};transport=${this.protocol} unsuccessfully, error: ${err}`
});
}
}
}
}
}
module.exports = async(logger, srf) => {
const updateSipGatewayOptsBot = async(logger, srf) => {
try {
const { lookupSipGatewaysByFilters } = srf.locals.dbHelpers;
const gws = await lookupSipGatewaysByFilters({send_options_ping: true, outbound: true, is_active: true});
if (gws.length > 0) {
logger.debug(`updateSipGatewayOptsBot: sending OPTIONS ping to ${gws.length} gateways`);
send_options_gateways.length = 0;
send_options_gateways.push(...gws);
for (const g of send_options_gateways) {
const optsBot = new OptionsBot(logger, g);
send_options_bots.push(optsBot);
optsBot.options(srf);
}
logger.debug(`updateSipGatewayOptsBot: we have started ${send_options_bots.length} optionsBots`);
}
} catch (err) {
logger.error({ err }, 'updateSipGatewayOptsBot Error');
}
};
setInterval(updateSipGatewayOptsBot.bind(null, logger, srf), (process.env.SEND_OPTIONS_PING_INTERVAL || 60) * 1000);
};