Sha256: ff5582db79f2455a9f9247eac9626e594615f22b55c3eff1a66dfec0295adc4e

Contents?: true

Size: 1.46 KB

Versions: 6

Compression:

Stored size: 1.46 KB

Contents

require 'set'

module Liquid

  # Strainer is the parent class for the filters system.
  # New filters are mixed into the strainer class which is then instantiated for each liquid template render run.
  #
  # The Strainer only allows method calls defined in filters given to it via Strainer.global_filter,
  # Context#add_filters or Template.register_filter
  class Strainer #:nodoc:
    @@filters = {}
    @@known_filters = Set.new
    @@known_methods = Set.new

    def initialize(context)
      @context = context
    end

    def self.global_filter(filter)
      raise ArgumentError, "Passed filter is not a module" unless filter.is_a?(Module)
      add_known_filter(filter)
      @@filters[filter.name] = filter
    end

    def self.add_known_filter(filter)
      unless @@known_filters.include?(filter)
        @@method_blacklist ||= Set.new(Strainer.instance_methods.map(&:to_s))
        new_methods = filter.instance_methods.map(&:to_s)
        new_methods.reject!{ |m| @@method_blacklist.include?(m) }
        @@known_methods.merge(new_methods)
        @@known_filters.add(filter)
      end
    end

    def self.create(context)
      strainer = Strainer.new(context)
      @@filters.each { |k,m| strainer.extend(m) }
      strainer
    end

    def invoke(method, *args)
      if invokable?(method)
        send(method, *args)
      else
        args.first
      end
    end

    def invokable?(method)
      @@known_methods.include?(method.to_s) && respond_to?(method)
    end
  end
end

Version data entries

6 entries across 6 versions & 2 rubygems

Version Path
liquid-2.5.5 lib/liquid/strainer.rb
liquid-2.5.4 lib/liquid/strainer.rb
liquid-2.5.3 lib/liquid/strainer.rb
liquid-2.5.1 lib/liquid/strainer.rb
wordify_liquid-2.5.1 lib/liquid/strainer.rb
liquid-2.5.0 lib/liquid/strainer.rb