Перенаправление SIP через прокси (SIP.js)
Я пытаюсь создать минимальный прокси-сервер SIP, который служит одной цели: перенаправляет запросы в другой домен. Подвох - это домен, на который я перенаправляю, требует авторизации, поэтому я предполагаю, что мне нужно переписать некоторые атрибуты SIP, поскольку авторизация SIP частично основана на доменном имени пункта назначения.
Я попытался выпустить редирект 302, а также просто прокси и изменить значения каждого SIP-запроса, но ни один из них, похоже, не сработал. Я использую библиотеку node.js (sip.js) и попробовал модули перенаправления и прокси ( https://github.com/kirm/sip.js/blob/master/doc/api.markdown).
Любые идеи, как именно мне нужно изменить данные SIP для перенаправления запросов в другой домен и включить аутентификацию на этом другом домене?
1 ответ
Ниже приведен базовый скрипт узла, с которым я работал на своем собственном SIP-сервере. Вам нужно будет заменить учетные данные и IP-адрес для собственного тестирования.
Прокси-скрипт не отправляет ответ перенаправления клиенту, а вместо этого инициирует новую транзакцию на сервер от имени клиента. Сервер SIP, работающий в этом режиме, более правильно называется Back-to-Back User Agent (B2BUA). Я не добавил все необходимые функции, такие как сопоставление и передача ответов исходному клиенту; в этом есть немало работы.
var sip = require('sip');
var digest = require('sip/digest');
var util = require('util');
var os = require('os');
var proxy = require('sip/proxy');
var registry = {
'user': { user: "user", password: "password", realm: "sipserver.com"},
};
function rstring() { return Math.floor(Math.random()*1e6).toString(); }
sip.start({
address: "192.168.33.116", // If the IP is not specified here the proxy uses a hostname in the Via header which will causes an issue if it's not fully qualified.
logger: {
send: function(message, address) { debugger; util.debug("send\n" + util.inspect(message, false, null)); },
recv: function(message, address) { debugger; util.debug("recv\n" + util.inspect(message, false, null)); }
}
},
function(rq) {
try {
if(rq.method === 'INVITE') {
proxy.send(sip.makeResponse(rq, 100, 'Trying'));
//looking up user info
var username = sip.parseUri(rq.headers.to.uri).user;
var creds = registry[username];
if(!creds) {
proxy.send(sip.makeResponse(rq, 404, 'User not found'));
}
else {
proxy.send(rq, function(rs) {
if(rs.status === 401) {
// Update the original request so that it's not treated as a duplicate.
rq.headers['cseq'].seq++;
rq.headers.via.shift ();
rq.headers['call-id'] = rstring();
digest.signRequest(creds, rq, rs, creds);
proxy.send(rq);
}
});
}
}
else {
proxy.send(sip.makeResponse(rq, 405, 'Method Not Allowed'));
}
} catch(e) {
util.debug(e);
util.debug(e.stack);
proxy.send(sip.makeResponse(rq, 500, "Server Internal Error"));
}
});