initializer.rb 24.6 KB
Newer Older
1
require 'logger'
2
require 'set'
3 4 5 6 7 8 9
require 'pathname'

$LOAD_PATH.unshift File.dirname(__FILE__)
require 'railties_path'
require 'rails/version'
require 'rails/plugin/locator'
require 'rails/plugin/loader'
10

11

12
RAILS_ENV = (ENV['RAILS_ENV'] || 'development').dup unless defined?(RAILS_ENV)
13 14

module Rails
15 16 17 18
  # The Initializer is responsible for processing the Rails configuration, such
  # as setting the $LOAD_PATH, requiring the right frameworks, initializing
  # logging, and more. It can be run either as a single command that'll just
  # use the default configuration, like this:
19 20 21
  #
  #   Rails::Initializer.run
  #
22 23
  # But normally it's more interesting to pass in a custom configuration
  # through the block running:
24 25 26 27 28
  #
  #   Rails::Initializer.run do |config|
  #     config.frameworks -= [ :action_web_service ]
  #   end
  #
29 30
  # This will use the default configuration options from Rails::Configuration,
  # but allow for overwriting on select areas.
31
  class Initializer
32
    # The Configuration instance used by this Initializer instance.
33
    attr_reader :configuration
34 35 36 37

    # The set of loaded plugins.
    attr_reader :loaded_plugins

38 39 40 41 42 43 44
    # Runs the initializer. By default, this will invoke the #process method,
    # which simply executes all of the initialization routines. Alternately,
    # you can specify explicitly which initialization routine you want:
    #
    #   Rails::Initializer.run(:set_load_path)
    #
    # This is useful if you only want the load path initialized, without
45
    # incuring the overhead of completely loading the entire environment.
46 47
    def self.run(command = :process, configuration = Configuration.new)
      yield configuration if block_given?
48 49 50
      initializer = new configuration
      initializer.send(command)
      initializer
51
    end
52

53 54
    # Create a new Initializer instance that references the given Configuration
    # instance.
55 56
    def initialize(configuration)
      @configuration = configuration
57
      @loaded_plugins = []
58
    end
59

60 61 62 63 64 65 66 67 68 69 70 71 72 73
    # Sequentially step through all of the available initialization routines,
    # in order:
    #
    # * #set_load_path
    # * #set_connection_adapters
    # * #require_frameworks
    # * #load_environment
    # * #initialize_database
    # * #initialize_logger
    # * #initialize_framework_logging
    # * #initialize_framework_views
    # * #initialize_dependency_mechanism
    # * #initialize_breakpoints
    # * #initialize_whiny_nils
74
    # * #initialize_framework_settings
75 76
    # * #load_environment
    # * #load_plugins
77
    # * #load_observers
78
    # * #initialize_routing
79 80
    # * #after_initialize
    # * #load_application_initializers
81 82 83 84
    #
    # (Note that #load_environment is invoked twice, once at the start and
    # once at the end, to support the legacy configuration style where the
    # environment could overwrite the defaults directly, instead of via the
85
    # Configuration instance.
86
    def process
87
      check_ruby_version
88
      set_load_path
89
      set_connection_adapters
90 91

      require_frameworks
92
      set_autoload_paths
93
      load_environment
94

95
      initialize_encoding
96 97 98
      initialize_database
      initialize_logger
      initialize_framework_logging
99
      initialize_framework_views
100 101 102
      initialize_dependency_mechanism
      initialize_breakpoints
      initialize_whiny_nils
103
      initialize_temporary_directories
104
      initialize_framework_settings
105

106 107 108 109
      # Support for legacy configuration style where the environment
      # could overwrite anything set from the defaults/global through
      # the individual base class configurations.
      load_environment
110

111
      add_support_load_paths
J
Jamis Buck 已提交
112 113

      load_plugins
114

115 116
      # Observers are loaded after plugins in case Observers or observed models are modified by plugins.
      load_observers
117 118 119

      # Routing must be initialized after plugins to allow the former to extend the routes
      initialize_routing
120

121 122
      # the framework is now fully initialized
      after_initialize
123 124

      load_application_initializers
125
    end
126 127 128 129

    # Check for valid Ruby version
    # This is done in an external file, so we can use it
    # from the `rails` program as well without duplication.
130
    def check_ruby_version
131 132 133
      require 'ruby_version_check'
    end

134 135
    # Set the <tt>$LOAD_PATH</tt> based on the value of
    # Configuration#load_paths. Duplicates are removed.
136
    def set_load_path
137 138
      load_paths = configuration.load_paths + configuration.framework_paths
      load_paths.reverse_each { |dir| $LOAD_PATH.unshift(dir) if File.directory?(dir) }
139 140
      $LOAD_PATH.uniq!
    end
141

142 143
    # Set the paths from which Rails will automatically load source files, and
    # the load_once paths.
144
    def set_autoload_paths
145
      Dependencies.load_paths = configuration.load_paths.uniq
146
      Dependencies.load_once_paths = configuration.load_once_paths.uniq
147

148 149 150
      extra = Dependencies.load_once_paths - Dependencies.load_paths
      unless extra.empty?
        abort <<-end_error
151
          load_once_paths must be a subset of the load_paths.
152 153 154
          Extra items in load_once_paths: #{extra * ','}
        end_error
      end
155

156 157
      # Freeze the arrays so future modifications will fail rather than do nothing mysteriously
      configuration.load_once_paths.freeze
158
    end
159

160 161 162 163
    # Sets the +RAILS_CONNECTION_ADAPTERS+ constant based on the value of
    # Configuration#connection_adapters. This constant is used to determine
    # which database adapters should be loaded (by default, all adapters are
    # loaded).
164
    def set_connection_adapters
165
      Object.const_set("RAILS_CONNECTION_ADAPTERS", configuration.connection_adapters) if configuration.connection_adapters
166
    end
167

168 169 170
    # Requires all frameworks specified by the Configuration#frameworks
    # list. By default, all frameworks (ActiveRecord, ActiveSupport,
    # ActionPack, ActionMailer, and ActionWebService) are loaded.
171 172 173
    def require_frameworks
      configuration.frameworks.each { |framework| require(framework.to_s) }
    end
174

175 176
    # Add the load paths used by support functions such as the info controller
    def add_support_load_paths
177
    end
178 179 180 181 182

    # Loads all plugins in <tt>config.plugin_paths</tt>.  <tt>plugin_paths</tt>
    # defaults to <tt>vendor/plugins</tt> but may also be set to a list of
    # paths, such as
    #   config.plugin_paths = ['lib/plugins', 'vendor/plugins']
183
    #
184 185 186
    # Each plugin discovered in <tt>plugin_paths</tt> is initialized:
    # * add its +lib+ directory, if present, to the beginning of the load path
    # * evaluate <tt>init.rb</tt> if present
187 188
    #
    # After all plugins are loaded, duplicates are removed from the load path.
189 190
    # If an array of plugin names is specified in config.plugins, only those plugins will be loaded
    # and they plugins will be loaded in that order. Otherwise, plugins are loaded in alphabetical
191
    # order.
J
Jamis Buck 已提交
192
    def load_plugins
193 194 195 196
      configuration.plugin_locators.each do |locator|
        locator.new(self).each do |plugin|
          plugin.load
        end
197
      end
198
      ensure_all_registered_plugins_are_loaded!
199
      $LOAD_PATH.uniq!
J
Jamis Buck 已提交
200 201
    end

202 203
    # Loads the environment specified by Configuration#environment_path, which
    # is typically one of development, testing, or production.
204 205 206
    def load_environment
      silence_warnings do
        config = configuration
207
        constants = self.class.constants
208
        
209
        eval(IO.read(configuration.environment_path), binding, configuration.environment_path)
210
        
211 212 213
        (self.class.constants - constants).each do |const|
          Object.const_set(const, self.class.const_get(const))
        end
214
      end
215
    end
216 217 218 219 220

    def load_observers
      ActiveRecord::Base.instantiate_observers
    end

221
    # This initialzation sets $KCODE to 'u' to enable the multibyte safe operations.
222
    # Plugin authors supporting other encodings should override this behaviour and
223 224 225 226 227
    # set the relevant +default_charset+ on ActionController::Base
    def initialize_encoding
      $KCODE='u'
    end

228 229 230 231
    # This initialization routine does nothing unless <tt>:active_record</tt>
    # is one of the frameworks to load (Configuration#frameworks). If it is,
    # this sets the database configuration from Configuration#database_configuration
    # and then establishes the connection.
232 233 234 235 236
    def initialize_database
      return unless configuration.frameworks.include?(:active_record)
      ActiveRecord::Base.configurations = configuration.database_configuration
      ActiveRecord::Base.establish_connection
    end
237

238 239 240 241 242 243 244 245
    # If the +RAILS_DEFAULT_LOGGER+ constant is already set, this initialization
    # routine does nothing. If the constant is not set, and Configuration#logger
    # is not +nil+, this also does nothing. Otherwise, a new logger instance
    # is created at Configuration#log_path, with a default log level of
    # Configuration#log_level.
    #
    # If the log could not be created, the log will be set to output to
    # +STDERR+, with a log level of +WARN+.
246
    def initialize_logger
247 248 249
      # if the environment has explicitly defined a logger, use it
      return if defined?(RAILS_DEFAULT_LOGGER)

250 251 252 253 254 255 256 257 258 259 260 261
      unless logger = configuration.logger
        begin
          logger = Logger.new(configuration.log_path)
          logger.level = Logger.const_get(configuration.log_level.to_s.upcase)
        rescue StandardError
          logger = Logger.new(STDERR)
          logger.level = Logger::WARN
          logger.warn(
            "Rails Error: Unable to access log file. Please ensure that #{configuration.log_path} exists and is chmod 0666. " +
            "The log level has been raised to WARN and the output directed to STDERR until the problem is fixed."
          )
        end
262
      end
263

264
      silence_warnings { Object.const_set "RAILS_DEFAULT_LOGGER", logger }
265
    end
266

267 268 269 270
    # Sets the logger for ActiveRecord, ActionController, and ActionMailer
    # (but only for those frameworks that are to be loaded). If the framework's
    # logger is already set, it is not changed, otherwise it is set to use
    # +RAILS_DEFAULT_LOGGER+.
271
    def initialize_framework_logging
272 273 274
      for framework in ([ :active_record, :action_controller, :action_mailer ] & configuration.frameworks)
        framework.to_s.camelize.constantize.const_get("Base").logger ||= RAILS_DEFAULT_LOGGER
      end
275
    end
276

277
    # Sets +ActionController::BaseEview_paths+ and +ActionMailer::Base#template_root+
278
    # (but only for those frameworks that are to be loaded). If the framework's
279
    # paths have already been set, it is not changed, otherwise it is
280
    # set to use Configuration#view_path.
281
    def initialize_framework_views
282 283
      ActionMailer::Base.template_root ||= configuration.view_path  if configuration.frameworks.include?(:action_mailer)
      ActionController::Base.view_paths = [configuration.view_path] if configuration.frameworks.include?(:action_controller) && ActionController::Base.view_paths.empty?
284 285
    end

286 287 288
    # If ActionController is not one of the loaded frameworks (Configuration#frameworks)
    # this does nothing. Otherwise, it loads the routing definitions and sets up
    # loading module used to lazily load controllers (Configuration#controller_paths).
289 290
    def initialize_routing
      return unless configuration.frameworks.include?(:action_controller)
291
      ActionController::Routing.controller_paths = configuration.controller_paths
292 293
      ActionController::Routing::Routes.reload
    end
294

295 296
    # Sets the dependency loading mechanism based on the value of
    # Configuration#cache_classes.
297 298
    def initialize_dependency_mechanism
      Dependencies.mechanism = configuration.cache_classes ? :require : :load
299
    end
300

301 302
    # Sets the +BREAKPOINT_SERVER_PORT+ if Configuration#breakpoint_server
    # is true.
303 304
    def initialize_breakpoints
      silence_warnings { Object.const_set("BREAKPOINT_SERVER_PORT", 42531) if configuration.breakpoint_server }
305
    end
306 307 308

    # Loads support for "whiny nil" (noisy warnings when methods are invoked
    # on +nil+ values) if Configuration#whiny_nils is true.
309 310 311
    def initialize_whiny_nils
      require('active_support/whiny_nil') if configuration.whiny_nils
    end
312

313 314
    def initialize_temporary_directories
      if configuration.frameworks.include?(:action_controller)
315
        session_path = "#{configuration.root_path}/tmp/sessions/"
316
        ActionController::Base.session_options[:tmpdir] = File.exist?(session_path) ? session_path : Dir::tmpdir
317

318
        cache_path = "#{configuration.root_path}/tmp/cache/"
319 320 321 322 323 324
        if File.exist?(cache_path)
          ActionController::Base.fragment_cache_store = :file_store, cache_path
        end
      end
    end

325
    # Initializes framework-specific settings for each of the loaded frameworks
326 327
    # (Configuration#frameworks). The available settings map to the accessors
    # on each of the corresponding Base classes.
328
    def initialize_framework_settings
329 330
      configuration.frameworks.each do |framework|
        base_class = framework.to_s.camelize.constantize.const_get("Base")
331

332 333 334
        configuration.send(framework).each do |setting, value|
          base_class.send("#{setting}=", value)
        end
335 336
      end
    end
337 338

    # Fires the user-supplied after_initialize block (Configuration#after_initialize)
339 340 341
    def after_initialize
      configuration.after_initialize_block.call if configuration.after_initialize_block
    end
342

343
    def load_application_initializers
344
      Dir["#{configuration.root_path}/config/initializers/**/*.rb"].each do |initializer|
345 346 347
        load(initializer)
      end
    end
348

349 350 351 352 353 354 355 356 357
    private
      def ensure_all_registered_plugins_are_loaded!
        unless configuration.plugins.nil?
          unless loaded_plugins == configuration.plugins
            missing_plugins = configuration.plugins - loaded_plugins
            raise LoadError, "Could not locate the following plugins: #{missing_plugins.to_sentence}"
          end
        end
      end
358
  end
359

360 361 362 363 364 365
  # The Configuration class holds all the parameters for the Initializer and
  # ships with defaults that suites most Rails applications. But it's possible
  # to overwrite everything. Usually, you'll create an Configuration file
  # implicitly through the block running on the Initializer, but it's also
  # possible to create the Configuration instance in advance and pass it in
  # like this:
366 367 368
  #
  #   config = Rails::Configuration.new
  #   Rails::Initializer.run(:process, config)
369
  class Configuration
370 371 372
    # The application's base directory.
    attr_reader :root_path

373 374
    # A stub for setting options on ActionController::Base
    attr_accessor :action_controller
375

376 377
    # A stub for setting options on ActionMailer::Base
    attr_accessor :action_mailer
378

379 380
    # A stub for setting options on ActionView::Base
    attr_accessor :action_view
381

382 383
    # A stub for setting options on ActionWebService::Base
    attr_accessor :action_web_service
384

385 386
    # A stub for setting options on ActiveRecord::Base
    attr_accessor :active_record
387

388 389
    # Whether or not to use the breakpoint server (boolean)
    attr_accessor :breakpoint_server
390

391 392 393
    # Whether or not classes should be cached (set to false if you want
    # application classes to be reloaded on each request)
    attr_accessor :cache_classes
394

395 396 397
    # The list of connection adapters to load. (By default, all connection
    # adapters are loaded. You can set this to be just the adapter(s) you
    # will use to reduce your application's load time.)
398
    attr_accessor :connection_adapters
399

400 401 402 403 404 405 406
    # The list of paths that should be searched for controllers. (Defaults
    # to <tt>app/controllers</tt> and <tt>components</tt>.)
    attr_accessor :controller_paths

    # The path to the database configuration file to use. (Defaults to
    # <tt>config/database.yml</tt>.)
    attr_accessor :database_configuration_file
407

408 409 410 411 412
    # The list of rails framework components that should be loaded. (Defaults
    # to <tt>:active_record</tt>, <tt>:action_controller</tt>,
    # <tt>:action_view</tt>, <tt>:action_mailer</tt>, and
    # <tt>:action_web_service</tt>).
    attr_accessor :frameworks
413

414 415 416
    # An array of additional paths to prepend to the load path. By default,
    # all +app+, +lib+, +vendor+ and mock paths are included in this list.
    attr_accessor :load_paths
417

418
    # An array of paths from which Rails will automatically load from only once.
419
    # All elements of this array must also be in +load_paths+.
420
    attr_accessor :load_once_paths
421

422 423 424 425
    # The log level to use for the default Rails logger. In production mode,
    # this defaults to <tt>:info</tt>. In development mode, it defaults to
    # <tt>:debug</tt>.
    attr_accessor :log_level
426

427 428 429
    # The path to the log file to use. Defaults to log/#{environment}.log
    # (e.g. log/development.log or log/production.log).
    attr_accessor :log_path
430

431 432 433 434 435
    # The specific logger to use. By default, a logger will be created and
    # initialized using #log_path and #log_level, but a programmer may
    # specifically set the logger to use via this accessor and it will be
    # used directly.
    attr_accessor :logger
436

437 438
    # The root of the application's views. (Defaults to <tt>app/views</tt>.)
    attr_accessor :view_path
439

440 441 442
    # Set to +true+ if you want to be warned (noisily) when you try to invoke
    # any method of +nil+. Set to +false+ for the standard Ruby behavior.
    attr_accessor :whiny_nils
443

444 445 446
    # The list of plugins to load. If this is set to <tt>nil</tt>, all plugins will
    # be loaded. If this is set to <tt>[]</tt>, no plugins will be loaded. Otherwise,
    # plugins will be loaded in the order specified.
447
    attr_accessor :plugins
448

449 450 451
    # The path to the root of the plugins directory. By default, it is in
    # <tt>vendor/plugins</tt>.
    attr_accessor :plugin_paths
452

453 454 455 456 457
    # The classes that handle finding the desired plugins that you'd like to load for
    # your application. By default it is the Rails::Plugin::FileSystemLocator which finds
    # plugins to load in <tt>vendor/plugins</tt>. You can hook into gem location by subclassing
    # Rails::Plugin::Locator and adding it onto the list of <tt>plugin_locators</tt>.
    attr_accessor :plugin_locators
458 459

    # The class that handles loading each plugin. Defaults to Rails::Plugin::Loader, but
460 461 462
    # a sub class would have access to fine grained modification of the loading behavior. See
    # the implementation of Rails::Plugin::Loader for more details.
    attr_accessor :plugin_loader
463

464 465
    # Create a new Configuration instance, initialized with the default
    # values.
466
    def initialize
467 468
      set_root_path!

469 470
      self.frameworks                   = default_frameworks
      self.load_paths                   = default_load_paths
471
      self.load_once_paths              = default_load_once_paths
472 473 474 475 476 477 478
      self.log_path                     = default_log_path
      self.log_level                    = default_log_level
      self.view_path                    = default_view_path
      self.controller_paths             = default_controller_paths
      self.cache_classes                = default_cache_classes
      self.breakpoint_server            = default_breakpoint_server
      self.whiny_nils                   = default_whiny_nils
479
      self.plugins                      = default_plugins
480
      self.plugin_paths                 = default_plugin_paths
481
      self.plugin_locators              = default_plugin_locators
482
      self.plugin_loader                = default_plugin_loader
483
      self.database_configuration_file  = default_database_configuration_file
484

485
      for framework in default_frameworks
486
        self.send("#{framework}=", Rails::OrderedOptions.new)
487
      end
488
    end
489

490 491 492 493 494 495 496 497 498 499 500 501 502 503 504
    # Set the root_path to RAILS_ROOT and canonicalize it.
    def set_root_path!
      raise 'RAILS_ROOT is not set' unless defined?(::RAILS_ROOT)
      raise 'RAILS_ROOT is not a directory' unless File.directory?(::RAILS_ROOT)

      @root_path =
        # Pathname is incompatible with Windows, but Windows doesn't have
        # real symlinks so File.expand_path is safe.
        if RUBY_PLATFORM =~ /(:?mswin|mingw)/
          File.expand_path(::RAILS_ROOT)

        # Otherwise use Pathname#realpath which respects symlinks.
        else
          Pathname.new(::RAILS_ROOT).realpath.to_s
        end
505 506

      ::RAILS_ROOT.replace @root_path
507 508
    end

509 510 511
    # Loads and returns the contents of the #database_configuration_file. The
    # contents of the file are processed via ERB before being sent through
    # YAML::load.
512
    def database_configuration
513
      YAML::load(ERB.new(IO.read(database_configuration_file)).result)
514
    end
515

516 517
    # The path to the current environment's file (development.rb, etc.). By
    # default the file is at <tt>config/environments/#{environment}.rb</tt>.
518
    def environment_path
519
      "#{root_path}/config/environments/#{environment}.rb"
520
    end
J
Jamis Buck 已提交
521

522 523
    # Return the currently selected environment. By default, it returns the
    # value of the +RAILS_ENV+ constant.
524 525 526
    def environment
      ::RAILS_ENV
    end
527

528
    # Sets a block which will be executed after rails has been fully initialized.
529
    # Useful for per-environment configuration which depends on the framework being
530 531 532 533
    # fully initialized.
    def after_initialize(&after_initialize_block)
      @after_initialize_block = after_initialize_block
    end
534

535 536 537 538
    # Returns the block set in Configuration#after_initialize
    def after_initialize_block
      @after_initialize_block
    end
539

N
Nicholas Seckar 已提交
540 541
    # Add a preparation callback that will run before every request in development
    # mode, or before the first request in production.
542
    #
N
Nicholas Seckar 已提交
543 544
    # See Dispatcher#to_prepare.
    def to_prepare(&callback)
545
      require 'dispatcher' unless defined?(::Dispatcher)
N
Nicholas Seckar 已提交
546 547
      Dispatcher.to_prepare(&callback)
    end
548

549 550 551 552
    def builtin_directories
      # Include builtins only in the development environment.
      (environment == 'development') ? Dir["#{RAILTIES_PATH}/builtin/*/"] : []
    end
553

554 555 556 557 558 559 560 561 562 563 564 565
    def framework_paths
      # TODO: Don't include dirs for frameworks that are not used
      %w(
        railties
        railties/lib
        actionpack/lib
        activesupport/lib
        activerecord/lib
        actionmailer/lib
        actionwebservice/lib
      ).map { |dir| "#{framework_root_path}/#{dir}" }.select { |dir| File.directory?(dir) }
    end
566

567
    private
568 569 570 571
      def framework_root_path
        defined?(::RAILS_FRAMEWORK_ROOT) ? ::RAILS_FRAMEWORK_ROOT : "#{root_path}/vendor/rails"
      end

572
      def default_frameworks
573
        [ :active_record, :action_controller, :action_view, :action_mailer, :action_web_service ]
574
      end
575

576
      def default_load_paths
577
        paths = ["#{root_path}/test/mocks/#{environment}"]
578

N
Nicholas Seckar 已提交
579 580
        # Add the app's controller directory
        paths.concat(Dir["#{root_path}/app/controllers/"])
581

582
        # Then components subdirectories.
583
        paths.concat(Dir["#{root_path}/components/[_a-z]*"])
584 585 586

        # Followed by the standard includes.
        paths.concat %w(
587 588
          app
          app/models
589 590 591
          app/controllers
          app/helpers
          app/services
592 593 594 595 596
          app/apis
          components
          config
          lib
          vendor
597
        ).map { |dir| "#{root_path}/#{dir}" }.select { |dir| File.directory?(dir) }
598

599
        paths.concat builtin_directories
600
      end
601

602
      # Doesn't matter since plugins aren't in load_paths yet.
603
      def default_load_once_paths
604
        []
605
      end
606 607

      def default_log_path
608
        File.join(root_path, 'log', "#{environment}.log")
609
      end
610

611
      def default_log_level
612
        environment == 'production' ? :info : :debug
613
      end
614

615
      def default_database_configuration_file
616
        File.join(root_path, 'config', 'database.yml')
617
      end
618

619
      def default_view_path
620
        File.join(root_path, 'app', 'views')
621
      end
622

623
      def default_controller_paths
624
        paths = [File.join(root_path, 'app', 'controllers')]
625 626
        paths.concat builtin_directories
        paths
627
      end
628

629 630 631
      def default_dependency_mechanism
        :load
      end
632

633 634 635
      def default_cache_classes
        false
      end
636

637 638 639
      def default_breakpoint_server
        false
      end
640

641 642 643
      def default_whiny_nils
        false
      end
644

645
      def default_plugins
646
        nil
647 648
      end

649 650 651
      def default_plugin_paths
        ["#{root_path}/vendor/plugins"]
      end
652

653 654 655
      def default_plugin_locators
        [Plugin::FileSystemLocator]
      end
656

657 658 659
      def default_plugin_loader
        Plugin::Loader
      end
660
  end
661
end
662

663
# Needs to be duplicated from Active Support since its needed before Active
664 665 666 667 668 669
# Support is available. Here both Options and Hash are namespaced to prevent
# conflicts with other implementations AND with the classes residing in ActiveSupport.
class Rails::OrderedOptions < Array #:nodoc:
  def []=(key, value)
    key = key.to_sym

670 671 672 673 674 675 676 677
    if pair = find_pair(key)
      pair.pop
      pair << value
    else
      self << [key, value]
    end
  end

678
  def [](key)
679 680
    pair = find_pair(key.to_sym)
    pair ? pair.last : nil
681 682 683 684 685 686 687 688 689
  end

  def method_missing(name, *args)
    if name.to_s =~ /(.*)=$/
      self[$1.to_sym] = args.first
    else
      self[name]
    end
  end
690 691 692 693 694 695

  private
    def find_pair(key)
      self.each { |i| return i if i.first == key }
      return false
    end
696
end