mirror of https://github.com/sipwise/sems.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
759 lines
26 KiB
759 lines
26 KiB
/*
|
|
* Copyright (C) 2013 Stefan Sayer
|
|
*
|
|
* This file is part of SEMS, a free SIP media server.
|
|
*
|
|
* SEMS is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version. This program is released under
|
|
* the GPL with the additional exemption that compiling, linking,
|
|
* and/or using OpenSSL is allowed.
|
|
*
|
|
* For a license to use the SEMS software under conditions
|
|
* other than those described here, or to purchase support for this
|
|
* software, please contact iptel.org by e-mail at the following addresses:
|
|
* info@iptel.org
|
|
*
|
|
* SEMS is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include "SBCDSMInstance.h"
|
|
#include "SBCCallLeg.h"
|
|
#include "SBCSimpleRelay.h"
|
|
|
|
#include "DSM.h"
|
|
#include "SBCDSMParams.h"
|
|
|
|
#include "AmAdvancedAudio.h"
|
|
#include "AmRingTone.h"
|
|
|
|
#include <algorithm>
|
|
|
|
using namespace std;
|
|
|
|
SBCDSMInstance::SBCDSMInstance(SBCCallLeg *call, const VarMapT& values)
|
|
: call(call), local_media_connected(false)
|
|
{
|
|
DBG("SBCDSMInstance::SBCDSMInstance()\n");
|
|
VarMapT::const_iterator it = values.find(DSM_SBC_CCVAR_START_DIAG);
|
|
if (it != values.end())
|
|
startDiagName = it->second;
|
|
|
|
it = values.find(DSM_SBC_CCVAR_APP_BUNDLE);
|
|
if (it != values.end())
|
|
appBundle = it->second;
|
|
|
|
if (startDiagName.empty()) {
|
|
throw string("DSM SBC call control " DSM_SBC_CCVAR_START_DIAG " parameter not set (see call profile)'");
|
|
}
|
|
|
|
map<string,string> config_vars;
|
|
bool SetParamVariables; // unused
|
|
|
|
if (!DSMFactory::instance()->
|
|
addScriptDiagsToEngine(appBundle,
|
|
&engine,
|
|
config_vars,
|
|
SetParamVariables)) {
|
|
ERROR("initializing call with DSM app bundle '%s'\n", appBundle.c_str());
|
|
throw string("initializing call with DSM app bundle '" +appBundle);
|
|
}
|
|
|
|
for (map<string, string>::const_iterator it =
|
|
config_vars.begin(); it != config_vars.end(); it++)
|
|
var["config."+it->first] = it->second;
|
|
|
|
// overwrite config. variables with cc-instance variables (map::insert doesn't overwrite)
|
|
for (VarMapT::const_iterator it = values.begin(); it != values.end(); it++) {
|
|
var[it->first] = it->second;
|
|
}
|
|
|
|
DBG("Running init of SBCDSMInstance...\n");
|
|
if (!engine.init(call, this, startDiagName, DSMCondition::Start)) {
|
|
WARN("Initialization failed for SBCDSMInstance\n");
|
|
// TODO: mark this as not running!
|
|
return;
|
|
}
|
|
}
|
|
|
|
SBCDSMInstance::~SBCDSMInstance()
|
|
{
|
|
DBG("SBCDSMInstance::~SBCDSMInstance()\n");
|
|
for (std::set<DSMDisposable*>::iterator it=
|
|
gc_trash.begin(); it != gc_trash.end(); it++)
|
|
delete *it;
|
|
|
|
for (vector<AmAudio*>::iterator it=
|
|
audiofiles.begin();it!=audiofiles.end();it++)
|
|
delete *it;
|
|
|
|
AmMediaProcessor::instance()->removeSession(call);
|
|
}
|
|
|
|
#define RETURN_CONTINUE_OR_STOP_PROCESSING \
|
|
if (event_params[DSM_SBC_PARAM_STOP_PROCESSING]==DSM_TRUE) \
|
|
return StopProcessing; \
|
|
return ContinueProcessing;
|
|
|
|
/** @return whether to continue processing */
|
|
/** called from A/B leg when in-dialog request comes in */
|
|
CCChainProcessing SBCDSMInstance::onInitialInvite(SBCCallLeg *call, InitialInviteHandlerParams ¶ms)
|
|
{
|
|
DBG("SBCDSMInstance::onInitialInvite()\n");
|
|
|
|
VarMapT event_params;
|
|
event_params["remote_party"] = params.remote_party;
|
|
event_params["remote_uri"] = params.remote_party;
|
|
event_params["from"] = params.remote_party;
|
|
|
|
avar[DSM_AVAR_REQUEST] = AmArg(params.original_invite);
|
|
avar[DSM_SBC_AVAR_MODIFIED_INVITE] = AmArg(params.modified_invite);
|
|
|
|
engine.runEvent(call, this, DSMCondition::Invite, &event_params);
|
|
|
|
avar.erase(DSM_SBC_AVAR_MODIFIED_INVITE);
|
|
|
|
RETURN_CONTINUE_OR_STOP_PROCESSING;
|
|
}
|
|
|
|
void extractRequestParameters(VarMapT& event_params, AVarMapT& avar, DSMSipRequest* request) {
|
|
if (NULL == request)
|
|
return;
|
|
|
|
if (NULL != request) {
|
|
event_params["method"] = request->req->method;
|
|
event_params["r_uri"] = request->req->r_uri;
|
|
event_params["from"] = request->req->from;
|
|
event_params["to"] = request->req->to;
|
|
event_params["hdrs"] = request->req->hdrs;
|
|
event_params["from_tag"] = request->req->from_tag;
|
|
event_params["to_tag"] = request->req->to_tag;
|
|
event_params["callid"] = request->req->callid;
|
|
|
|
vector<string> hdrs = explode(request->req->hdrs, CRLF);
|
|
for (vector<string>::iterator it=hdrs.begin(); it!=hdrs.end();it++) {
|
|
size_t p = it->find(":");
|
|
if (p==string::npos)
|
|
continue;
|
|
size_t p1=p;
|
|
if (++p>=it->size())
|
|
continue;
|
|
while (p<it->size() && ((*it)[p] == ' ' || (*it)[p] == '\t'))
|
|
p++;
|
|
event_params["hdr."+it->substr(0,p1)]=it->substr(p);
|
|
}
|
|
|
|
// avar[DSM_AVAR_REQUEST] = AmArg(const_cast<AmSipRequest*>(request));
|
|
avar[DSM_AVAR_REQUEST] = AmArg(request);
|
|
}
|
|
}
|
|
|
|
void clearRequestParameters(AVarMapT& avar) {
|
|
avar.erase(DSM_AVAR_REQUEST);
|
|
}
|
|
|
|
void extractReplyParameters(VarMapT& event_params, AVarMapT& avar, DSMSipReply* reply) {
|
|
if (NULL == reply)
|
|
return;
|
|
|
|
event_params["sip_reason"] = reply->reply->reason;
|
|
event_params["sip_code"] = int2str(reply->reply->code);
|
|
event_params["from"] = reply->reply->from;
|
|
event_params["from_tag"] = reply->reply->from_tag;
|
|
event_params["to"] = reply->reply->to;
|
|
event_params["to_tag"] = reply->reply->to_tag;
|
|
event_params["callid"] = reply->reply->callid;
|
|
event_params["hdrs"] = reply->reply->hdrs;
|
|
#ifdef PROPAGATE_UNPARSED_REPLY_HEADERS
|
|
for (list<AmSipHeader>::const_iterator it = reply->reply->unparsed_headers.begin();
|
|
it != reply->reply->unparsed_headers.end(); it++) {
|
|
event_params["hdr."+it->name] = it->value;
|
|
}
|
|
#else
|
|
vector<string> hdrs = explode(reply->reply->hdrs, CRLF);
|
|
for (vector<string>::iterator it=hdrs.begin(); it!=hdrs.end();it++) {
|
|
size_t p = it->find(":");
|
|
if (p==string::npos)
|
|
continue;
|
|
size_t p1=p;
|
|
if (++p>=it->size())
|
|
continue;
|
|
while (p<it->size() && ((*it)[p] == ' ' || (*it)[p] == '\t'))
|
|
p++;
|
|
event_params["hdr."+it->substr(0,p1)]=it->substr(p);
|
|
}
|
|
#endif
|
|
avar[DSM_AVAR_REPLY] = AmArg(reply);
|
|
}
|
|
|
|
void clearReplyParameters(AVarMapT& avar) {
|
|
avar.erase(DSM_AVAR_REPLY);
|
|
}
|
|
|
|
void SBCDSMInstance::onStateChange(SBCCallLeg *call, const CallLeg::StatusChangeCause &cause) {
|
|
DBG("SBCDSMInstance::onStateChange()\n");
|
|
VarMapT event_params;
|
|
|
|
event_params["SBCCallStatus"] = call->getCallStatusStr();
|
|
unique_ptr<DSMSipRequest> dsm_request;
|
|
unique_ptr<DSMSipReply> dsm_reply;
|
|
|
|
switch (cause.reason) {
|
|
case CallLeg::StatusChangeCause::SipReply:
|
|
event_params["reason"] = "SipReply";
|
|
dsm_reply.reset(new DSMSipReply(cause.param.reply));
|
|
extractReplyParameters(event_params, avar, dsm_reply.get());
|
|
break;
|
|
case CallLeg::StatusChangeCause::SipRequest:
|
|
event_params["reason"] = "SipRequest";
|
|
dsm_request.reset(new DSMSipRequest(cause.param.request));
|
|
extractRequestParameters(event_params, avar, dsm_request.get());
|
|
break;
|
|
case CallLeg::StatusChangeCause::Other:
|
|
event_params["reason"] = "other";
|
|
if (NULL != cause.param.desc)
|
|
event_params["desc"] = string(cause.param.desc);
|
|
break;
|
|
case CallLeg::StatusChangeCause::Canceled: event_params["reason"] = "Canceled"; break;
|
|
case CallLeg::StatusChangeCause::NoAck: event_params["reason"] = "NoAck"; break;
|
|
case CallLeg::StatusChangeCause::NoPrack: event_params["reason"] = "NoPrack"; break;
|
|
case CallLeg::StatusChangeCause::RtpTimeout: event_params["reason"] = "RtpTimeout"; break;
|
|
case CallLeg::StatusChangeCause::SessionTimeout: event_params["reason"] = "SessionTimeout"; break;
|
|
case CallLeg::StatusChangeCause::InternalError: event_params["reason"] = "InternalError"; break;
|
|
defaut: break;
|
|
};
|
|
|
|
engine.runEvent(call, this, DSMCondition::LegStateChange, &event_params);
|
|
|
|
switch (cause.reason) {
|
|
case CallLeg::StatusChangeCause::SipReply: clearReplyParameters(avar); break;
|
|
case CallLeg::StatusChangeCause::SipRequest: clearRequestParameters(avar); break;
|
|
default: break;
|
|
}
|
|
|
|
}
|
|
|
|
/** called from A/B leg when in-dialog request comes in */
|
|
CCChainProcessing SBCDSMInstance::onInDialogRequest(SBCCallLeg* call, const AmSipRequest& req) {
|
|
DBG("SBCDSMInstance::onInDialogRequest()\n");
|
|
VarMapT event_params;
|
|
DSMSipRequest dsm_request(&req);
|
|
|
|
extractRequestParameters(event_params, avar, &dsm_request);
|
|
|
|
engine.runEvent(call, this, DSMCondition::SipRequest, &event_params);
|
|
|
|
clearRequestParameters(avar);
|
|
RETURN_CONTINUE_OR_STOP_PROCESSING;
|
|
}
|
|
|
|
CCChainProcessing SBCDSMInstance::onInDialogReply(SBCCallLeg* call, const AmSipReply& reply) {
|
|
DBG("SBCDSMInstance::onInDialogReply()\n");
|
|
VarMapT event_params;
|
|
DSMSipReply dsm_reply(&reply);
|
|
extractReplyParameters(event_params, avar, &dsm_reply);
|
|
|
|
engine.runEvent(call, this, DSMCondition::SipReply, &event_params);
|
|
|
|
clearReplyParameters(avar);
|
|
RETURN_CONTINUE_OR_STOP_PROCESSING;
|
|
}
|
|
|
|
CCChainProcessing SBCDSMInstance::onEvent(SBCCallLeg* call, AmEvent* event) {
|
|
DBG("SBCDSMInstance::onEvent()\n");
|
|
|
|
if (event->event_id == DSM_EVENT_ID) {
|
|
DSMEvent* dsm_event = dynamic_cast<DSMEvent*>(event);
|
|
if (dsm_event) {
|
|
DBG("SBCDSMInstance processing DSM event\n");
|
|
|
|
engine.runEvent(call, this, DSMCondition::DSMEvent, &dsm_event->params);
|
|
|
|
if (dsm_event->params[DSM_SBC_PARAM_STOP_PROCESSING]==DSM_TRUE)
|
|
return StopProcessing;
|
|
return ContinueProcessing;
|
|
}
|
|
}
|
|
|
|
B2BEvent* b2b_ev = dynamic_cast<B2BEvent*>(event);
|
|
if(b2b_ev && b2b_ev->ev_type == B2BEvent::B2BApplication) {
|
|
engine.runEvent(call, this, DSMCondition::B2BEvent, &b2b_ev->params);
|
|
|
|
if (b2b_ev->params[DSM_SBC_PARAM_PROCESSED] == DSM_TRUE) {
|
|
ReliableB2BEvent* rel_b2b_ev = dynamic_cast<ReliableB2BEvent*>(b2b_ev);
|
|
if (NULL != rel_b2b_ev) {
|
|
rel_b2b_ev->markAsProcessed();
|
|
} else {
|
|
DBG("possible script writer error: marked #processed on non-reliable B2BEvent");
|
|
}
|
|
}
|
|
|
|
if (b2b_ev->params[DSM_SBC_PARAM_STOP_PROCESSING]==DSM_TRUE)
|
|
return StopProcessing;
|
|
return ContinueProcessing;
|
|
}
|
|
|
|
if(b2b_ev && b2b_ev->ev_type == B2BEvent::B2BCore) {
|
|
B2BSipRequestEvent* b2b_req_ev = dynamic_cast<B2BSipRequestEvent*>(b2b_ev);
|
|
if (b2b_req_ev) {
|
|
VarMapT event_params;
|
|
DSMMutableSipRequest sip_req(&b2b_req_ev->req);
|
|
extractRequestParameters(event_params, avar, &sip_req);
|
|
event_params["forward"] = b2b_req_ev->forward?"true":"false";
|
|
engine.runEvent(call, this, DSMCondition::B2BOtherRequest, &event_params);
|
|
avar.erase(DSM_AVAR_REQUEST);
|
|
if (event_params[DSM_SBC_PARAM_STOP_PROCESSING]==DSM_TRUE)
|
|
return StopProcessing;
|
|
} else {
|
|
B2BSipReplyEvent* b2b_reply_ev = dynamic_cast<B2BSipReplyEvent*>(b2b_ev);
|
|
if (b2b_reply_ev) {
|
|
VarMapT event_params;
|
|
DSMMutableSipReply dsm_reply(&b2b_reply_ev->reply);
|
|
extractReplyParameters(event_params, avar, &dsm_reply);
|
|
event_params["forward"] = b2b_reply_ev->forward?"true":"false";
|
|
event_params["trans_method"] = b2b_reply_ev->trans_method;
|
|
engine.runEvent(call, this, DSMCondition::B2BOtherReply, &event_params);
|
|
if (event_params[DSM_SBC_PARAM_STOP_PROCESSING]==DSM_TRUE)
|
|
return StopProcessing;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
AmPluginEvent* plugin_event = dynamic_cast<AmPluginEvent*>(event);
|
|
if(plugin_event && plugin_event->name == "timer_timeout") {
|
|
int timer_id = plugin_event->data.get(0).asInt();
|
|
map<string, string> params;
|
|
params["id"] = int2str(timer_id);
|
|
engine.runEvent(call, this, DSMCondition::Timer, ¶ms);
|
|
|
|
if (params[DSM_SBC_PARAM_STOP_PROCESSING]==DSM_TRUE)
|
|
return StopProcessing;
|
|
return ContinueProcessing;
|
|
}
|
|
|
|
AmPlaylistSeparatorEvent* sep_ev = dynamic_cast<AmPlaylistSeparatorEvent*>(event);
|
|
if (sep_ev) {
|
|
map<string, string> params;
|
|
params["id"] = int2str(sep_ev->event_id);
|
|
engine.runEvent(call, this, DSMCondition::PlaylistSeparator, ¶ms);
|
|
|
|
if (params[DSM_SBC_PARAM_STOP_PROCESSING]==DSM_TRUE)
|
|
return StopProcessing;
|
|
return ContinueProcessing;
|
|
}
|
|
|
|
AmAudioEvent* audio_event = dynamic_cast<AmAudioEvent*>(event);
|
|
if(audio_event &&
|
|
((audio_event->event_id == AmAudioEvent::cleared) ||
|
|
(audio_event->event_id == AmAudioEvent::noAudio))){
|
|
map<string, string> params;
|
|
params["type"] = audio_event->event_id == AmAudioEvent::cleared?"cleared":"noAudio";
|
|
engine.runEvent(call, this, DSMCondition::NoAudio, ¶ms);
|
|
|
|
if (params[DSM_SBC_PARAM_STOP_PROCESSING]==DSM_TRUE)
|
|
return StopProcessing;
|
|
return ContinueProcessing;
|
|
}
|
|
|
|
// todo: process JsonRPCEvents (? see DSMCall::process)
|
|
|
|
return ContinueProcessing;
|
|
}
|
|
|
|
CCChainProcessing SBCDSMInstance::onDtmf(SBCCallLeg *call, int event, int duration) {
|
|
DBG("* Got DTMF key %d duration %d\n",
|
|
event, duration);
|
|
|
|
map<string, string> params;
|
|
params["key"] = int2str(event);
|
|
params["duration"] = int2str(duration);
|
|
|
|
engine.runEvent(call, this, DSMCondition::Key, ¶ms);
|
|
|
|
if (params[DSM_SBC_PARAM_STOP_PROCESSING]==DSM_TRUE)
|
|
return StopProcessing;
|
|
return ContinueProcessing;
|
|
}
|
|
|
|
/** @return whether to continue processing */
|
|
CCChainProcessing SBCDSMInstance::onBLegRefused(SBCCallLeg* call, const AmSipReply& reply)
|
|
{
|
|
DBG("SBCDSMInstance::onBLegRefused()\n");
|
|
VarMapT event_params;
|
|
DSMSipReply dsm_reply(&reply);
|
|
extractReplyParameters(event_params, avar, &dsm_reply);
|
|
|
|
engine.runEvent(call, this, DSMCondition::BLegRefused, &event_params);
|
|
|
|
clearRequestParameters(avar);
|
|
RETURN_CONTINUE_OR_STOP_PROCESSING;
|
|
}
|
|
|
|
// --- hold related ------------------------
|
|
|
|
CCChainProcessing SBCDSMInstance::putOnHold(SBCCallLeg* call) {
|
|
DBG("SBCDSMInstance::putOnHold()\n");
|
|
VarMapT event_params;
|
|
engine.runEvent(call, this, DSMCondition::PutOnHold, &event_params);
|
|
RETURN_CONTINUE_OR_STOP_PROCESSING;
|
|
}
|
|
|
|
CCChainProcessing SBCDSMInstance::resumeHeld(SBCCallLeg* call, bool send_reinvite) {
|
|
DBG("SBCDSMInstance::resumeHeld()\n");
|
|
VarMapT event_params;
|
|
event_params["send_reinvite"] = send_reinvite?"true":"false";
|
|
engine.runEvent(call, this, DSMCondition::ResumeHeld, &event_params);
|
|
RETURN_CONTINUE_OR_STOP_PROCESSING;
|
|
}
|
|
|
|
CCChainProcessing SBCDSMInstance::createHoldRequest(SBCCallLeg* call, AmSdp& sdp) {
|
|
DBG("SBCDSMInstance::createHoldRequest()\n");
|
|
VarMapT event_params;
|
|
// TODO: encapsulate SDP so actions can manipulate Hold request (?)
|
|
engine.runEvent(call, this, DSMCondition::CreateHoldRequest, &event_params);
|
|
RETURN_CONTINUE_OR_STOP_PROCESSING;
|
|
}
|
|
|
|
CCChainProcessing SBCDSMInstance::handleHoldReply(SBCCallLeg* call, bool succeeded) {
|
|
DBG("SBCDSMInstance::handleHoldReply()\n");
|
|
VarMapT event_params;
|
|
event_params["succeeded"] = succeeded?"true":"false";
|
|
engine.runEvent(call, this, DSMCondition::HandleHoldReply, &event_params);
|
|
RETURN_CONTINUE_OR_STOP_PROCESSING;
|
|
}
|
|
|
|
// pretty much nonsense, but necessary because DSM is passing around AmSession
|
|
// everywhere; so we need this for non-call relays
|
|
void SBCDSMInstance::resetDummySession(SimpleRelayDialog *relay) {
|
|
if (NULL == dummy_session.get()) {
|
|
dummy_session.reset(new AmSession());
|
|
// copy the most important things
|
|
// TODO: initialize stuff from relay dialog in dummy session to be visible in DSM
|
|
dummy_session->dlg->setCallid(relay->getCallid());
|
|
dummy_session->dlg->setLocalTag(relay->getLocalTag());
|
|
dummy_session->dlg->setRemoteTag(relay->getRemoteTag());
|
|
dummy_session->dlg->setLocalUri(relay->getLocalUri());
|
|
dummy_session->dlg->setRemoteUri(relay->getRemoteUri());
|
|
}
|
|
}
|
|
|
|
AmPlaylist* SBCDSMInstance::getPlaylist() {
|
|
if (NULL == playlist.get())
|
|
playlist.reset(new AmPlaylist(call));
|
|
|
|
return playlist.get();
|
|
}
|
|
|
|
// ------------ simple relay interface --------------------------------------- */
|
|
bool SBCDSMInstance::init(SBCCallProfile &profile, SimpleRelayDialog *relay) {
|
|
DBG("SBCDSMInstance::init() - simple relay\n");
|
|
resetDummySession(relay);
|
|
|
|
VarMapT event_params;
|
|
event_params["relay_event"] = "init";
|
|
avar[DSM_SBC_AVAR_PROFILE] = AmArg(&profile);
|
|
engine.runEvent(dummy_session.get(), this, DSMCondition::RelayInit, &event_params);
|
|
avar.erase(DSM_SBC_AVAR_PROFILE);
|
|
|
|
return true;
|
|
}
|
|
|
|
void SBCDSMInstance::initUAC(SBCCallProfile &profile, SimpleRelayDialog *relay, const AmSipRequest &req) {
|
|
DBG("SBCDSMInstance::initUAC() - simple relay\n");
|
|
resetDummySession(relay);
|
|
|
|
VarMapT event_params;
|
|
event_params["relay_event"] = "initUAC";
|
|
avar[DSM_SBC_AVAR_PROFILE] = AmArg(&profile);
|
|
DSMSipRequest sip_req(&req);
|
|
extractRequestParameters(event_params, avar, &sip_req);
|
|
engine.runEvent(dummy_session.get(), this, DSMCondition::RelayInitUAC, &event_params);
|
|
clearRequestParameters(avar);
|
|
avar.erase(DSM_SBC_AVAR_PROFILE);
|
|
}
|
|
|
|
void SBCDSMInstance::initUAS(SBCCallProfile &profile, SimpleRelayDialog *relay, const AmSipRequest &req) {
|
|
DBG("SBCDSMInstance::initUAS() - simple relay\n");
|
|
resetDummySession(relay);
|
|
VarMapT event_params;
|
|
event_params["relay_event"] = "initUAS";
|
|
avar[DSM_SBC_AVAR_PROFILE] = AmArg(&profile);
|
|
DSMSipRequest sip_req(&req);
|
|
extractRequestParameters(event_params, avar, &sip_req);
|
|
engine.runEvent(dummy_session.get(), this, DSMCondition::RelayInitUAS, &event_params);
|
|
clearRequestParameters(avar);
|
|
avar.erase(DSM_SBC_AVAR_PROFILE);
|
|
}
|
|
|
|
void SBCDSMInstance::finalize(SBCCallProfile &profile, SimpleRelayDialog *relay) {
|
|
DBG("SBCDSMInstance::finalize() - relay\n");
|
|
resetDummySession(relay);
|
|
VarMapT event_params;
|
|
event_params["relay_event"] = "finalize";
|
|
avar[DSM_SBC_AVAR_PROFILE] = AmArg(&profile);
|
|
engine.runEvent(dummy_session.get(), this, DSMCondition::RelayFinalize, &event_params);
|
|
avar.erase(DSM_SBC_AVAR_PROFILE);
|
|
}
|
|
|
|
void SBCDSMInstance::onSipRequest(SBCCallProfile &profile, SimpleRelayDialog *relay, const AmSipRequest& req) {
|
|
DBG("SBCDSMInstance::onSipRequest() - simple relay\n");
|
|
resetDummySession(relay);
|
|
VarMapT event_params;
|
|
event_params["relay_event"] = "onSipRequest";
|
|
avar[DSM_SBC_AVAR_PROFILE] = AmArg(&profile);
|
|
DSMSipRequest sip_req(&req);
|
|
extractRequestParameters(event_params, avar, &sip_req);
|
|
engine.runEvent(dummy_session.get(), this, DSMCondition::RelayOnSipRequest, &event_params);
|
|
clearRequestParameters(avar);
|
|
avar.erase(DSM_SBC_AVAR_PROFILE);
|
|
}
|
|
|
|
void SBCDSMInstance::onSipReply(SBCCallProfile &profile, SimpleRelayDialog *relay, const AmSipRequest& req,
|
|
const AmSipReply& reply,
|
|
AmBasicSipDialog::Status old_dlg_status) {
|
|
DBG("SBCDSMInstance::onSipReply() - simple relay\n");
|
|
resetDummySession(relay);
|
|
VarMapT event_params;
|
|
event_params["relay_event"] = "onSipReply";
|
|
avar[DSM_SBC_AVAR_PROFILE] = AmArg(&profile);
|
|
DSMSipRequest sip_req(&req);
|
|
extractRequestParameters(event_params, avar, &sip_req);
|
|
DSMSipReply dsm_reply(&reply);
|
|
extractReplyParameters(event_params, avar, &dsm_reply); // TODO: shadows request
|
|
event_params["old_dlg_status"] = AmBasicSipDialog::getStatusStr(old_dlg_status);
|
|
engine.runEvent(dummy_session.get(), this, DSMCondition::RelayOnSipReply, &event_params);
|
|
clearReplyParameters(avar);
|
|
clearRequestParameters(avar);
|
|
avar.erase(DSM_SBC_AVAR_PROFILE);
|
|
}
|
|
|
|
void SBCDSMInstance::onB2BRequest(SBCCallProfile &profile, SimpleRelayDialog *relay, const AmSipRequest& req) {
|
|
DBG("SBCDSMInstance::onB2BRequest() - relay\n");
|
|
resetDummySession(relay);
|
|
VarMapT event_params;
|
|
event_params["relay_event"] = "onB2BRequest";
|
|
avar[DSM_SBC_AVAR_PROFILE] = AmArg(&profile);
|
|
DSMSipRequest sip_req(&req);
|
|
extractRequestParameters(event_params, avar, &sip_req);
|
|
engine.runEvent(dummy_session.get(), this, DSMCondition::RelayOnB2BRequest, &event_params);
|
|
clearRequestParameters(avar);
|
|
avar.erase(DSM_SBC_AVAR_PROFILE);
|
|
}
|
|
|
|
void SBCDSMInstance::onB2BReply(SBCCallProfile &profile, SimpleRelayDialog *relay, const AmSipReply& reply) {
|
|
DBG("SBCDSMInstance::onB2BReply() - relay\n");
|
|
resetDummySession(relay);
|
|
VarMapT event_params;
|
|
event_params["relay_event"] = "onB2BReply";
|
|
avar[DSM_SBC_AVAR_PROFILE] = AmArg(&profile);
|
|
DSMSipReply dsm_reply(&reply);
|
|
extractReplyParameters(event_params, avar, &dsm_reply);
|
|
engine.runEvent(dummy_session.get(), this, DSMCondition::RelayOnB2BReply, &event_params);
|
|
clearReplyParameters(avar);
|
|
avar.erase(DSM_SBC_AVAR_PROFILE);
|
|
}
|
|
|
|
// --- garbage collector related ------------------------
|
|
|
|
void SBCDSMInstance::transferOwnership(DSMDisposable* d) {
|
|
gc_trash.insert(d);
|
|
}
|
|
|
|
void SBCDSMInstance::releaseOwnership(DSMDisposable* d) {
|
|
gc_trash.erase(d);
|
|
}
|
|
|
|
// --- DSM session API -------------------------------------------
|
|
|
|
#define NOT_IMPLEMENTED_UINT(_func) \
|
|
unsigned int SBCDSMInstance::_func { \
|
|
throw DSMException("core", "cause", "not implemented in DSM SBC"); \
|
|
}
|
|
|
|
#define NOT_IMPLEMENTED(_func) \
|
|
void SBCDSMInstance::_func { \
|
|
throw DSMException("core", "cause", "not implemented in DSM SBC"); \
|
|
}
|
|
|
|
NOT_IMPLEMENTED(playPrompt(const string& name, bool loop, bool front));
|
|
NOT_IMPLEMENTED(setPromptSet(const string& name));
|
|
|
|
void SBCDSMInstance::playFile(const string& name, bool loop, bool front) {
|
|
AmAudioFile* af = new AmAudioFile();
|
|
if(af->open(name,AmAudioFile::Read)) {
|
|
ERROR("audio file '%s' could not be opened for reading.\n",
|
|
name.c_str());
|
|
delete af;
|
|
|
|
throw DSMException("file", "path", name);
|
|
|
|
return;
|
|
}
|
|
if (loop)
|
|
af->loop = true;
|
|
|
|
if (front)
|
|
getPlaylist()->addToPlayListFront(new AmPlaylistItem(af, NULL));
|
|
else
|
|
getPlaylist()->addToPlaylist(new AmPlaylistItem(af, NULL));
|
|
|
|
audiofiles.push_back(af);
|
|
CLR_ERRNO;
|
|
}
|
|
|
|
void SBCDSMInstance::playSilence(unsigned int length, bool front) {
|
|
AmNullAudio* af = new AmNullAudio();
|
|
af->setReadLength(length);
|
|
if (front)
|
|
getPlaylist()->addToPlayListFront(new AmPlaylistItem(af, NULL));
|
|
else
|
|
getPlaylist()->addToPlaylist(new AmPlaylistItem(af, NULL));
|
|
|
|
audiofiles.push_back(af);
|
|
CLR_ERRNO;
|
|
}
|
|
|
|
void SBCDSMInstance::playRingtone(int length, int on, int off, int f, int f2, bool front) {
|
|
AmRingTone* af = new AmRingTone(length, on, off, f, f2);
|
|
if (front)
|
|
getPlaylist()->addToPlayListFront(new AmPlaylistItem(af, NULL));
|
|
else
|
|
getPlaylist()->addToPlaylist(new AmPlaylistItem(af, NULL));
|
|
|
|
audiofiles.push_back(af);
|
|
CLR_ERRNO;
|
|
}
|
|
|
|
NOT_IMPLEMENTED(recordFile(const string& name));
|
|
NOT_IMPLEMENTED_UINT(getRecordLength());
|
|
NOT_IMPLEMENTED_UINT(getRecordDataSize());
|
|
NOT_IMPLEMENTED(stopRecord());
|
|
|
|
NOT_IMPLEMENTED(setInOutPlaylist());
|
|
|
|
// void SBCDSMInstance::setInOutPlaylist() {
|
|
// AmB2BMedia *media = call->getMediaSession();
|
|
// if (NULL == media) {
|
|
// ERROR("could not set InOutPlaylist - no media session!\n");
|
|
// return;
|
|
// }
|
|
// media->setFirstStreamInOut(call->isALeg(), getPlaylist(), getPlaylist());
|
|
// }
|
|
|
|
void SBCDSMInstance::setInputPlaylist() {
|
|
AmB2BMedia *media = call->getMediaSession();
|
|
if (NULL == media) {
|
|
ERROR("could not setInputPlaylist - no media session!\n");
|
|
return;
|
|
}
|
|
|
|
media->setFirstStreamInput(call->isALeg(), getPlaylist());
|
|
}
|
|
|
|
NOT_IMPLEMENTED(setOutputPlaylist());
|
|
// void SBCDSMInstance::setOutputPlaylist() {
|
|
// AmB2BMedia *media = call->getMediaSession();
|
|
// if (NULL == media) {
|
|
// ERROR("could not setOutputPlaylist - no media session!\n");
|
|
// return;
|
|
// } media->setFirstStreamOutput(call->isALeg(), getPlaylist());
|
|
// }
|
|
|
|
void SBCDSMInstance::addToPlaylist(AmPlaylistItem* item, bool front) {
|
|
DBG("add item to playlist\n");
|
|
if (front)
|
|
getPlaylist()->addToPlayListFront(item);
|
|
else
|
|
getPlaylist()->addToPlaylist(item);
|
|
}
|
|
|
|
|
|
void SBCDSMInstance::flushPlaylist() {
|
|
DBG("flush playlist\n");
|
|
getPlaylist()->flush();
|
|
}
|
|
|
|
void SBCDSMInstance::addSeparator(const string& name, bool front) {
|
|
unsigned int id = 0;
|
|
if (str2int(name, id)) {
|
|
SET_ERRNO(DSM_ERRNO_UNKNOWN_ARG);
|
|
SET_STRERROR("separator id '"+name+"' not a number");
|
|
return;
|
|
}
|
|
|
|
AmPlaylistSeparator* sep = new AmPlaylistSeparator(call, id);
|
|
if (front)
|
|
getPlaylist()->addToPlayListFront(new AmPlaylistItem(sep, sep));
|
|
else
|
|
getPlaylist()->addToPlaylist(new AmPlaylistItem(sep, sep));
|
|
// for garbage collector
|
|
audiofiles.push_back(sep);
|
|
CLR_ERRNO;
|
|
}
|
|
|
|
void SBCDSMInstance::connectMedia() {
|
|
AmB2BMedia *media = call->getMediaSession();
|
|
if (NULL == media) {
|
|
DBG("media session was not set, creating new one\n");
|
|
media = new AmB2BMedia(call->isALeg() ? call : NULL , call->isALeg() ? NULL : call);
|
|
call->setMediaSession(media);
|
|
// TODO: media stream initialization here (does changeRtpMode help?)
|
|
} else {
|
|
media->pauseRelay();
|
|
}
|
|
|
|
media->addToMediaProcessor();
|
|
|
|
local_media_connected = true;
|
|
}
|
|
|
|
void SBCDSMInstance::disconnectMedia() {
|
|
if (!local_media_connected) {
|
|
DBG("local media not connected, not disconnecting\n");
|
|
return;
|
|
}
|
|
DBG("disconnecting from local media processing, enabling Relay...\n");
|
|
local_media_connected = false;
|
|
|
|
AmB2BMedia *media = call->getMediaSession();
|
|
if (NULL == media) {
|
|
DBG("media session not set, not disconnecting\n");
|
|
return;
|
|
}
|
|
AmMediaProcessor::instance()->removeSession(media);
|
|
media->restartRelay();
|
|
}
|
|
|
|
NOT_IMPLEMENTED(mute());
|
|
NOT_IMPLEMENTED(unmute());
|
|
|
|
NOT_IMPLEMENTED(replaceHdrsCRLF(string& hdrs));
|
|
|
|
/** B2BUA functions */
|
|
NOT_IMPLEMENTED(B2BconnectCallee(const string& remote_party,
|
|
const string& remote_uri,
|
|
bool relayed_invite));
|
|
NOT_IMPLEMENTED(B2BterminateOtherLeg());
|
|
NOT_IMPLEMENTED(B2BaddReceivedRequest(const AmSipRequest& req));
|
|
NOT_IMPLEMENTED(B2BsetRelayEarlyMediaSDP(bool enabled));
|
|
NOT_IMPLEMENTED(B2BsetHeaders(const string& hdr, bool replaceCRLF));
|
|
NOT_IMPLEMENTED(B2BclearHeaders());
|
|
NOT_IMPLEMENTED(B2BaddHeader(const string& hdr));
|
|
NOT_IMPLEMENTED(B2BgetHeaderRequest(const string& hdr, string& out));
|
|
NOT_IMPLEMENTED(B2BgetHeaderReply(const string& hdr, string& out));
|
|
NOT_IMPLEMENTED(B2BgetHeaderParamRequest(const string& hdr, const string& param, string& out));
|
|
NOT_IMPLEMENTED(B2BgetHeaderParamReply(const string& hdr, const string& param, string& out));
|
|
NOT_IMPLEMENTED(B2BremoveHeader(const string& hdr));
|
|
|
|
#undef NOT_IMPLEMENTED
|
|
#undef NOT_IMPLEMENTED_UINT
|