/rapidsms-httprouter

Implements a router in the HTTP thread, one message per call.

Primary LanguagePython

RapidSMS HTTP Router

Implements HTTP endpoints to allow for the RapidSMS 'routing' process to be done in the HTTP thread.

Distinct features

  • All message handling is done in the Django HTTP thread
  • Easy 'AJAX' endpoints for receiving messages, marking messages as delivered and getting the current outbox
  • Supports either queueing messages in the DB to be sent off by an outside process (pull) or sending messages to a configured URL. (push)
  • Includes a simple HTTP console web application that integrates with RapidSMS
  • "play alike" implementation with RapidSMS, no changes needed to existing applications
  • Easy and reliable standalone integration with Kannel
The official source code repository is:
http://www.github.com/nyaruka/rapidsms-httprouter
Built by Nyaruka Ltd:
http://www.nyaruka.com
Contributions by UNICEF Uganda:
http://www.unicef.org

Caveats

Since this obviously does things differently, some things break. Specifically the following apps will no longer work and should be removed from your INSTALLED_APPS and RAPIDSMS_TABS:

rapidsms.contrib.messagelog
rapidsms.contrib.httptester
rapidsms.contrib.ajax

The HTTP Router app replaces most of the functionality provided by these packages. Specifically all messages passing through the HTTP router will be logged automatically and it provides a web interface for viewing and submitting new messages.

Installation From Cheese Shop

You can install the latest version of the rapidsms-httprouter library straight from the cheese shop:

% pip install rapidsms-httprouter

Installation From Github

You can always get the latest version of rapidsms-httprouter from github. Note that the tip of the repo is not guaranteed to be stable. You should use the verison available via pip unless you have a specific reason not to.

You can install the requirements using the pip-requires.txt file:

% pip install -r pip-requires.txt

Configuration

Put rapidsms_httprouter in your path, then add it to your INSTALLED_APPS setting:

INSTALLED_APPS = [
  "rapidsms",
  ..
  ..
  "rapidsms_httprouter"
]

And add it to your project's urls.py:

urlpatterns = patterns('',
   .. other url patterns ..
   ('', include('rapidsms_httprouter.urls'))
)

rapidsms-httprouter also only pulls in the applications you specify for SMS handling. This lets you use the models from an existing SMS application. So you'll need to add an SMS_APPS list to your settings.py:

SMS_APPS = [
    "mysms.coolapp",
]

Finally, if you want to have the router push messages to a specific URL when they are sent, you need to specify that in the settings.py as well:

ROUTER_URL = "http://backend.server.com/send?backend=%(backend)s&recipient=%(recipient)s&text=%(text)s"

The following fields will be substituted into that string and the resulting URL will then be called via an HTTP GET:

'backend': the backend that is sending this message
'recipient': the phone number to send to
'text': the text to send
'id': the internal rapidsms id for this message

If you want to use the included console and http tester, add it as a tab:

RAPIDSMS_TABS = [
  ..
  ("httprouter-console", "Console"),
]

Usage

If you installed the tab, you should be able to click on the Console tab and immediately begin sending messages. Note that you do not need to run the router process for this to work, instead the HTTP backend detects that there is no router and queues the messages for sending later. (the 'Q' status represents this)

In your app.py, httprouter-aware applications can make use of an extra attribute that will be added to the IncomingMessage object, 'db_message'. This is a reference to the actual database-persisted message:

def handle (self, message):
    if hasattr(message, 'db_message'):
       # do something cool, like add the message as a foreign key
       # to one of your app's models, so you know where the model
       # originated

Endpoints

The HTTP router provides the following endpoints:

Receive

Messages can be handled and put through the router process using the URL, the result is json:

/router/receive?backend=<backend name>&sender=<sender number>&message=<message text>

Outbox

You can see any pending messages which need to be sent using the URL, the result is json:

/router/outbox

Delivered

You can mark a message as sent, or delivered usign the URL:

/router/delivered?message_id=<message id>

Kannel Integration

RapidSMS-HttpRouter works especially well when used with a standalone Kannel configuration. You just need to configure it to send messages in the format Kannel expects and vice versa.

In your settings.py set your ROUTER_URL like so, adjusting appropriately based on your Kannel configuration:

ROUTER_URL = "http://localhost:13013/cgi-bin/sendsms?from=123&username=kannel&password=kannel&text=%(text)s&to=%(recipient)s&smsc=%(backend)s&dlr_url=http%%3A%%2F%%2Fmyrapid.com%%2Frouter%%2Fdelivered%%2F%%3Fmessage_id%%3D%(id)s"

The important thing to note here is the dlr_url parameter, which while optional, lets you get delivery reports and mark messages as not just sent but actually delivered according to the SMSC.

A basic Kannel sms-service configuration that would work for this might be:

group = sms-service
keyword = default
max-messages = 0
get-url = "http://myrapid.com/router/receive/?backend=%i&sender=%p&message=%b"
allowed-receiver-prefix = 123;+123
concatenation = true
assume-plain-text = true
accept-x-kannel-headers = true
omit-empty = true

Multiple Backends

Note that if you have multiple backends, you can set your ROUTER_URL setting to map between backend names and URLs, ie:

ROUTER_URL = {
    'tigo': 'http://kannel.tigo.com/cgi-bin/sendsms?from=123&username=kannel&password=kannel&text=%(text)s&to=%(recipient)s&smsc=%(backend)s',
    'default': 'http://kannel.mtn.com/cgi-bin/sendsms?from=123&username=kannel&password=kannel&text=%(text)s&to=%(recipient)s&smsc=%(backend)s',
}

Note that you must either have one entry per backend, or include a 'default' element, which will be used whenever there is not a specific match.

Security

It is a good idea to have some security on who can deliver messages to your system, who can see the outbox and who can can mark messages as delivered. You can lock these down in a rudimentary fashion by settings the ROUTER_PASSWORD attribute in your settings.py:

ROUTER_PASSWORD = "landshark"

Any incoming requests to those endpoints will fail if it is not included.

Celery & Redis

HttpRouter has two dependencies if you plan on using a ROUTER_URL and not just have message queue up, Redis and Celery. (note you can use Redis as a backend for Celery) You'll need something like the following your settings.py in order to get things working:

#-----------------------------------------------------------------------------------
# Async tasks with django-celery
#-----------------------------------------------------------------------------------

import djcelery
djcelery.setup_loader()

CELERY_RESULT_BACKEND = 'database'

BROKER_BACKEND = 'redis'
BROKER_HOST = 'localhost'
BROKER_PORT = 6379
BROKER_VHOST = '1'

REDIS_PORT=6379
REDIS_HOST='localhost'
REDIS_DB=1

#-----------------------------------------------------------------------------------
# Crontab Schedule
#-----------------------------------------------------------------------------------

from datetime import timedelta

CELERYBEAT_SCHEDULE = {
     "runs-every-five-minutes": {
         'task': 'rapidsms_httprouter.tasks.resend_errored_messages_task',
         'schedule': timedelta(minutes=1),
     },
}