mtrudel-adhearsion 0.8.3

Sign up to get free protection for your applications and to get access to all the features.
Files changed (110) hide show
  1. data/CHANGELOG +26 -0
  2. data/EVENTS +11 -0
  3. data/LICENSE +456 -0
  4. data/Rakefile +127 -0
  5. data/adhearsion.gemspec +149 -0
  6. data/app_generators/ahn/USAGE +5 -0
  7. data/app_generators/ahn/ahn_generator.rb +91 -0
  8. data/app_generators/ahn/templates/.ahnrc +34 -0
  9. data/app_generators/ahn/templates/README +8 -0
  10. data/app_generators/ahn/templates/Rakefile +25 -0
  11. data/app_generators/ahn/templates/components/ami_remote/ami_remote.rb +15 -0
  12. data/app_generators/ahn/templates/components/disabled/HOW_TO_ENABLE +7 -0
  13. data/app_generators/ahn/templates/components/disabled/restful_rpc/README.markdown +11 -0
  14. data/app_generators/ahn/templates/components/disabled/restful_rpc/example-client.rb +48 -0
  15. data/app_generators/ahn/templates/components/disabled/restful_rpc/restful_rpc.rb +87 -0
  16. data/app_generators/ahn/templates/components/disabled/restful_rpc/restful_rpc.yml +34 -0
  17. data/app_generators/ahn/templates/components/disabled/restful_rpc/spec/restful_rpc_spec.rb +263 -0
  18. data/app_generators/ahn/templates/components/disabled/sandbox/sandbox.rb +104 -0
  19. data/app_generators/ahn/templates/components/disabled/sandbox/sandbox.yml +2 -0
  20. data/app_generators/ahn/templates/components/disabled/stomp_gateway/README.markdown +47 -0
  21. data/app_generators/ahn/templates/components/disabled/stomp_gateway/stomp_gateway.rb +34 -0
  22. data/app_generators/ahn/templates/components/disabled/stomp_gateway/stomp_gateway.yml +12 -0
  23. data/app_generators/ahn/templates/components/simon_game/simon_game.rb +56 -0
  24. data/app_generators/ahn/templates/config/startup.rb +50 -0
  25. data/app_generators/ahn/templates/dialplan.rb +3 -0
  26. data/app_generators/ahn/templates/events.rb +32 -0
  27. data/bin/ahn +28 -0
  28. data/bin/ahnctl +68 -0
  29. data/bin/jahn +42 -0
  30. data/examples/asterisk_manager_interface/standalone.rb +51 -0
  31. data/lib/adhearsion.rb +37 -0
  32. data/lib/adhearsion/cli.rb +223 -0
  33. data/lib/adhearsion/component_manager.rb +207 -0
  34. data/lib/adhearsion/component_manager/component_tester.rb +55 -0
  35. data/lib/adhearsion/component_manager/spec_framework.rb +24 -0
  36. data/lib/adhearsion/events_support.rb +84 -0
  37. data/lib/adhearsion/foundation/all.rb +9 -0
  38. data/lib/adhearsion/foundation/blank_slate.rb +5 -0
  39. data/lib/adhearsion/foundation/custom_daemonizer.rb +45 -0
  40. data/lib/adhearsion/foundation/event_socket.rb +203 -0
  41. data/lib/adhearsion/foundation/future_resource.rb +36 -0
  42. data/lib/adhearsion/foundation/global.rb +1 -0
  43. data/lib/adhearsion/foundation/metaprogramming.rb +17 -0
  44. data/lib/adhearsion/foundation/numeric.rb +13 -0
  45. data/lib/adhearsion/foundation/pseudo_guid.rb +10 -0
  46. data/lib/adhearsion/foundation/relationship_properties.rb +42 -0
  47. data/lib/adhearsion/foundation/string.rb +26 -0
  48. data/lib/adhearsion/foundation/synchronized_hash.rb +96 -0
  49. data/lib/adhearsion/foundation/thread_safety.rb +7 -0
  50. data/lib/adhearsion/host_definitions.rb +67 -0
  51. data/lib/adhearsion/initializer.rb +373 -0
  52. data/lib/adhearsion/initializer/asterisk.rb +81 -0
  53. data/lib/adhearsion/initializer/configuration.rb +254 -0
  54. data/lib/adhearsion/initializer/database.rb +50 -0
  55. data/lib/adhearsion/initializer/drb.rb +31 -0
  56. data/lib/adhearsion/initializer/freeswitch.rb +22 -0
  57. data/lib/adhearsion/initializer/rails.rb +41 -0
  58. data/lib/adhearsion/logging.rb +92 -0
  59. data/lib/adhearsion/tasks.rb +16 -0
  60. data/lib/adhearsion/tasks/database.rb +5 -0
  61. data/lib/adhearsion/tasks/deprecations.rb +59 -0
  62. data/lib/adhearsion/tasks/generating.rb +20 -0
  63. data/lib/adhearsion/tasks/lint.rb +4 -0
  64. data/lib/adhearsion/tasks/testing.rb +37 -0
  65. data/lib/adhearsion/version.rb +9 -0
  66. data/lib/adhearsion/voip/asterisk.rb +4 -0
  67. data/lib/adhearsion/voip/asterisk/agi_server.rb +84 -0
  68. data/lib/adhearsion/voip/asterisk/commands.rb +1314 -0
  69. data/lib/adhearsion/voip/asterisk/config_generators/agents.conf.rb +140 -0
  70. data/lib/adhearsion/voip/asterisk/config_generators/config_generator.rb +101 -0
  71. data/lib/adhearsion/voip/asterisk/config_generators/queues.conf.rb +250 -0
  72. data/lib/adhearsion/voip/asterisk/config_generators/voicemail.conf.rb +240 -0
  73. data/lib/adhearsion/voip/asterisk/config_manager.rb +71 -0
  74. data/lib/adhearsion/voip/asterisk/manager_interface.rb +597 -0
  75. data/lib/adhearsion/voip/asterisk/manager_interface/ami_lexer.rb +1589 -0
  76. data/lib/adhearsion/voip/asterisk/manager_interface/ami_lexer.rl.rb +286 -0
  77. data/lib/adhearsion/voip/asterisk/manager_interface/ami_messages.rb +78 -0
  78. data/lib/adhearsion/voip/asterisk/manager_interface/ami_protocol_lexer_machine.rl +87 -0
  79. data/lib/adhearsion/voip/asterisk/special_dial_plan_managers.rb +80 -0
  80. data/lib/adhearsion/voip/asterisk/super_manager.rb +19 -0
  81. data/lib/adhearsion/voip/call.rb +453 -0
  82. data/lib/adhearsion/voip/call_routing.rb +64 -0
  83. data/lib/adhearsion/voip/commands.rb +9 -0
  84. data/lib/adhearsion/voip/constants.rb +39 -0
  85. data/lib/adhearsion/voip/conveniences.rb +18 -0
  86. data/lib/adhearsion/voip/dial_plan.rb +218 -0
  87. data/lib/adhearsion/voip/dsl/dialing_dsl.rb +151 -0
  88. data/lib/adhearsion/voip/dsl/dialing_dsl/dialing_dsl_monkey_patches.rb +37 -0
  89. data/lib/adhearsion/voip/dsl/dialplan/control_passing_exception.rb +27 -0
  90. data/lib/adhearsion/voip/dsl/dialplan/dispatcher.rb +124 -0
  91. data/lib/adhearsion/voip/dsl/dialplan/parser.rb +71 -0
  92. data/lib/adhearsion/voip/dsl/dialplan/thread_mixin.rb +16 -0
  93. data/lib/adhearsion/voip/dsl/numerical_string.rb +117 -0
  94. data/lib/adhearsion/voip/freeswitch/basic_connection_manager.rb +48 -0
  95. data/lib/adhearsion/voip/freeswitch/event_handler.rb +58 -0
  96. data/lib/adhearsion/voip/freeswitch/freeswitch_dialplan_command_factory.rb +129 -0
  97. data/lib/adhearsion/voip/freeswitch/inbound_connection_manager.rb +38 -0
  98. data/lib/adhearsion/voip/freeswitch/oes_server.rb +195 -0
  99. data/lib/adhearsion/voip/menu_state_machine/calculated_match.rb +80 -0
  100. data/lib/adhearsion/voip/menu_state_machine/matchers.rb +123 -0
  101. data/lib/adhearsion/voip/menu_state_machine/menu_builder.rb +58 -0
  102. data/lib/adhearsion/voip/menu_state_machine/menu_class.rb +149 -0
  103. data/lib/theatre.rb +151 -0
  104. data/lib/theatre/README.markdown +64 -0
  105. data/lib/theatre/callback_definition_loader.rb +84 -0
  106. data/lib/theatre/guid.rb +23 -0
  107. data/lib/theatre/invocation.rb +121 -0
  108. data/lib/theatre/namespace_manager.rb +153 -0
  109. data/lib/theatre/version.rb +2 -0
  110. metadata +182 -0
@@ -0,0 +1,55 @@
1
+ module ComponentTester
2
+
3
+ class << self
4
+
5
+ ##
6
+ #
7
+ #
8
+ # @return [Module] an anonymous module which includes the ComponentTester module.
9
+ #
10
+ def new(component_name, component_directory)
11
+ component_directory = File.expand_path component_directory
12
+ main_file = component_directory + "/#{component_name}/#{component_name}.rb"
13
+
14
+ component_manager = Adhearsion::Components::ComponentManager.new(component_directory)
15
+ component_module = Adhearsion::Components::ComponentManager::ComponentDefinitionContainer.load_file main_file
16
+
17
+ Module.new do
18
+
19
+ extend ComponentTester
20
+
21
+ (class << self; self; end).send(:define_method, :component_manager) { component_manager }
22
+ (class << self; self; end).send(:define_method, :component_name) { component_name }
23
+ (class << self; self; end).send(:define_method, :component_module) { component_module }
24
+ (class << self; self; end).send(:define_method, :component_directory) { component_directory }
25
+
26
+
27
+ define_method(:component_manager) { component_manager }
28
+ define_method(:component_name) { component_name }
29
+ define_method(:component_module) { component_module }
30
+ define_method(:component_directory) { component_directory }
31
+
32
+ def self.const_missing(name)
33
+ component_module.const_get name
34
+ end
35
+
36
+ end
37
+ end
38
+ end
39
+
40
+ def helper_method(name)
41
+ Object.new.extend(component_module).method(name)
42
+ end
43
+
44
+ def config
45
+ component_manager.configuration_for_component_named component_name
46
+ end
47
+
48
+ def initialize!
49
+ metadata = component_module.metaclass.send(:instance_variable_get, :@metadata)
50
+ if metadata && metadata[:initialization_block].kind_of?(Proc)
51
+ metadata[:initialization_block].call
52
+ end
53
+ end
54
+
55
+ end
@@ -0,0 +1,24 @@
1
+ require 'adhearsion/component_manager/component_tester'
2
+ begin
3
+ require 'spec'
4
+ rescue LoadError
5
+ abort 'You do not have the "rspec" gem installed! You must install it to continue.\n\nsudo gem install rspec\n\n'
6
+ end
7
+
8
+ begin
9
+ require 'rr'
10
+ rescue LoadError
11
+ abort 'You do not have the "rr" gem installed! You must install it to continue.\n\nsudo gem install rr\n\n'
12
+ end
13
+
14
+ module ComponentConfigurationSpecHelper
15
+ def mock_component_config_with(new_config)
16
+ Object.send(:remove_const, :COMPONENTS) rescue nil
17
+ Object.send(:const_set, :COMPONENTS, OpenStruct.new(new_config))
18
+ end
19
+ end
20
+
21
+ Spec::Runner.configure do |config|
22
+ config.mock_with :rr
23
+ config.include ComponentConfigurationSpecHelper
24
+ end
@@ -0,0 +1,84 @@
1
+ %w[theatre jicksta-theatre].each do |theatre_gem_name|
2
+ # Some older versions of the master branch required the theatre gem be installed. This is deprecation logic to inform the
3
+ # user to uninstall the gem if they happened to have used theatre in the past.
4
+ begin
5
+ gem theatre_gem_name
6
+ rescue LoadError
7
+ # Good. It should not be installed.
8
+ else
9
+ abort <<-MESSAGE
10
+ It seems you have the "#{theatre_gem_name}" gem installed. As of Dec. 7th, 2008 Theatre has been rolled into Adhearsion
11
+ and will be distributed with it.
12
+
13
+ Please uninstall the gem by doing "sudo gem uninstall #{theatre_gem_name}".
14
+
15
+ You will not need to install it again in the future. Sorry for the inconvenience.
16
+ MESSAGE
17
+ end
18
+ end
19
+
20
+ require 'theatre'
21
+
22
+
23
+ module Adhearsion
24
+ module Events
25
+
26
+ DEFAULT_FRAMEWORK_EVENT_NAMESPACES = %w[
27
+ /after_initialized
28
+ /shutdown
29
+ /asterisk/manager_interface
30
+ /asterisk/before_call
31
+ /asterisk/after_call
32
+ /asterisk/hungup_call
33
+ /asterisk/failed_call
34
+ ]
35
+
36
+ class << self
37
+
38
+ def framework_theatre
39
+ defined?(@@framework_theatre) ? @@framework_theatre : reinitialize_theatre!
40
+ end
41
+
42
+ def trigger(*args)
43
+ framework_theatre.trigger(*args)
44
+ end
45
+
46
+ def trigger_immediately(*args)
47
+ framework_theatre.trigger_immediately(*args)
48
+ end
49
+
50
+ def reinitialize_theatre!
51
+ @@framework_theatre.gracefully_stop! if defined? @@framework_theatre
52
+ rescue
53
+ # Recover and reinitalize
54
+ ensure
55
+ # TODO: Extract number of threads to use from AHN_CONFIG
56
+ @@framework_theatre = Theatre::Theatre.new
57
+ DEFAULT_FRAMEWORK_EVENT_NAMESPACES.each do |namespace|
58
+ register_namespace_name namespace
59
+ end
60
+ return @@framework_theatre
61
+ end
62
+
63
+ def register_namespace_name(name)
64
+ framework_theatre.register_namespace_name name
65
+ end
66
+
67
+ def stop!
68
+ Events.trigger :shutdown
69
+ framework_theatre.graceful_stop!
70
+ framework_theatre.join
71
+ end
72
+
73
+ def register_callback(namespace, block_arg=nil, &method_block)
74
+ raise ArgumentError, "Cannot supply two blocks!" if block_arg && block_given?
75
+ block = method_block || block_arg
76
+ raise ArgumentError, "Must supply a callback!" unless block
77
+
78
+ framework_theatre.register_callback_at_namespace(namespace, block)
79
+ end
80
+
81
+ end
82
+
83
+ end
84
+ end
@@ -0,0 +1,9 @@
1
+ require 'English'
2
+ require 'tmpdir'
3
+ require 'tempfile'
4
+ require 'active_support'
5
+
6
+ # Require all other files here.
7
+ Dir.glob File.join(File.dirname(__FILE__), "*rb") do |file|
8
+ require file
9
+ end
@@ -0,0 +1,5 @@
1
+ class BlankSlate
2
+ instance_methods.each do |method|
3
+ undef_method method unless method =~ /^__/ || method == 'instance_eval'
4
+ end
5
+ end
@@ -0,0 +1,45 @@
1
+ # This is largely based on the Daemonize library by Travis Whitton and
2
+ # Judson Lester. http://grub.ath.cx/daemonize. I cleaned it up a bit to
3
+ # meet Adhearsion's quality standards.
4
+ module Adhearsion
5
+ module CustomDaemonizer
6
+
7
+ # Try to fork if at all possible retrying every 5 sec if the
8
+ # maximum process limit for the system has been reached
9
+ def safefork
10
+ begin
11
+ pid = fork
12
+ return pid if pid
13
+ rescue Errno::EWOULDBLOCK
14
+ sleep 5
15
+ retry
16
+ end
17
+ end
18
+
19
+ # This method causes the current running process to become a daemon
20
+ def daemonize(log_file='/dev/null')
21
+ oldmode = 0
22
+ srand # Split rand streams between spawning and daemonized process
23
+ safefork and exit # Fork and exit from the parent
24
+
25
+ # Detach from the controlling terminal
26
+ unless sess_id = Process.setsid
27
+ raise 'Cannot detach from controlled terminal'
28
+ end
29
+
30
+ # Prevent the possibility of acquiring a controlling terminal
31
+ if oldmode.zero?
32
+ trap 'SIGHUP', 'IGNORE'
33
+ exit if pid = safefork
34
+ end
35
+
36
+ Dir.chdir "/" # Release old working directory
37
+ File.umask 0000 # Ensure sensible umask
38
+
39
+ STDIN.reopen "/dev/null"
40
+ STDOUT.reopen '/dev/null', "a"
41
+ STDERR.reopen log_file
42
+ return oldmode ? sess_id : 0
43
+ end
44
+ end
45
+ end
@@ -0,0 +1,203 @@
1
+ ##
2
+ # EventSocket is a small abstraction of TCPSocket which causes it to behave much like an EventMachine Connection object for
3
+ # the sake of better testability. The EventMachine Connection paradigm (as well as other networking libraries such as the
4
+ # Objective-C HTTP library) uses callbacks to signal different stages of a socket's lifecycle.
5
+ #
6
+ # A handler can be registered in one of two ways: through registrations on an object yielded by the constructor or
7
+ # pre-defined on the object given as a constructor parameter. Below is an example definition which uses the block way:
8
+ #
9
+ # EventSocket.new do |handler|
10
+ # def handler.receive_data(data)
11
+ # # Do something here
12
+ # end
13
+ # def handler.disconnected
14
+ # # Do something here
15
+ # end
16
+ # def handler.connected
17
+ # # Do something here
18
+ # end
19
+ # end
20
+ #
21
+ # Note: this is also a valid way of defining block callbacks:
22
+ #
23
+ # EventSocket.new do |handler|
24
+ # handler.receive_data { |data| do_something }
25
+ # handler.disconnected { do_something }
26
+ # handler.connected { do_something }
27
+ # end
28
+ #
29
+ # and here is an example of using a handler object:
30
+ #
31
+ # class MyCallbackHandler
32
+ # def receive_data(data) end
33
+ # def connected() end
34
+ # def disconnected() end
35
+ # end
36
+ # EventSocket.new(MyCallbackHandler.new)
37
+ #
38
+ # If you wish to ask the EventSocket what state it is in, you can call the Thread-safe EventSocket#state method. The
39
+ # supported states are:
40
+ #
41
+ # - :new
42
+ # - :connected
43
+ # - :stopped
44
+ # - :connection_dropped
45
+ #
46
+ # Note: the EventSocket's state will be changed before these callbacks are executed. For example, if your "connected"
47
+ # callback queried its own EventSocket for its state, it will have already transitioned to the connected() state.
48
+ #
49
+ # Warning: If an exception occurs in your EventSocket callbacks, they will be "eaten" and never bubbled up the call stack.
50
+ # You should always wrap your callbacks in a begin/rescue clause and handle exceptions explicitly.
51
+ #
52
+ require "thread"
53
+ require "socket"
54
+
55
+ class EventSocket
56
+
57
+ class << self
58
+
59
+ ##
60
+ # Creates and returns a connected EventSocket instance.
61
+ #
62
+ def connect(*args, &block)
63
+ instance = new(*args, &block)
64
+ instance.connect!
65
+ instance
66
+ end
67
+ end
68
+
69
+ MAX_CHUNK_SIZE = 256 * 1024
70
+
71
+ def initialize(host, port, handler=nil, &block)
72
+ raise ArgumentError, "Cannot supply both a handler object and a block" if handler && block_given?
73
+ raise ArgumentError, "Must supply either a handler object or a block" if !handler && !block_given?
74
+
75
+ @state_lock = Mutex.new
76
+ @host = host
77
+ @port = port
78
+
79
+ @state = :new
80
+ @handler = handler || new_handler_from_block(&block)
81
+ end
82
+
83
+ def state
84
+ @state_lock.synchronize { @state }
85
+ end
86
+
87
+ def connect!
88
+ @state_lock.synchronize do
89
+ if @state.equal? :connected
90
+ raise ConnectionError, "Already connected!"
91
+ else
92
+ @socket = TCPSocket.new(@host, @port)
93
+ @state = :connected
94
+ end
95
+ end
96
+ @handler.connected rescue nil
97
+ @reader_thread = spawn_reader_thread
98
+ self
99
+ rescue => error
100
+ @state = :failed
101
+ raise error
102
+ end
103
+
104
+ ##
105
+ # Thread-safe implementation of write.
106
+ #
107
+ # @param [String] data Data to write
108
+ #
109
+ def send_data(data)
110
+ # Note: TCPSocket#write is intrinsically Thread-safe
111
+ @socket.write data
112
+ rescue
113
+ connection_dropped!
114
+ end
115
+ ##
116
+ # Disconnects this EventSocket and sets the state to :stopped
117
+ #
118
+ def disconnect!
119
+ @state_lock.synchronize do
120
+ @socket.close rescue nil
121
+ @state = :stopped
122
+ end
123
+ end
124
+
125
+ ##
126
+ # Joins the Thread which reads data off the socket.
127
+ #
128
+ def join
129
+ @state_lock.synchronize do
130
+ if @state.equal? :connected
131
+ @reader_thread.join
132
+ else
133
+ nil
134
+ end
135
+ end
136
+ end
137
+
138
+ def receive_data(data)
139
+ @handler.receive_data(data)
140
+ end
141
+
142
+ protected
143
+
144
+ def connection_dropped!
145
+ @state_lock.synchronize do
146
+ unless @state.equal? :connection_dropped
147
+ @state = :connection_dropped
148
+ @handler.disconnected
149
+ end
150
+ end
151
+ end
152
+
153
+ def spawn_reader_thread
154
+ Thread.new(&method(:reader_loop))
155
+ end
156
+
157
+ def reader_loop
158
+ until state.equal? :stopped
159
+ data = @socket.readpartial(MAX_CHUNK_SIZE)
160
+ @handler.receive_data data
161
+ end
162
+ rescue EOFError
163
+ connection_dropped!
164
+ end
165
+
166
+ def new_handler_from_block(&handler_block)
167
+ handler = Object.new
168
+ handler.metaclass.send :attr_accessor, :set_callbacks
169
+ handler.set_callbacks = {:receive_data => false, :disconnected => false, :connected => false }
170
+
171
+ def handler.receive_data(&block)
172
+ self.metaclass.send(:remove_method, :receive_data)
173
+ self.metaclass.send(:define_method, :receive_data) { |data| block.call data }
174
+ set_callbacks[:receive_data] = true
175
+ end
176
+ def handler.connected(&block)
177
+ self.metaclass.send(:remove_method, :connected)
178
+ self.metaclass.send(:define_method, :connected) { block.call }
179
+ set_callbacks[:connected] = true
180
+ end
181
+ def handler.disconnected(&block)
182
+ self.metaclass.send(:remove_method, :disconnected)
183
+ self.metaclass.send(:define_method, :disconnected) { block.call }
184
+ set_callbacks[:disconnected] = true
185
+ end
186
+
187
+ def handler.singleton_method_added(name)
188
+ set_callbacks[name.to_sym] = true
189
+ end
190
+
191
+ yield handler
192
+
193
+ handler.set_callbacks.each_pair do |callback_name,was_set|
194
+ handler.send(callback_name) {} unless was_set
195
+ end
196
+
197
+ handler
198
+
199
+ end
200
+
201
+ class ConnectionError < Exception; end
202
+
203
+ end
@@ -0,0 +1,36 @@
1
+ require "thread"
2
+
3
+ class FutureResource
4
+
5
+ def initialize
6
+ @resource_lock = Monitor.new
7
+ @resource_value_blocker = @resource_lock.new_cond
8
+ end
9
+
10
+ def set_yet?
11
+ @resource_lock.synchronize { defined? @resource }
12
+ end
13
+
14
+ def resource
15
+ @resource_lock.synchronize do
16
+ @resource_value_blocker.wait unless defined? @resource
17
+ @resource
18
+ end
19
+ end
20
+
21
+ def resource=(resource)
22
+ @resource_lock.synchronize do
23
+ raise ResourceAlreadySetException if defined? @resource
24
+ @resource = resource
25
+ @resource_value_blocker.broadcast
26
+ @resource_value_blocker = nil # Don't really need it anymore.
27
+ end
28
+ end
29
+
30
+ class ResourceAlreadySetException < Exception
31
+ def initialize
32
+ super "Cannot set this resource twice!"
33
+ end
34
+ end
35
+
36
+ end