rene-adhearsion 0.8.6
Sign up to get free protection for your applications and to get access to all the features.
- data/CHANGELOG +73 -0
- data/EVENTS +11 -0
- data/LICENSE +456 -0
- data/Rakefile +130 -0
- data/adhearsion.gemspec +173 -0
- data/app_generators/ahn/USAGE +5 -0
- data/app_generators/ahn/ahn_generator.rb +96 -0
- data/app_generators/ahn/templates/.ahnrc +34 -0
- data/app_generators/ahn/templates/README +8 -0
- data/app_generators/ahn/templates/Rakefile +25 -0
- data/app_generators/ahn/templates/components/ami_remote/ami_remote.rb +15 -0
- data/app_generators/ahn/templates/components/disabled/HOW_TO_ENABLE +7 -0
- data/app_generators/ahn/templates/components/disabled/restful_rpc/README.markdown +11 -0
- data/app_generators/ahn/templates/components/disabled/restful_rpc/example-client.rb +48 -0
- data/app_generators/ahn/templates/components/disabled/restful_rpc/restful_rpc.rb +91 -0
- data/app_generators/ahn/templates/components/disabled/restful_rpc/restful_rpc.yml +34 -0
- data/app_generators/ahn/templates/components/disabled/restful_rpc/spec/restful_rpc_spec.rb +263 -0
- data/app_generators/ahn/templates/components/disabled/sandbox/sandbox.rb +104 -0
- data/app_generators/ahn/templates/components/disabled/sandbox/sandbox.yml +2 -0
- data/app_generators/ahn/templates/components/disabled/stomp_gateway/README.markdown +47 -0
- data/app_generators/ahn/templates/components/disabled/stomp_gateway/stomp_gateway.rb +34 -0
- data/app_generators/ahn/templates/components/disabled/stomp_gateway/stomp_gateway.yml +12 -0
- data/app_generators/ahn/templates/components/disabled/xmpp_gateway/README.markdown +3 -0
- data/app_generators/ahn/templates/components/disabled/xmpp_gateway/xmpp_gateway.rb +11 -0
- data/app_generators/ahn/templates/components/disabled/xmpp_gateway/xmpp_gateway.yml +0 -0
- data/app_generators/ahn/templates/components/simon_game/simon_game.rb +56 -0
- data/app_generators/ahn/templates/config/startup.rb +83 -0
- data/app_generators/ahn/templates/dialplan.rb +3 -0
- data/app_generators/ahn/templates/events.rb +32 -0
- data/bin/ahn +28 -0
- data/bin/ahnctl +68 -0
- data/bin/jahn +42 -0
- data/examples/asterisk_manager_interface/standalone.rb +51 -0
- data/lib/adhearsion.rb +45 -0
- data/lib/adhearsion/cli.rb +228 -0
- data/lib/adhearsion/component_manager.rb +272 -0
- data/lib/adhearsion/component_manager/component_tester.rb +55 -0
- data/lib/adhearsion/component_manager/spec_framework.rb +24 -0
- data/lib/adhearsion/events_support.rb +84 -0
- data/lib/adhearsion/foundation/all.rb +15 -0
- data/lib/adhearsion/foundation/blank_slate.rb +3 -0
- data/lib/adhearsion/foundation/custom_daemonizer.rb +45 -0
- data/lib/adhearsion/foundation/event_socket.rb +204 -0
- data/lib/adhearsion/foundation/future_resource.rb +36 -0
- data/lib/adhearsion/foundation/metaprogramming.rb +17 -0
- data/lib/adhearsion/foundation/numeric.rb +13 -0
- data/lib/adhearsion/foundation/pseudo_guid.rb +10 -0
- data/lib/adhearsion/foundation/relationship_properties.rb +42 -0
- data/lib/adhearsion/foundation/string.rb +26 -0
- data/lib/adhearsion/foundation/synchronized_hash.rb +96 -0
- data/lib/adhearsion/foundation/thread_safety.rb +7 -0
- data/lib/adhearsion/host_definitions.rb +67 -0
- data/lib/adhearsion/initializer.rb +395 -0
- data/lib/adhearsion/initializer/asterisk.rb +87 -0
- data/lib/adhearsion/initializer/configuration.rb +321 -0
- data/lib/adhearsion/initializer/database.rb +60 -0
- data/lib/adhearsion/initializer/drb.rb +31 -0
- data/lib/adhearsion/initializer/freeswitch.rb +22 -0
- data/lib/adhearsion/initializer/ldap.rb +57 -0
- data/lib/adhearsion/initializer/rails.rb +41 -0
- data/lib/adhearsion/initializer/xmpp.rb +42 -0
- data/lib/adhearsion/logging.rb +92 -0
- data/lib/adhearsion/tasks.rb +16 -0
- data/lib/adhearsion/tasks/database.rb +5 -0
- data/lib/adhearsion/tasks/deprecations.rb +59 -0
- data/lib/adhearsion/tasks/generating.rb +20 -0
- data/lib/adhearsion/tasks/lint.rb +4 -0
- data/lib/adhearsion/tasks/testing.rb +37 -0
- data/lib/adhearsion/version.rb +33 -0
- data/lib/adhearsion/voip/asterisk.rb +4 -0
- data/lib/adhearsion/voip/asterisk/agi_server.rb +115 -0
- data/lib/adhearsion/voip/asterisk/commands.rb +1510 -0
- data/lib/adhearsion/voip/asterisk/config_generators/agents.conf.rb +140 -0
- data/lib/adhearsion/voip/asterisk/config_generators/config_generator.rb +101 -0
- data/lib/adhearsion/voip/asterisk/config_generators/queues.conf.rb +250 -0
- data/lib/adhearsion/voip/asterisk/config_generators/voicemail.conf.rb +240 -0
- data/lib/adhearsion/voip/asterisk/config_manager.rb +71 -0
- data/lib/adhearsion/voip/asterisk/manager_interface.rb +705 -0
- data/lib/adhearsion/voip/asterisk/manager_interface/ami_lexer.rb +1680 -0
- data/lib/adhearsion/voip/asterisk/manager_interface/ami_lexer.rl.rb +340 -0
- data/lib/adhearsion/voip/asterisk/manager_interface/ami_messages.rb +78 -0
- data/lib/adhearsion/voip/asterisk/manager_interface/ami_protocol_lexer_machine.rl +87 -0
- data/lib/adhearsion/voip/asterisk/special_dial_plan_managers.rb +80 -0
- data/lib/adhearsion/voip/asterisk/super_manager.rb +19 -0
- data/lib/adhearsion/voip/call.rb +497 -0
- data/lib/adhearsion/voip/call_routing.rb +64 -0
- data/lib/adhearsion/voip/commands.rb +9 -0
- data/lib/adhearsion/voip/constants.rb +39 -0
- data/lib/adhearsion/voip/conveniences.rb +18 -0
- data/lib/adhearsion/voip/dial_plan.rb +246 -0
- data/lib/adhearsion/voip/dsl/dialing_dsl.rb +151 -0
- data/lib/adhearsion/voip/dsl/dialing_dsl/dialing_dsl_monkey_patches.rb +37 -0
- data/lib/adhearsion/voip/dsl/dialplan/control_passing_exception.rb +27 -0
- data/lib/adhearsion/voip/dsl/dialplan/dispatcher.rb +124 -0
- data/lib/adhearsion/voip/dsl/dialplan/parser.rb +69 -0
- data/lib/adhearsion/voip/dsl/dialplan/thread_mixin.rb +16 -0
- data/lib/adhearsion/voip/dsl/numerical_string.rb +115 -0
- data/lib/adhearsion/voip/freeswitch/basic_connection_manager.rb +48 -0
- data/lib/adhearsion/voip/freeswitch/event_handler.rb +58 -0
- data/lib/adhearsion/voip/freeswitch/freeswitch_dialplan_command_factory.rb +129 -0
- data/lib/adhearsion/voip/freeswitch/inbound_connection_manager.rb +38 -0
- data/lib/adhearsion/voip/freeswitch/oes_server.rb +195 -0
- data/lib/adhearsion/voip/menu_state_machine/calculated_match.rb +80 -0
- data/lib/adhearsion/voip/menu_state_machine/matchers.rb +123 -0
- data/lib/adhearsion/voip/menu_state_machine/menu_builder.rb +58 -0
- data/lib/adhearsion/voip/menu_state_machine/menu_class.rb +149 -0
- data/lib/adhearsion/xmpp/connection.rb +61 -0
- data/lib/theatre.rb +151 -0
- data/lib/theatre/README.markdown +64 -0
- data/lib/theatre/callback_definition_loader.rb +84 -0
- data/lib/theatre/guid.rb +23 -0
- data/lib/theatre/invocation.rb +121 -0
- data/lib/theatre/namespace_manager.rb +153 -0
- data/lib/theatre/version.rb +2 -0
- metadata +241 -0
@@ -0,0 +1,58 @@
|
|
1
|
+
require File.join(File.dirname(__FILE__), 'matchers.rb')
|
2
|
+
|
3
|
+
module Adhearsion
|
4
|
+
module VoIP
|
5
|
+
class MenuBuilder
|
6
|
+
|
7
|
+
def initialize
|
8
|
+
@patterns = []
|
9
|
+
@menu_callbacks = {}
|
10
|
+
end
|
11
|
+
|
12
|
+
def method_missing(match_payload, *patterns, &block)
|
13
|
+
name_string = match_payload.to_s
|
14
|
+
if patterns.any?
|
15
|
+
patterns.each do |pattern|
|
16
|
+
@patterns << MatchCalculator.build_with_pattern(pattern, match_payload)
|
17
|
+
end
|
18
|
+
else
|
19
|
+
raise ArgumentError, "You cannot call this method without patterns!"
|
20
|
+
end
|
21
|
+
nil
|
22
|
+
end
|
23
|
+
|
24
|
+
def weighted_match_calculators
|
25
|
+
@patterns
|
26
|
+
end
|
27
|
+
|
28
|
+
def execute_hook_for(symbol, input)
|
29
|
+
callback = @menu_callbacks[symbol]
|
30
|
+
callback.call input if callback
|
31
|
+
end
|
32
|
+
|
33
|
+
def on_invalid(&block)
|
34
|
+
raise LocalJumpError, "Must supply a block!" unless block_given?
|
35
|
+
@menu_callbacks[:invalid] = block
|
36
|
+
end
|
37
|
+
|
38
|
+
def on_premature_timeout(&block)
|
39
|
+
raise LocalJumpError, "Must supply a block!" unless block_given?
|
40
|
+
@menu_callbacks[:premature_timeout] = block
|
41
|
+
end
|
42
|
+
|
43
|
+
def on_failure(&block)
|
44
|
+
raise LocalJumpError, "Must supply a block!" unless block_given?
|
45
|
+
@menu_callbacks[:failure] = block
|
46
|
+
end
|
47
|
+
|
48
|
+
def calculate_matches_for(result)
|
49
|
+
CalculatedMatchCollection.new.tap do |collection|
|
50
|
+
weighted_match_calculators.each do |pattern|
|
51
|
+
collection << pattern.match(result)
|
52
|
+
end
|
53
|
+
end
|
54
|
+
end
|
55
|
+
|
56
|
+
end
|
57
|
+
end
|
58
|
+
end
|
@@ -0,0 +1,149 @@
|
|
1
|
+
require 'adhearsion/voip/menu_state_machine/menu_builder'
|
2
|
+
module Adhearsion
|
3
|
+
module VoIP
|
4
|
+
class Menu
|
5
|
+
|
6
|
+
DEFAULT_MAX_NUMBER_OF_TRIES = 1
|
7
|
+
DEFAULT_TIMEOUT = 5 # seconds
|
8
|
+
|
9
|
+
relationships :menu_builder => MenuBuilder
|
10
|
+
|
11
|
+
attr_reader :builder, :timeout, :tries_count, :max_number_of_tries
|
12
|
+
def initialize(options={}, &block)
|
13
|
+
@tries_count = 0 # Counts the number of tries the menu's been executed
|
14
|
+
|
15
|
+
@timeout = options[:timeout] || DEFAULT_TIMEOUT
|
16
|
+
@max_number_of_tries = options[:tries] || DEFAULT_MAX_NUMBER_OF_TRIES
|
17
|
+
|
18
|
+
@builder = menu_builder.new
|
19
|
+
yield @builder
|
20
|
+
|
21
|
+
initialize_digit_buffer
|
22
|
+
end
|
23
|
+
|
24
|
+
def <<(other)
|
25
|
+
digit_buffer << other
|
26
|
+
end
|
27
|
+
|
28
|
+
def digit_buffer
|
29
|
+
@digit_buffer
|
30
|
+
end
|
31
|
+
|
32
|
+
def digit_buffer_string
|
33
|
+
digit_buffer.to_s
|
34
|
+
end
|
35
|
+
|
36
|
+
def digit_buffer_empty?
|
37
|
+
digit_buffer.empty?
|
38
|
+
end
|
39
|
+
|
40
|
+
def continue
|
41
|
+
raise MenuGetAnotherDigitOrTimeout if digit_buffer_empty?
|
42
|
+
|
43
|
+
calculated_matches = builder.calculate_matches_for digit_buffer_string
|
44
|
+
|
45
|
+
if calculated_matches.exact_match_count >= 1
|
46
|
+
first_exact_match = calculated_matches.exact_matches.first
|
47
|
+
if calculated_matches.potential_match_count.zero?
|
48
|
+
# Match found with no extenuating ambiguities! Go with the first exact match
|
49
|
+
menu_result_found! first_exact_match.match_payload, digit_buffer_string
|
50
|
+
else
|
51
|
+
get_another_digit_or_finish!(first_exact_match.match_payload, first_exact_match.query)
|
52
|
+
end
|
53
|
+
elsif calculated_matches.potential_match_count >= 1
|
54
|
+
get_another_digit_or_timeout!
|
55
|
+
else
|
56
|
+
invalid!
|
57
|
+
end
|
58
|
+
end
|
59
|
+
|
60
|
+
def should_continue?
|
61
|
+
tries_count < max_number_of_tries
|
62
|
+
end
|
63
|
+
|
64
|
+
def restart!
|
65
|
+
@tries_count += 1
|
66
|
+
digit_buffer.clear!
|
67
|
+
end
|
68
|
+
|
69
|
+
def execute_invalid_hook
|
70
|
+
builder.execute_hook_for(:invalid, digit_buffer_string)
|
71
|
+
end
|
72
|
+
|
73
|
+
def execute_timeout_hook
|
74
|
+
builder.execute_hook_for(:premature_timeout, digit_buffer_string)
|
75
|
+
end
|
76
|
+
|
77
|
+
def execute_failure_hook
|
78
|
+
builder.execute_hook_for(:failure, digit_buffer_string)
|
79
|
+
end
|
80
|
+
|
81
|
+
protected
|
82
|
+
|
83
|
+
# If you're using a more complex class in subclasses, you may want to override this method in addition to the
|
84
|
+
# digit_buffer, digit_buffer_empty, and digit_buffer_string methods
|
85
|
+
def initialize_digit_buffer
|
86
|
+
@digit_buffer = ClearableStringBuffer.new
|
87
|
+
end
|
88
|
+
|
89
|
+
def invalid!
|
90
|
+
raise MenuResultInvalid
|
91
|
+
end
|
92
|
+
|
93
|
+
def menu_result_found!(match_payload, new_extension)
|
94
|
+
raise MenuResultFound.new(match_payload, new_extension)
|
95
|
+
end
|
96
|
+
|
97
|
+
def get_another_digit_or_finish!(match_payload, new_extension)
|
98
|
+
raise MenuGetAnotherDigitOrFinish.new(match_payload, new_extension)
|
99
|
+
end
|
100
|
+
|
101
|
+
def get_another_digit_or_timeout!
|
102
|
+
raise MenuGetAnotherDigitOrTimeout
|
103
|
+
end
|
104
|
+
|
105
|
+
# The superclass from which all message-like exceptions descend. It should never
|
106
|
+
# be instantiated directly.
|
107
|
+
class MenuResult < StandardError; end
|
108
|
+
|
109
|
+
# Raised when the user's input matches
|
110
|
+
class MenuResultFound < MenuResult
|
111
|
+
attr_reader :match_payload, :new_extension
|
112
|
+
def initialize(match_payload, new_extension)
|
113
|
+
super()
|
114
|
+
@match_payload = match_payload
|
115
|
+
@new_extension = new_extension
|
116
|
+
end
|
117
|
+
end
|
118
|
+
|
119
|
+
module MenuGetAnotherDigit; end
|
120
|
+
|
121
|
+
class MenuGetAnotherDigitOrFinish < MenuResultFound
|
122
|
+
include MenuGetAnotherDigit
|
123
|
+
end
|
124
|
+
|
125
|
+
class MenuGetAnotherDigitOrTimeout < MenuResult
|
126
|
+
include MenuGetAnotherDigit
|
127
|
+
end
|
128
|
+
|
129
|
+
class MenuResultFound < MenuResult; end
|
130
|
+
|
131
|
+
# Raised when the user's input matches no patterns
|
132
|
+
class MenuResultInvalid < MenuResult; end
|
133
|
+
|
134
|
+
# For our default purposes, we need the digit_buffer to behave much like a normal String except that it should
|
135
|
+
# handle its own resetting (clearing).
|
136
|
+
class ClearableStringBuffer < String
|
137
|
+
def clear!
|
138
|
+
replace ""
|
139
|
+
end
|
140
|
+
|
141
|
+
def <<(other)
|
142
|
+
super other.to_s
|
143
|
+
end
|
144
|
+
|
145
|
+
end
|
146
|
+
|
147
|
+
end
|
148
|
+
end
|
149
|
+
end
|
@@ -0,0 +1,61 @@
|
|
1
|
+
module Adhearsion
|
2
|
+
module XMPP
|
3
|
+
module Connection
|
4
|
+
|
5
|
+
mattr_accessor :client
|
6
|
+
class << self
|
7
|
+
|
8
|
+
# Open the XMPP connection
|
9
|
+
#
|
10
|
+
# @param [String] jid the client/component JID to connect to
|
11
|
+
# @param [String] password
|
12
|
+
# @param [String] server
|
13
|
+
# @param [Integer] port
|
14
|
+
def start(jid, password, server, port)
|
15
|
+
Blather.logger = ahn_log.xmpp
|
16
|
+
setup_client_object(jid, password, server, port)
|
17
|
+
register_event_namespaces
|
18
|
+
register_default_client_handlers
|
19
|
+
Events.register_callback(:after_initialized) do
|
20
|
+
connect
|
21
|
+
end
|
22
|
+
end
|
23
|
+
|
24
|
+
# Close the XMPP connection
|
25
|
+
def stop
|
26
|
+
shutdown
|
27
|
+
end
|
28
|
+
|
29
|
+
private
|
30
|
+
|
31
|
+
def setup_client_object(jid, password, server, port)
|
32
|
+
self.client = Blather::Client.setup(jid, password, server, port)
|
33
|
+
end
|
34
|
+
|
35
|
+
def connect
|
36
|
+
EventMachine.run {client.connect}
|
37
|
+
end
|
38
|
+
|
39
|
+
def register_event_namespaces
|
40
|
+
Events.register_namespace_name "/xmpp"
|
41
|
+
end
|
42
|
+
|
43
|
+
def register_default_client_handlers
|
44
|
+
client.register_handler(:ready) do
|
45
|
+
ahn_log.xmpp.info "Connected to XMPP server! Send messages to #{client.jid.stripped}."
|
46
|
+
end
|
47
|
+
|
48
|
+
client.register_handler(:disconnected) do
|
49
|
+
if Adhearsion.status == :running
|
50
|
+
ahn_log.xmpp.warning "XMPP Disconnected. Reconnecting."
|
51
|
+
connect
|
52
|
+
end
|
53
|
+
# TODO: fix this to reconnect XMPP cleanly
|
54
|
+
end
|
55
|
+
end
|
56
|
+
|
57
|
+
end
|
58
|
+
|
59
|
+
end
|
60
|
+
end
|
61
|
+
end
|
data/lib/theatre.rb
ADDED
@@ -0,0 +1,151 @@
|
|
1
|
+
require 'thread'
|
2
|
+
require 'rubygems'
|
3
|
+
|
4
|
+
$: << File.expand_path(File.dirname(__FILE__))
|
5
|
+
|
6
|
+
require 'theatre/version'
|
7
|
+
require 'theatre/namespace_manager'
|
8
|
+
require 'theatre/invocation'
|
9
|
+
require 'theatre/callback_definition_loader'
|
10
|
+
|
11
|
+
module Theatre
|
12
|
+
|
13
|
+
class Theatre
|
14
|
+
|
15
|
+
attr_reader :namespace_manager
|
16
|
+
|
17
|
+
##
|
18
|
+
# Creates a new stopped Theatre. You must call start!() after you instantiate this for it to begin processing events.
|
19
|
+
#
|
20
|
+
# @param [Fixnum] thread_count Number of Threads to spawn when started.
|
21
|
+
#
|
22
|
+
def initialize(thread_count=6)
|
23
|
+
@thread_count = thread_count
|
24
|
+
@started = false
|
25
|
+
@namespace_manager = ActorNamespaceManager.new
|
26
|
+
@thread_group = ThreadGroup.new
|
27
|
+
@master_queue = Queue.new
|
28
|
+
@loader_mixins = []
|
29
|
+
end
|
30
|
+
|
31
|
+
##
|
32
|
+
# Send a message to this Theatre for asynchronous processing.
|
33
|
+
#
|
34
|
+
# @param [String] namespace The namespace to which the payload should be sent
|
35
|
+
# @param [Object] payload The actual content to be sent to the callback. Optional.
|
36
|
+
# @return [Array<Theatre::Invocation>] An Array of Invocation objects
|
37
|
+
# @raise Theatre::NamespaceNotFound Raised when told to enqueue an unrecognized namespace
|
38
|
+
#
|
39
|
+
def trigger(namespace, payload=:argument_undefined)
|
40
|
+
@namespace_manager.callbacks_for_namespaces(namespace).map do |callback|
|
41
|
+
invocation = if payload.equal?(:argument_undefined)
|
42
|
+
Invocation.new(namespace, callback)
|
43
|
+
else
|
44
|
+
Invocation.new(namespace, callback, payload)
|
45
|
+
end
|
46
|
+
invocation.queued
|
47
|
+
@master_queue << invocation
|
48
|
+
invocation
|
49
|
+
end
|
50
|
+
end
|
51
|
+
|
52
|
+
##
|
53
|
+
# Send a message to this Theatre for synchronous processing. The execution of this will not go through this Theatre's
|
54
|
+
# Thread pool. If an error occurred in any of callbacks, the Exception object will be placed in the returned Array
|
55
|
+
# instead for you to act upon.
|
56
|
+
#
|
57
|
+
# @param [String] namespace The namespace to which the payload should be sent
|
58
|
+
# @param [Object] payload The actual content to be sent to the callback. Optional.
|
59
|
+
# @return [Array] An Array containing each callback's return value (or Exception raised, if any) when given the payload
|
60
|
+
# @raise Theatre::NamespaceNotFound Raised when told to enqueue an unrecognized namespace
|
61
|
+
#
|
62
|
+
def trigger_immediately(namespace, payload=:argument_undefined)
|
63
|
+
@namespace_manager.callbacks_for_namespaces(namespace).map do |callback|
|
64
|
+
begin
|
65
|
+
invocation = if payload.equal?(:argument_undefined)
|
66
|
+
callback.call
|
67
|
+
else
|
68
|
+
callback.call payload
|
69
|
+
end
|
70
|
+
rescue => captured_error_to_be_returned
|
71
|
+
captured_error_to_be_returned
|
72
|
+
end
|
73
|
+
end
|
74
|
+
end
|
75
|
+
|
76
|
+
def load_events_code(code, *args)
|
77
|
+
loader = CallbackDefinitionLoader.new(self, *args)
|
78
|
+
loader.load_events_code code
|
79
|
+
end
|
80
|
+
|
81
|
+
def load_events_file(file, *args)
|
82
|
+
loader = CallbackDefinitionLoader.new(self, *args)
|
83
|
+
loader.load_events_file file
|
84
|
+
end
|
85
|
+
|
86
|
+
def register_namespace_name(*args)
|
87
|
+
@namespace_manager.register_namespace_name(*args)
|
88
|
+
end
|
89
|
+
|
90
|
+
def register_callback_at_namespace(*args)
|
91
|
+
@namespace_manager.register_callback_at_namespace(*args)
|
92
|
+
end
|
93
|
+
|
94
|
+
def register_loader_mixin(mod)
|
95
|
+
@loader_mixins << mod
|
96
|
+
end
|
97
|
+
|
98
|
+
def join
|
99
|
+
@thread_group.list.each do |thread|
|
100
|
+
begin
|
101
|
+
thread.join
|
102
|
+
rescue
|
103
|
+
# Ignore any exceptions
|
104
|
+
end
|
105
|
+
end
|
106
|
+
end
|
107
|
+
|
108
|
+
##
|
109
|
+
# Starts this Theatre.
|
110
|
+
#
|
111
|
+
# When this method is called, the Threads are spawned and begin pulling messages off this Theatre's master queue.
|
112
|
+
#
|
113
|
+
def start!
|
114
|
+
return false if @thread_group.list.any? # Already started
|
115
|
+
@started_time = Time.now
|
116
|
+
@thread_count.times do
|
117
|
+
@thread_group.add Thread.new(&method(:thread_loop))
|
118
|
+
end
|
119
|
+
end
|
120
|
+
|
121
|
+
##
|
122
|
+
# Notifies all Threads for this Theatre to stop by sending them special messages. Any messages which were queued and
|
123
|
+
# untriggered when this method is received will still be processed. Note: you may start this Theatre again later once it
|
124
|
+
# has been stopped.
|
125
|
+
#
|
126
|
+
def graceful_stop!
|
127
|
+
@thread_count.times { @master_queue << :THEATRE_SHUTDOWN! }
|
128
|
+
@started_time = nil
|
129
|
+
end
|
130
|
+
|
131
|
+
protected
|
132
|
+
|
133
|
+
# This will use the Adhearsion logger eventually.
|
134
|
+
def warn(exception)
|
135
|
+
# STDERR.puts exception.message, *exception.backtrace
|
136
|
+
end
|
137
|
+
|
138
|
+
def thread_loop
|
139
|
+
loop do
|
140
|
+
begin
|
141
|
+
next_invocation = @master_queue.pop
|
142
|
+
return :stopped if next_invocation.equal? :THEATRE_SHUTDOWN!
|
143
|
+
next_invocation.start
|
144
|
+
rescue => error
|
145
|
+
warn error
|
146
|
+
end
|
147
|
+
end
|
148
|
+
end
|
149
|
+
|
150
|
+
end
|
151
|
+
end
|
@@ -0,0 +1,64 @@
|
|
1
|
+
Theatre
|
2
|
+
=======
|
3
|
+
|
4
|
+
Present status: stable
|
5
|
+
|
6
|
+
A library for choreographing a dynamic pool of hierarchically organized actors. This was originally extracted from the [Adhearsion](http://adhearsion.com) framework by Jay Phillips.
|
7
|
+
|
8
|
+
In the Adhearsion framework, it was necessary to develop an internal message-passing system that could work either synchronously or asynchronously. This is used by the framework itself and for framework extensions (called _components_) to talk with each other. The source of the events is completely undefined -- events could originate from within the framework out outside the framework. For example, a Message Queue such as [Stomp](http://stomp.codehaus.org) can wire incoming events into Theatre and catch events going to a particular destination so it can proxy them out to the server.
|
9
|
+
|
10
|
+
Motivations and Design Decisions
|
11
|
+
--------------------------------
|
12
|
+
|
13
|
+
* Must maintain Ruby 1.8 and JRuby compatibility
|
14
|
+
* Must be Thread-safe
|
15
|
+
* Must provide some level of transparency into the events going through it
|
16
|
+
* Must be dynamic enough to reallocate the number of triggerrs based on load
|
17
|
+
* Must help facilitate test-driven development of Actor functionality
|
18
|
+
* Must allow external persistence in case of a crash
|
19
|
+
|
20
|
+
Example
|
21
|
+
-------
|
22
|
+
|
23
|
+
Below is an example taken from Adhearsion for executing framework-level callbacks. Note: the framework treats this callback synchronously.
|
24
|
+
|
25
|
+
events.framework.asterisk.before_call.each do |event|
|
26
|
+
# Pull headers from event and respond to it here.
|
27
|
+
end
|
28
|
+
|
29
|
+
Below is an example of integration with [Stomp](http://stomp.codehaus.org/), a simple yet robust open-protocol message queue.
|
30
|
+
|
31
|
+
events.stomp.new_call.each do |event|
|
32
|
+
# Handle all events from the Stomp MQ server whose name is "new_call" (the String)
|
33
|
+
end
|
34
|
+
|
35
|
+
This will filter all events whose name is "new_call" and yield the Stomp::Message to the block.
|
36
|
+
|
37
|
+
Framework terminology
|
38
|
+
--------------------
|
39
|
+
|
40
|
+
Below are definitions of terms I use in Theatre. See the respective links for more information.
|
41
|
+
|
42
|
+
* **callback**: This is the block given to the `each` method which triggers events coming in.
|
43
|
+
* **payload**: This is the "message" sent to the Theatre and is what will ultimately be yielded to the callback
|
44
|
+
* **[Actor](http://en.wikipedia.org/wiki/Actor_model)**: This refers to concurrent responders to events in a concurrent system.
|
45
|
+
|
46
|
+
Synchronous vs. Asynchronous
|
47
|
+
----------------------------
|
48
|
+
|
49
|
+
With Theatre, all events are asynchronous with the optional ability to synchronously block until the event is scheduled, triggered, and has returned. If you wish to synchronously trigger the event, simple call `wait` on an `Invocation` object returned from `trigger` and then check the `Invocation#current_state` property for `:success` or `:error`. Optionally the `Invocation#success?` and `Invocation#error?` methods also provide more intuitive access to the finished state. If the event finished with `:success`, you may retrieve the returned value of the event Proc by calling `Invocation#returned_value`. If the event finished with `:error`, you may get the Exception with `Invocation#error`.
|
50
|
+
|
51
|
+
Because many callbacks can be registered for a particular namespace, each needing its own Invocation object, the `Theatre#trigger` method returns an Array of Invocation objects.
|
52
|
+
|
53
|
+
# We'll assume only one callback is registered and call #first on the Array of Invocations returned by #trigger
|
54
|
+
invocation = my_theatre.trigger("your/namespace/here", YourSpecialClass.new("this can be anything")).first
|
55
|
+
invocation.wait
|
56
|
+
raise invocation.error if invocation.error?
|
57
|
+
log "Actor finished with return value #{invocation.returned_value}"
|
58
|
+
|
59
|
+
Ruby 1.8 vs. Ruby 1.9 vs. JRuby
|
60
|
+
-------------------------------
|
61
|
+
|
62
|
+
Theatre was created for Ruby 1.8 because no good Actor system existed on Ruby 1.8 that met Adhearsion's needs (e.g. hierarchal with synchronous and asynchronous modes. If you wish to achieve real processor-level concurrency, use JRuby.
|
63
|
+
|
64
|
+
Presently Ruby 1.9 compatibility is not a priority but patches for compatibility will be accepted, as long as they preserve compatibility with both MRI and JRuby.
|