Skip to main content

Scaling Asterisk applications

· 4 min read
Sylvain Baubeau
Sylvain Afchain

This article is the first one of a series about the work that we do at Wazo around Asterisk application scalability. For this first post, we will describe some high level aspects of the approach we take to scale Stasis applications. We will explain which components are involved and their role. Let’s take a simple application as an example.

Our sample application

Our application will be a simple conference app : when a user calls the 8000 extension, he will be placed in a conference. Typical call flow would be like :

  • Call arrives on an Asterisk instance
  • The App creates a bridge if it doesn’t exist already
  • The App answers the call and place it in the bridge

The difficulty we have here is that we have to deal with multiple Asterisk instances as we want to distribute the load and not put all calls and bridges of the app on the same Asterisk node.

High level overview

Let’s see how the applications and the different components interact :

High level overview

Consul

  • Will hold the list of the applications and their state
  • Maintain the list of Asterisk instances

RabbitMQ

Used for forwarding of stasis messages

App

Our conference app, written in Python for demo purpose.

API gateway

In order to route API calls to the appropriate Asterisk instance we need to introduce a new component. It will be in charge of :

  • Register the app into Consul
  • Handle the REST API app requests, acting a bit like an ARI proxy
  • Keep track of the resources belonging to Asterisk instance, so that we can route the Answer request to the Asterisk that actually received the call.

Asterisk cluster

A set of Asterisk instances. Each Asterisk instance will:

  • Registers at startup into a Service Discovery.
  • Create the Stasis application, fetched from Consul.
  • Forward Stasis messages through the AMQP bus

Kamailio

Randomly dispatches the SIP calls to the Asterisk instances

API gateway deep dive

To create our app, we have to do more than redirecting requests to the right Asterisk instance : we need to have 2 users connected to 2 different Asterisk instances to be able to actually talk together. To do so, the API gateway will generate a call (using Originate) between the Asterisk instances and bridge them together.

A typical call flow for an application answering a call and then playing a sound would be like this

Workflow

A python code would be like following

import asyncio

from app_sdk import Application, Config

APP_ID = "conf"
APP_NAME = "conf"
BRIDGE_ID = "conf"


class BridgeApplication(Application):

async def on_start(self, context):
await self.get_or_create_bridge(context, BRIDGE_ID, "mixing")
await self.answer(context)

async def on_up(self, context):
await self.bridge_add_channel(context, BRIDGE_ID)


def main():
config = Config()
app = BridgeApplication(config, APP_ID, APP_NAME)
app.launch()


if __name__ == "__main__":
main()

Conclusion

As we will continue to improve the solution more features will be added to the API Gateway mainly to support complex workflow hiding the complexity of having to deal with multiple Asterisk instances. Stay tuned !

Show me the code

All the work presented in the blog post is available in the following Git repositories:

Discussion

Comments or questions in this forum post.