Freitag, 25. April 2008

Using the Asterisk Voip PBX as SIP Mediaserver

Here I show how the open source Voip PBX Asterisk can be used as a simple mediaserver for Sip applications.

Prerequisites

  • I use the AsteriskNOW software appliance which is prepackaged ready-to-run version of Asterisk. It can be downloaded from http://www.asterisknow.org/downloads . There is a very good documentation avaliable as PDF download from Asterisk Book Download .
  • Asterisk is running on Linux only, so in order to run it on Windows or other OS you need a virtual machine environment e.g. VMware. The download from AsteriskNOW is an ISO image, to run it on VMWare you need VMWare Server which you can get for free from http://www.vmware.com/download/server/ . Alternatively you can use VMware Player and run an Asterisk Virtual Appliance which you can get from the VMware virtual appliance directory at http://www.vmware.com/appliances/ .
  • I prefer to run it from VMware Server. To install it create a new virtual machine, insert a CD made from the ISO image or mount the image directly and follow the instructions of the graphic installer. There is quickstart guide available at http://www.asterisknow.org/support/install .
  • Asterisk provides a Web based administration, but for our usecases it is required to have direct access to the Asterisk console. Use an ssh client (e.g. Putty from http://www.chiark.greenend.org.uk/~sgtatham/putty/ ) to connect to the Asterisk server.

Configuration

  • Connect to Asterisk using the ssh client and login (default user is admin and password is password). Configuration of Asterisk is done by editing various configuration files. When you edit the configuration files you must give yourself the access rights with sudo, e.g sudo vi sip.conf.
  • The sip configuration file is sip.conf in /etc/asterisk. Rename the file from the installation (it is very complicated and contains lots of examples) and create a new sip.conf as follows:

    [general]
    context=default
    allowoverlap=no
    bindport=5060
    bindaddr=0.0.0.0
    srvlookup=yes
    [sailfin]
    type=peer
    context=from-fwd
    host=192.168.0.12
    insecure=port

    The configuration is done by declaring contexts, in our case the sailfin context, type=peer means that this context can only receive calls.The host address has to be replaced with the ip address of the sailfin server (yes the sailfin server since it is the peer, not the Asterisk server). Detailed explanation about contents can be found in the Asterisk book.
  • The next configuration file is extensions.conf, which contains the dialplan. Rename the file and create a new one like for sip.conf. A dialplan tells Asterisk what to do when a call is received. In our example we are using a very simple dialplan which tells Asterisk to play an announcement when a callis received on a certain extension.

    [from-fwd]
    exten => 1234,1,Answer()
    exten => 1234,2,Playback(to-reach-operator)
    exten => 1234,3,Hangup()


    When a call is received for the context from-fwd with the extension 1234 the call is answered, then an announcement is played and the call is finished. Announcement are defined as .wav files with a full path, if no path is specified the default directory /var/lib/asterisk/sounds is used.
  • To activate the context type sip reload from the console (the console can be reached from the Asterisk main menu with ALT-F9). There are many other sip related console commands, they are explained in detail in the Asterisk book in appendix E.
  • The log files for sip can be found in /var/log/asterisk in file full, errors are in file messages.
  • When you now make a sip call to Asterisk (sip:1234@192.168.0.12 in our example) Asterisk answers the call with 200 OK, then receives an ACK, plays the announcement and ends the call with BYE when the announcement is finished.

Next Steps

The usecase described here is an extremely simple one, Asterisk can do much more than that, e.g. it can act as

  • IVR with voice menus reacting on keypresses and event database access.
  • Voice mail server
  • Call recording server

I will try to explore these possibilities and explain in a future post.

Donnerstag, 24. April 2008

Hunting Sample App using ECharts

In this example I show how to build a more complex sample using the ECharts state machine language. Details about ECharts can be found in http://echarts.org/ and also in one my previous posts in screening sample app with ECharts . The sources for the sample are available in cvs using cvs -d:pserver:[your java.net account]@cvs.dev.java.net:/cvs co sailfin/sailfin-tests and then can be found then in community/samples/hunting, or send an email to me to get them.

It provides the functionality to define a list of alternative Sip URI's in a database which are tried in sequence to connect when a call is received until a successful connection is established.

  • The application is triggered only in terminating case via the ECharts application router if the To address matches the specfied pattern in the app router.
  • It then connects the call to a mediaserver (MRF) which plays a dialtone or other announcement to the caller.
  • When the dialtone is established it sends a INVITE to the original called party of the call.
  • When the called party answers with 200 OK it disconnects from the mediaserver and reconnects via REINVITE to the caller. How to setup a mediaserver I will show in a future blog entry.
  • If the called party returns a 486 BUSY or 487 NOT REACHABLE it retrieves the next alternative Sip URI from the database and sends INVITE again.
  • If no alternative called party is available it stops the dialtone and signals a failure to the caller.

Sip Callflow for Redirect on Busy

The following callflow shows a scenario where the B Party is busy and the call is forwarded to C-Party.

A-Party......Sailfin......B-Party......MRF......C-Party

-INVITE SDPa->.....................................
.............-INVITE-------------------->..........
.............<-200 OK SDPm---------------..........
<-200 OK SDPm-.....................................
-ACK--------->.....................................
.............-ACK----------------------->..........
.............-INVITE SDPa--->......................
.............<-486 BUSY------......................
.............-ACK----------->......................
.............-INVITE SDPa------------------------->
.............<-200 OK SDPc-------------------------
.............-BYE----------------------->..........
<-200 OK SDPc-.....................................
.............<-200 OK--------------------..........
-ACK--------->.....................................
.............-ACK--------------------------------->
..................t a l k i n g....................
-BYE--------->.....................................
.............-BYE--------------------------------->
.............<-200 OK------------------------------
<-200 OK------.....................................


State Machines

A graphical description of the state machines can be found in the downloaded sources in hunting/src/echarts/hunting/ech/doc-files

  • The MainFSM machine connects to the MRF to play the dialtone using the ConnectFSM machine. When the dialtone is established it transitions to the HuntingFSM which performs the search and connect for the called party. Finally when the call is successfully established it transitions to the TransparentFSM which handles the talking state of the call and also the turndown at the end.
  • The HuntingFSM runs three paralles states.
    The PLAY_DIALTONE state uses the TransparentFSM and monitors the connection to the MRF.
    The CALL state uses the CallFSM to make a connection the called party until it has a successful connection.
    The REINVITE state uses the ReinviteFSM to handle the reinvite to the caller after a successful connection to the B-Party has been established.
  • The ConnectFSM machine handles the conenction to the MRF.
  • The CallFSM handles connect attempts to the B-Party.
  • The SendReinviteFSM handles the reinvite to the caller.
  • The TransparentFSM handles the talking state and turndown together with the TransparentHandleRequestFSM.

Samstag, 1. März 2008

Screening App uses ECharts Application Router

The screening application now uses the ECharts application router. A good description of the application router can be found in http://echarts.org/ECharts-for-SIP-Servlets/JSR116-Application-Router-Deployment.html.

In the moment the screening app uses the application router only to control triggering so that it is only triggered in originating but not in terminating calls. In the future different kinds of screening could be applied to originating or terminating calls or the screening app could be integrated with other applications in application composition.

Following changes were made to the screening app:
  • A new servlet "InitialServlet" was added.This servlet is triggered when the application is not determined yet and handles the initial routing.
  • In sip.xml the InitialServlet was added and the mapping rules were enhanced to use the results of application routing.
  • The ConnectFSM was enhanced to call the approuter before it sends out an initial INVITE.
  • The approuter.xml file was created, this file has to be copied to sailfin/domains/domain1/config.

Sonntag, 24. Februar 2008

Screening Sample App using ECharts (Part 2 Web Admin)

The third parts shows how to build a web based administration for the screening application. It is built using the Visual Web JavaServer Faces framework. It provides basic CRUD (Create, Read, Update, Delete) functionality for each of the entities in the database.

Database Contents

The database contains the following entities:

  1. UserData:
    holds the user id, the name of the used screening profile, the reference to the provider data and the users's Sip Uri.
  2. ProviderData
    holds the system specific data, in our case the Sip Uri of the announcement to be played when a call is not allowed.
  3. ScreeningProfile
    holds the screening data to be checked in blacklists and whitlelists. Each entry consists of a profile name, a blacklist/whitelist indicator and the Sip Uri to be checked itself.

Page Creation

The application consists of a main page which has links to the Users page, the Providers page and the Screening page. The application is modeled according to the tutorial "Modifying Database Table Rows with the Java Persistence API" in http://www.netbeans.org/kb/60/web/web-jpa-part2.html. Do the following steps to build the application:

  1. Right click on the screening application in the project window and select "New-Visual Web JSF Page" for each page.
  2. Right click on the screening application in the project window and select "New-Visual Web JSF Session Bean" once.
  3. Drag and Drop the hyperlinks for users, providers and screening to the main page.
  4. Add the code for three properties userDataList, providerDataList and screeningProfileList to SessionBean1 as well as the corresponding methods to fill this lists from the database (updateUserData(), updateProviderData() and updateScreeningProfile() ) and one helper method findProviderDataById(String providerId).

Page Layout, each entity page contains:

  • a static text field for the headline
  • a grid panel with id mainContainer
  • a table which is bound to the data in SessionBean1
  • a grid panel with id buttonPanel with 2 buttons, an add-update button and a delete button
  • a grid panel with name addUpdatePanel with one label and one text field or dropdown list for each field in the entity.
  • a hyperlink to go back to the main page

Page Flow, open the faces-config.xml file and in the graphical editor

  1. Connect each of the hyperlinks on the main page with the corresponding entity page
  2. Connect the backlink on the entity pages with the main page
  3. Connect the buttons on each entity page with the entity page itself

Binding to Data

  1. Right click the table and select "Table Layout", then select the corresponding list in "Get Data From".Set the id's of the table columns as [property name]Column and the table fields as [property name]Text.
  2. Set the id's of the addUpdatePanel elements as [property name][field or list]
  3. .For the dropdown list for providers in the Users page select it, right click and select "Bind to Data", then choose providerIdList.
  4. For the dropdown list for screening profiles in the Users page select it, right click and select "Bind to Data", then choose screeningNameList

Handling Selection using an addtional radio button in each table row

  1. Add the code for a TableSelectPhaseListener to the java bean for the page as shown in the example code
  2. Add the javascript code for initAllRows() as shown in the example code. This code is specífic for each page because it also updates the addUpdatePanel fields when a table row is selected.
  3. Right click the table and select "Table Layout", then "New" and select Radio Button from component type and click "Up" until the button is on the top of the column. Set the button's id to selectButton.
  4. Select the radio button, right click in the navigator window and select "Property Bindings". Add the bindings for selected, selected value and name. For name enter manually
    "#{[page name]selectButton.id}". This binds the select button to the TableSelectPhaseListener object.
  5. Select the selectColumn in the navigator window and set the bindings for selectId to
    "#{[page name]selectButton.id}" and the bindings for onClick to "setTimeout('initAllRows()', 0)".This performs the update of the table when a radio button is clicked.
  6. Select the tableRowGroup in the navigator window and set the bindings for selected to the selectedState property.

Command Handlers

  1. Add persistency to the page by right clicking in the source editor and then select"Persistence->Use Entity Manager".
  2. Add the code for merge() and delete() as shown in the example code. You can delete the generated code for persist().
  3. Doubleclick the "add-update" button and enter the java code as shown in the example code.
  4. Doubleclick the "delete" button and enter the java code as shown in the example code.

Build and deploy the application, then point the webbrowser to http://localhost:8080/screening

To make an initial fill of the database go to the screening directory and execute ant -f tests.xml filldata

Freitag, 8. Februar 2008

Screening Sample App - ECharts Details

The following diagram show the structure of the screening application, the diagram is generated by the build process in the echarts/screening/ech/doc-files folder. From the initial state RECEIVE_INVITE either the state PLAY_ANNO is reached (if screening is applied) or the state CONNECT if screening is not applied. When the connection is successfully established the state TALKING becomes active which handles the call until finish.




Montag, 4. Februar 2008

Screening Sample App using ECharts (Part 1 - Application)

In this example I will show how to build a simple screening application for Sailfin using the ECharts state machine language. The application checks the dialed number and either let the call proceed or plays an announcement to the caller that the call is forbidden. Each user is assigned a screening profile which contains his screening rules consisting of blacklisted numbers and exceptional whitelisted numbers. A call is forbidden if the dialed number is contained in the blacklist but not in the whitelist. A call is allowed if the dialed number is not contained in any list or in the whitelist only.

What is ECharts

According to the description on the ECharts website (http://echarts.org/) it is a state machine-based programming language for event-driven systems derived from the standardized UML stacharts language. ECharts is a hosted language which means that it is dependent on an underlying programming language such as Java.

Benefits of ECharts


  • It provides a problem oriented language. State machines are a typical method to describe message processing applications and a language whose paradigm fits to the domain makes application development easier.
  • From an application developers point of view your application has to handle only a single call in contrast to the servlet programming model where you have to deal with all calls handled by the servlet in parallel. The session and dialog management is abstracted by the so called box model.
  • It allows to build reuseable parts by defining state machines which handle a certain task and can be inserted as a single state in an outer state machine. In the example I will show a Connect machine which builds up a connection to the called party, a Transparent machine which handles the talking state and a PlayAnnouncement machine which plays announcements to the caller.

Prerequisites

  1. Get the ECharts 2.2. SDK from http://echarts.org/Downloads.html and unzip it.
  2. The ECharts compiler uses Python, if do not have it get it from http://www.python.org/download/
  3. If on order to use the documentation generator for state machines on Windows you need graphviz from http://www.graphviz.org/. For some strange reasons I had to directly set the path for the graphviz "dot" program in ech2javadoc to make it working.
  4. In the sailfin application server set the following JVM options with the admin console on http://localhost:4848/ by "Application Server->JVM Settings->JVM Options->Add Option".
    -Dorg.echarts.servlet.sip.messagelog=true
    -Dorg.echarts.debugging=true
    -Dorg.echarts.servlet.sip.debugging=true
    -Dorg.echarts.servlet.sip.logdir=${com.sun.aas.instanceRoot}/echlogs
    -Dorg.echarts.system.transitionTimerManager.class=
    org.echarts.servlet.sip.TransitionTimerManager
  5. The sources are available in cvs using
    cvs -d:pserver:[your java.net account]@cvs.dev.java.net:/cvs co sailfin/sailfin-tests
    and then can be found then in community/samples/screening, or send an email to
    peter.c.klein@siemens.com to get them.

Creating the Screening Sample

  1. Create a new database with database name "screening" and a new Converged Servlet Application with name screening like in http://petersjb.blogspot.com/2008/01/create-sip-application-for-sailfin.html
  2. Add the entity classes for UserData, SceeningProfile and ProviderDataand copy their sources from the downloaded cvs sources.
  3. Add the java class for ModelFacade as well as the java class for RegisterSipServlet and copy its sources from the downloaded cvs sources.
  4. Add a java class FSMSupport which contains helper functions called by the state machines and copy its sources from the downloaded cvs sources.
  5. Add antlr.jar, approuter.jar, echarts.jar and echarts-sipserlet.jar in the /lib folder to the projects libraries.
  6. Create a new folder structure echarts/screening/ech in screening/src to hold the echarts sources and copy the *.ech files from the downloaded cvs sources. Add this folder in the project properties to sources.
  7. Add to the build.xml file the properties and targets -pre-compile (for ech2java compilation) and -post-compile (for generating state machine documentation) as in the build.xml file in the downloaded sources.
  8. Add the EChartsSipServlet and the RegisterSipServlet to the sip.xml as shown in the example code. Note that the state machines themselves are no servlets but are called from the EChartsSipServlet. Also add ModelFacade to web.xml
  9. Compile and deploy the application.
  10. Setup the data in the database as shown in the filldata.sql file in the
    "scenarios/tc-001-basic-ua" folder.
  11. Depending on the dialed number and contents of the screening profile the call will either be connected to the called party or the party defined in the ProviderData.anno attribute.

I the next post I will show the web based admin and some insights how it works.

Notes

The web based admin is already included in the sources but in the moment only viewing works, modify and delete have to be added.

Freitag, 1. Februar 2008

Create a SIP Application for Sailfin (Part 3 Webapp)

The third and last part shows how to build a simple web based administration for the user database with basic create, read, update and delete functionality. The application is modeled after the sample in http://www.netbeans.org/kb/60/web/inserts-updates-deletes.html.

View the Data
  1. Create the web page with "New->Visual Web JSF Page" and a session bean with
    "New->Visual Web JSF Session Bean".
  2. Open the visual web page editor for the created page. Then go to the "Services" tab, connect to the database and select the USERDATA table. Drag and drop it to the editor's canvas. You will see in the navigator window that a Data Provider object and a Row Set object were created.
  3. Drag and drop a table on the canvas. Select the table, right click and select "Bind to Data". Choose the userdataDataProvider which was created before and click OK. The table is now able to display data for users.
  4. Deploy the application and point your browser to localhost:8080/b2bua to view the contents of USERDATA.

Create and Delete

  1. Add a button "Add User" and a button "Save Changes".
  2. Doubleclick the "Add User" button and enter the action code from the sources.
  3. Doubleclick the "Save Changes" button and enter the action code from the sources.
  4. Select the table and open the layout editor. Add a new column and change the type of the column to "Button".
  5. Doubleclick the "Delete" button and enter the action code from the sources.
  6. Deploy the application again. You can now create and delete user data. The changes are stored in the database when you click "Save Changes".