freebsd-ports/net/sems/files/patch-mwi-57
Nathan Whitehorn d0170de30a net/sems: SIP Express Media Server (new port)
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
2019-08-27 03:15:35 +00:00

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);