module Rails
module Initializable
def self.included(base)
base.extend ClassMethods
end
class Initializer
attr_reader :name, :before, :after, :global, :block
def initialize(name, context, options, &block)
@name, @context, @options, @block = name, context, options, block
end
def before
@options[:before]
end
def after
@options[:after]
end
def global
@options[:global]
end
alias global? global
def run(*args)
@context.instance_exec(*args, &block)
end
def bind(context)
return self if @context
Initializer.new(@name, context, @options, &block)
end
end
class Collection < Array
def initialize(initializers = [])
super()
initializers.each do |initializer|
if initializer.before
index = index_for(initializer.before)
elsif initializer.after
index = index_for(initializer.after)
index += 1 if index
else
index = length
end
insert(index || -1, initializer)
end
end
def +(other)
Collection.new(to_a + other.to_a)
end
def index_for(name)
initializer = find { |i| i.name == name }
initializer && index(initializer)
end
end
def run_initializers(*args)
return if @ran
initializers.each do |initializer|
initializer.run(*args)
end
@ran = true
end
def initializers
@initializers ||= begin
initializers = self.class.initializers_for(:instance)
Collection.new(initializers.map { |i| i.bind(self) })
end
end
module ClassMethods
def initializers
@initializers ||= []
end
def initializers_for(scope = :global)
initializers = Collection.new
ancestors.reverse_each do |klass|
next unless klass.respond_to?(:initializers)
initializers = initializers + klass.initializers.select { |i|
(scope == :global) == !!i.global?
}
end
initializers
end
def initializer(name, opts = {}, &blk)
@initializers ||= []
@initializers << Initializer.new(name, nil, opts, &blk)
end
def run_initializers(*args)
return if @ran
initializers_for(:global).each do |initializer|
instance_exec(*args, &initializer.block)
end
@ran = true
end
end
end
include Initializable
# Check for valid Ruby version (1.8.2 or 1.8.4 or higher). This is done in an
# external file, so we can use it from the `rails` program as well without duplication.
initializer :check_ruby_version, :global => true do
require 'rails/ruby_version_check'
end
# For Ruby 1.8, this initialization sets $KCODE to 'u' to enable the
# multibyte safe operations. Plugin authors supporting other encodings
# should override this behaviour and set the relevant +default_charset+
# on ActionController::Base.
#
# For Ruby 1.9, UTF-8 is the default internal and external encoding.
initializer :initialize_encoding, :global => true do
if RUBY_VERSION < '1.9'
$KCODE='u'
else
Encoding.default_external = Encoding::UTF_8
end
end
end