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"]}]}
= AGIMenu.new(hash)
.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