d0170de30a
SEMS is an open-source SIP media server, implementing a B2BUA, voicemail, IVRs, announcements, etc. It is designed to be complementary to SIP proxy-only tools like Kamailio, OpenSIPS, etc. This adds a new port for the SEMS SIP media server, which provides a number of functions (Voicemail, conferencing, B2BUA, etc.) in conjunction with a SIP router like Kamailio or OpenSIPS. The most recent release (1.6.0) is both old and has a significant number of issues on FreeBSD, so this corresponds to the current development branch, which should hopefully become 1.7 in not too very long. I've added one other patch (from SEMS pull request 57) that is required to make message-waiting notifications behave in a useful way but has yet to be merged upstream. WWW: https://github.com/sems-server/sems/ PR: 240048 Reviewed by: koobs (ports) Approved by: koobs (ports) Differential Revision: D21410
266 lines
7.2 KiB
Plaintext
266 lines
7.2 KiB
Plaintext
diff --git a/apps/mwi/etc/mwi.conf b/apps/mwi/etc/mwi.conf
|
|
index 13fcce62c..ef7e5182b 100644
|
|
--- apps/mwi/etc/mwi.conf
|
|
+++ apps/mwi/etc/mwi.conf
|
|
@@ -1,2 +1,13 @@
|
|
-# Presence Server:
|
|
-presence_server=127.0.0.1
|
|
\ No newline at end of file
|
|
+# use domain instead of presence_server
|
|
+use_domain=yes
|
|
+
|
|
+# Presence Server, not used if use_domain is true
|
|
+#presence_server=127.0.0.1
|
|
+
|
|
+# the from user for publish requests, not used if empty
|
|
+from_user=mwi-publisher
|
|
+# the to user for publish requests, not used if empty
|
|
+#to_user=
|
|
+
|
|
+# give a preset route set, not used if empty
|
|
+#route_set="sip:127.0.0.1"
|
|
diff --git a/apps/mwi/mwi.cpp b/apps/mwi/mwi.cpp
|
|
index 4e7035011..47b808be0 100644
|
|
--- apps/mwi/mwi.cpp
|
|
+++ apps/mwi/mwi.cpp
|
|
@@ -1,12 +1,13 @@
|
|
/*
|
|
Copyright (C) Anton Zagorskiy amberovsky@gmail.com
|
|
Oyster-Telecom Laboratory
|
|
-
|
|
+
|
|
Published under BSD License
|
|
*/
|
|
-
|
|
+
|
|
#include "AmPlugIn.h"
|
|
#include "AmSession.h"
|
|
+#include "AmConfig.h"
|
|
#include "AmConfigReader.h"
|
|
#include "AmUtils.h"
|
|
#include "log.h"
|
|
@@ -18,9 +19,9 @@ AmDynInvoke* MWI::MessageStorage = 0;
|
|
|
|
EXPORT_PLUGIN_CLASS_FACTORY(MWI, MOD_NAME);
|
|
|
|
-MWI::MWI(const string& name)
|
|
- : AmDynInvokeFactory(name) {
|
|
- _instance = this;
|
|
+MWI::MWI(const string& name) : AmDynInvokeFactory(name)
|
|
+{
|
|
+ _instance = this;
|
|
};
|
|
|
|
MWI::~MWI() { };
|
|
@@ -28,10 +29,10 @@ MWI::~MWI() { };
|
|
|
|
int MWI::onLoad()
|
|
{
|
|
- AmDynInvokeFactory* ms_fact =
|
|
+ AmDynInvokeFactory* ms_fact =
|
|
AmPlugIn::instance()->getFactory4Di("msg_storage");
|
|
|
|
- if(!ms_fact || !(MessageStorage = ms_fact->getInstance())) {
|
|
+ if (!ms_fact || !(MessageStorage = ms_fact->getInstance())) {
|
|
ERROR("could not load msg_storage. Load a msg_storage implementation module.\n");
|
|
return -1;
|
|
};
|
|
@@ -41,21 +42,38 @@ int MWI::onLoad()
|
|
es_args.push(this);
|
|
es_args.push("publish");
|
|
MessageStorage->invoke("events_subscribe",es_args,ret);
|
|
-
|
|
+
|
|
AmConfigReader cfg;
|
|
- if(cfg.loadFile(AmConfig::ModConfigPath + "mwi.conf")) {
|
|
- ERROR("can not load configuration file\n");
|
|
- return -1;
|
|
+
|
|
+ use_domain = true;
|
|
+ from_user = "mwi-publisher";
|
|
+ to_user = "";
|
|
+ route_set = "";
|
|
+ presence_server = "";
|
|
+
|
|
+ if(cfg.loadFile(AmConfig::ModConfigPath + string(MOD_NAME ".conf"))) {
|
|
+ INFO(MOD_NAME "configuration file (%s) not found, "
|
|
+ "assuming default configuration is fine\n",
|
|
+ (AmConfig::ModConfigPath + string(MOD_NAME ".conf")).c_str());
|
|
+
|
|
+ return 0;
|
|
};
|
|
-
|
|
- presence_server = cfg.getParameter("presence_server");
|
|
- if (presence_server.length())
|
|
- DBG("set presence server '%s'\n", presence_server.c_str());
|
|
- else {
|
|
- ERROR("parameter 'presence_server' did not found in the configuration file\n");
|
|
- return -1;
|
|
- }
|
|
-
|
|
+
|
|
+ use_domain = cfg.getParameter("use_domain", "yes") == "yes";
|
|
+ from_user = cfg.getParameter("from_user", from_user);
|
|
+ to_user = cfg.getParameter("to_user", to_user);
|
|
+ route_set = cfg.getParameter("route_set", route_set);
|
|
+
|
|
+ if (!use_domain) {
|
|
+ presence_server = cfg.getParameter("presence_server", presence_server);
|
|
+ if (presence_server.length() == 0) {
|
|
+ ERROR("use domain set to false, but parameter 'presence_server' not found in the configuration file\n");
|
|
+ return -1;
|
|
+ }
|
|
+
|
|
+ DBG("set presence server '%s'\n", presence_server.c_str());
|
|
+ }
|
|
+
|
|
DBG("MWI module loaded.\n");
|
|
return 0;
|
|
};
|
|
@@ -66,54 +84,80 @@ void MWI::publish(const string& user, const string& domain)
|
|
int new_msgs = 0;
|
|
int all_msgs = 0;
|
|
string headers, body;
|
|
-
|
|
+
|
|
AmArg di_args, ret;
|
|
di_args.push(domain.c_str());
|
|
di_args.push(user.c_str());
|
|
-
|
|
+
|
|
MessageStorage->invoke("userdir_open",di_args,ret);
|
|
-
|
|
+
|
|
if (!ret.size() || !isArgInt(ret.get(0))) {
|
|
ERROR("userdir_open for user '%s' domain '%s' returned no (valid) result.\n", user.c_str(), domain.c_str());
|
|
return;
|
|
};
|
|
-
|
|
+
|
|
all_msgs = ret.get(1).size();
|
|
for (size_t i = 0; i < ret.get(1).size(); i++) {
|
|
AmArg& elem = ret.get(1).get(i);
|
|
-
|
|
- if (elem.get(2).asInt()) // skip empty messages
|
|
+
|
|
+ if (elem.get(2).asInt()) { // skip empty messages
|
|
new_msgs += elem.get(1).asInt();
|
|
- else
|
|
+ }
|
|
+ else {
|
|
all_msgs--;
|
|
+ }
|
|
};
|
|
-
|
|
+
|
|
DBG("Found %d new and %d old messages\n", new_msgs, all_msgs - new_msgs);
|
|
string vm_buf = int2str(new_msgs) + "/" + int2str(all_msgs - new_msgs);
|
|
|
|
headers = "Event: message-summary\r\n";
|
|
headers += "Subscription-State: active\r\n";
|
|
-
|
|
- if (new_msgs > 0)
|
|
+
|
|
+ if (new_msgs > 0) {
|
|
body = "Messages-Waiting: yes\r\n";
|
|
- else
|
|
+ }
|
|
+ else {
|
|
body = "Messages-Waiting: no\r\n";
|
|
+ }
|
|
|
|
body += "Message-Account: sip:" + user + "@" + domain + "\r\n";
|
|
body += "Voice-Message: " + vm_buf + " (" + vm_buf + ")\r\n";
|
|
|
|
AmMimeBody sms_body;
|
|
sms_body.addPart("application/simple-message-summary");
|
|
- sms_body.setPayload((const unsigned char*)body.c_str(),body.length());
|
|
+ sms_body.setPayload((const unsigned char*) body.c_str(), body.length());
|
|
+
|
|
+ string from_uri = "sip:";
|
|
+ string to_uri = "sip:";
|
|
+
|
|
+ if (from_user.length() != 0) {
|
|
+ from_uri += from_user + "@";
|
|
+ }
|
|
+
|
|
+ if (to_user.length() != 0) {
|
|
+ to_uri += to_user + "@";
|
|
+ }
|
|
+
|
|
+ if (use_domain) {
|
|
+ from_uri += domain;
|
|
+ to_uri += domain;
|
|
+ }
|
|
+ else {
|
|
+ from_uri += presence_server;
|
|
+ to_uri += presence_server;
|
|
+ }
|
|
|
|
AmSipDialog tmp_d(NULL);
|
|
- tmp_d.setLocalParty(string("<sip:mwi-publisher@") + presence_server + ">");
|
|
- tmp_d.setRemoteParty(domain.c_str());
|
|
- tmp_d.setRouteSet("sip:" + presence_server);
|
|
tmp_d.setRemoteUri("sip:" + user + "@" + domain);
|
|
- tmp_d.setCallid(AmSession::getNewId() + "@" + presence_server);
|
|
+ tmp_d.setLocalParty("<" + from_uri + ">");
|
|
+ tmp_d.setRemoteParty("<" + to_uri + ">");
|
|
+ if (route_set.length() != 0) {
|
|
+ tmp_d.setRouteSet(route_set);
|
|
+ }
|
|
+ tmp_d.setCallid(AmSession::getNewId() + "@" + AmConfig::SIP_Ifs[tmp_d.getOutboundIf()].getIP());
|
|
tmp_d.setLocalTag(AmSession::getNewId());
|
|
- tmp_d.sendRequest(SIP_METH_NOTIFY, &sms_body, headers);
|
|
+ tmp_d.sendRequest(SIP_METH_PUBLISH, &sms_body, headers);
|
|
};
|
|
|
|
void MWI::invoke(const string& method, const AmArg& args, AmArg& ret)
|
|
@@ -125,6 +169,7 @@ void MWI::invoke(const string& method, const AmArg& args, AmArg& ret)
|
|
publish(user, domain);
|
|
ret.push(0);
|
|
}
|
|
- else
|
|
- throw AmDynInvoke::NotImplemented(method);
|
|
+ else {
|
|
+ throw AmDynInvoke::NotImplemented(method);
|
|
+ }
|
|
};
|
|
diff --git a/apps/mwi/mwi.h b/apps/mwi/mwi.h
|
|
index c1d7f1d65..0ec8ca46e 100644
|
|
--- apps/mwi/mwi.h
|
|
+++ apps/mwi/mwi.h
|
|
@@ -1,7 +1,7 @@
|
|
/*
|
|
Copyright (C) Anton Zagorskiy amberovsky@gmail.com
|
|
Oyster-Telecom Laboratory
|
|
-
|
|
+
|
|
Published under BSD License
|
|
*/
|
|
|
|
@@ -16,15 +16,21 @@ class MWI : public AmDynInvokeFactory, public AmDynInvoke
|
|
private:
|
|
static MWI* _instance;
|
|
static AmDynInvoke* MessageStorage;
|
|
-
|
|
+
|
|
+ bool use_domain;
|
|
+
|
|
+ string from_user;
|
|
+ string to_user;
|
|
+
|
|
+ string route_set;
|
|
string presence_server;
|
|
-
|
|
+
|
|
typedef struct
|
|
{
|
|
unsigned int new_msgs;
|
|
unsigned int saved_msgs;
|
|
} msg_info_struct;
|
|
-
|
|
+
|
|
void getMsgInfo (const string& name, const string& domain, msg_info_struct& msg_info);
|
|
void publish (const string& name, const string& domain);
|
|
|