blob: 300ec842a94d7e0cb442b699a52b81b4fff0b310 (
plain) (
blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
|
module ActiveSupport
module Notifications
# This is a default queue implementation that ships with Notifications. It
# just pushes events to all registered log subscribers.
class Fanout
def initialize
@subscribers = []
@listeners_for = {}
end
def bind(pattern)
Binding.new(self, pattern)
end
def subscribe(pattern = nil, &block)
@listeners_for.clear
@subscribers << Subscriber.new(pattern, &block)
@subscribers.last
end
def unsubscribe(subscriber)
@listeners_for.clear
@subscribers.reject! {|s| s.matches?(subscriber)}
end
def publish(name, *args)
if listeners = @listeners_for[name]
listeners.each { |s| s.publish(name, *args) }
else
@listeners_for[name] = @subscribers.select { |s| s.publish(name, *args) }
end
end
# This is a sync queue, so there is not waiting.
def wait
end
# Used for internal implementation only.
class Binding #:nodoc:
def initialize(queue, pattern)
@queue = queue
@pattern =
case pattern
when Regexp, NilClass
pattern
else
/^#{Regexp.escape(pattern.to_s)}$/
end
end
def subscribe(&block)
@queue.subscribe(@pattern, &block)
end
end
class Subscriber #:nodoc:
def initialize(pattern, &block)
@pattern = pattern
@block = block
end
def publish(*args)
return unless subscribed_to?(args.first)
push(*args)
true
end
def drained?
true
end
def subscribed_to?(name)
!@pattern || @pattern =~ name.to_s
end
def matches?(subscriber_or_name)
case subscriber_or_name
when String
@pattern && @pattern =~ subscriber_or_name
when self
true
end
end
private
def push(*args)
@block.call(*args)
end
end
end
end
end
|