Author: Michael Komitee <[email protected]>

License: BSD License

Copyright (c) 2007, Vonage Holdings

All rights reserved.

Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:

    * Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
    * Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
    * Neither the name of Vonage Holdings nor the names of its       
contributors may be used to endorse or promote products derived from this
software without specific prior written permission.

THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
POSSIBILITY OF SUCH DAMAGE.

Asterisk Ruby Framework – AGI for Ruby

AGI, AGIMenu, AGISelection, and AGIServer work together with their supporting libraries to complete an Asterisk AGI Framework for applications.

* AGI: handles interaction between ruby and Asterisk. Asterisk connects to the AGI via either pipes or sockets
* AGIMenu: implements more complex logic dealing with menus and prompts
* AGISelection: implements more complex logic dealing with audio playback and digit strings
* AGIServer: implements a TCPServer via a block or rails routes, which instantiates AGI objects for Asterisk interaction

Download

The latest version of asterisk-ruby can be found at

* http://rubyforge.org/project/

Installation

Normal Installation

Setup based installation is currently unsupported

GEM Installation

Download and install asterisk-ruby with the following.

% gem install asterisk-ruby

Usage, …

AGI

The AGI object can be used to interact with an Asterisk. It can be used within the AGIServer framework, or independantly. Simply instantiate an AGI object and pass it input and output IO objects. Asterisk extensions can exec an agi script (in asterisk parlance, agi or deadagi), in which case you'll want to use stdin and stdout, or asterisk extensions can connect to a daemonized agi server (in asterisk parlance, fagi) and you'd want to use a tcp socket.

  agi = AGI.new(:input => STDIN, :output => STDOUT)
  agi.init
  agi.answer
  agi.hangup

Note, all agi instances will be uninitialized. The initialization process of an agi channel must be performed before any other interaction with the channel can be accomplished.

AGIMenu

Sometimes when dealing with Asterisk AGI, a generic AGIMenu object can come in handy. Thats where the aptly named AGIMenu object comes into play. With AGIMenu, you can configure a menu of options based on a yaml configuration or a hash. For example:

AGIMenu.sounds_dir = 'agimenu-test/sounds/'
hash = {:introduction=>["welcome", "instructions"],
  :conclusion=>"what-is-your-choice",
  :timeout=>17,
  :choices=>
  [{:dtmf=>"*", :audio=>["to-go-back", "press", "digits/star"]},
    {:dtmf=>1, :audio=>["press", "digits/1", "for-option-1"]},
    {:dtmf=>2, :audio=>["press", "digits/2", "for-option-2"]},
    {:dtmf=>"#", :audio=>["or", "press", "digits/pound", "to-repeat"]}]}

menu = AGIMenu.new(hash)
menu.play(:agi => AGI.new())

This results in Asterisk using sounds in it’s agimenu-test/sounds directory to play a menu which accepts the DTMF *, 1, 2, or #. It will first play the introduction sound files ‘welcome’ and ‘instructions’, then play the menu sound files ‘to-go-back’ ‘press’ ‘digits/star’ ‘press’ ‘digits/1’ ‘for-option-1’ ‘press ’digits/2’ ‘for-option-2’ ‘or’ ‘press’ ‘digits/pound’ ‘to-repeat’, and then play the conslusion sound file ‘what-is-your-choice’ and then wait 17 seconds.

You could hand AGIMenu.new() a filename, a file, a yaml oject, a hash, or nothing at all. You can then modify the menu with the classes various methods.

AGISelection

AGISelection can be used to get a string of digits from an asterisk channel. You configure it similarly to an AGIMenu, but it takes a max_digits paramater and only accepts a single audio file. AGIMenu only accepts a single dtmf digit, whereas AGISelection can accept multiple digits. It would be used to have a user input a PIN or a telephone number or something similar.

agi = AGI.new()
yaml_hash = {:audio => 'tt-monkeys', :max_digits => 4}
foo = AGISelection.new(yaml_hash)
foo.read(:agi => AGI.new())

AGIServer

There are several ways to use the AGIServer module. All of them have a few things in common. In general, since we’re creating a server, we need a way to cleanly kill it. So we setup sigint anf sigterm handlers to shutdown all instances of AGIServer Objects

trap('INT')   { AGIServer.shutdown }
trap('TERM')   { AGIServer.shutdown }

We also tend to use a logger because this should be daemonized. While developing, I reccomend you log to STDERR

logger = Logger.new(STDERR)
logger.level = Logger::DEBUG

I use YAML for configuration options. This just sets up the bind port, address, and some threading configuration options.

config = YAML.load_file('config/example-config.yaml')
config[:logger] = logger
config[:params] = {:custom1 => 'data1'}

And then we generate our server

begin
  MyAgiServer = AGIServer.new(config)
rescue Errno::EADDRINUSE
  error = "Cannot start MyAgiServer, Address already in use."
  logger.fatal(error)
  print "#{error}\n"
  exit
else
  print "#{$$}"
end

In this example, I’ll show you the rails-routing means of working with the AGIServer. Define a Route class along with a few routes, and start the server.

class TestRoutes < AGIRoute
  def sample
    agi.answer
    print  "CUSTOM1 = [#{params[:custom1]}]\n"
    print  "URI     = [#{request[:uri]}]\n"
    print  "ID      = [#{request[:id]}]\n"
    print  "METHOD  = [#{request[:method]}]\n"
    print  "OPTIONS = #{request[:options].pretty_inspect}"
    print  "FOO     = [#{request[:options]['foo']}]\n"
    print '-' * 10 + "\n"
    helper_method
    agi.hangup
  end
  private
  def helper_method
    print "I'm private which means I'm not accessible as a route!\n"
  end
end
MyAgiServer.start
MyAgiServer.finish

Pointing an asterisk extension at agi://localhost:4573/TestRoutes/sample/1/?foo=bar will execute the sample method in the TestRoutes class.

In this example, I’ll show you how to use a block to define the AGI logic. Simply start the server and pass it a block expecting an agi object:

MyAgiServer.start do |agi|
  agi.answer
  puts "I'm Alive!"
  agi.hangup
end
MyAgiServer.finish

In this example, I’ll show you another way to use a block to define the AGI logic. This block makes configuration parameters available during the call:

MyAgiServer.start do |agi,params|
  agi.answer
  print  "PARAMS = #{params.pretty_inspect}"
  agi.hangup
end
MyAgiServer.finish