mtrudel-adhearsion 0.8.3
Sign up to get free protection for your applications and to get access to all the features.
- data/CHANGELOG +26 -0
- data/EVENTS +11 -0
- data/LICENSE +456 -0
- data/Rakefile +127 -0
- data/adhearsion.gemspec +149 -0
- data/app_generators/ahn/USAGE +5 -0
- data/app_generators/ahn/ahn_generator.rb +91 -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 +87 -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/simon_game/simon_game.rb +56 -0
- data/app_generators/ahn/templates/config/startup.rb +50 -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 +37 -0
- data/lib/adhearsion/cli.rb +223 -0
- data/lib/adhearsion/component_manager.rb +207 -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 +9 -0
- data/lib/adhearsion/foundation/blank_slate.rb +5 -0
- data/lib/adhearsion/foundation/custom_daemonizer.rb +45 -0
- data/lib/adhearsion/foundation/event_socket.rb +203 -0
- data/lib/adhearsion/foundation/future_resource.rb +36 -0
- data/lib/adhearsion/foundation/global.rb +1 -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 +373 -0
- data/lib/adhearsion/initializer/asterisk.rb +81 -0
- data/lib/adhearsion/initializer/configuration.rb +254 -0
- data/lib/adhearsion/initializer/database.rb +50 -0
- data/lib/adhearsion/initializer/drb.rb +31 -0
- data/lib/adhearsion/initializer/freeswitch.rb +22 -0
- data/lib/adhearsion/initializer/rails.rb +41 -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 +9 -0
- data/lib/adhearsion/voip/asterisk.rb +4 -0
- data/lib/adhearsion/voip/asterisk/agi_server.rb +84 -0
- data/lib/adhearsion/voip/asterisk/commands.rb +1314 -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 +597 -0
- data/lib/adhearsion/voip/asterisk/manager_interface/ami_lexer.rb +1589 -0
- data/lib/adhearsion/voip/asterisk/manager_interface/ami_lexer.rl.rb +286 -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 +453 -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 +218 -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 +71 -0
- data/lib/adhearsion/voip/dsl/dialplan/thread_mixin.rb +16 -0
- data/lib/adhearsion/voip/dsl/numerical_string.rb +117 -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/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 +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,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
|