aboutsummaryrefslogblamecommitdiffstats
path: root/activesupport/lib/active_support/notifications/fanout.rb
blob: 300ec842a94d7e0cb442b699a52b81b4fff0b310 (plain) (tree)
1
2
3
4
5
6
7
8


                                                                              
                                                           
                
                    
                         
                           






                                          
                            




                                                       
                            
                                                         

         





                                                                                   

         
                                                      
              




                                              





                                 
                                                
               






                                            
                               
                                       
                            
                        

           
                          
                                                  

                     

           



                    









                                                      
             


               

                         

                              
         


       
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