initializer.rb 24.5 KB
Newer Older
1
require 'logger'
2
require 'set'
3
require File.join(File.dirname(__FILE__), 'railties_path')
4

5
RAILS_ENV = (ENV['RAILS_ENV'] || 'development').dup unless defined?(RAILS_ENV)
6 7

module Rails
8 9 10 11
  # 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:
12 13 14
  #
  #   Rails::Initializer.run
  #
15 16
  # But normally it's more interesting to pass in a custom configuration
  # through the block running:
17 18 19 20 21
  #
  #   Rails::Initializer.run do |config|
  #     config.frameworks -= [ :action_web_service ]
  #   end
  #
22 23
  # This will use the default configuration options from Rails::Configuration,
  # but allow for overwriting on select areas.
24
  class Initializer
25
    # The Configuration instance used by this Initializer instance.
26
    attr_reader :configuration
27 28 29 30

    # The set of loaded plugins.
    attr_reader :loaded_plugins

31 32 33 34 35 36 37
    # 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
38
    # incuring the overhead of completely loading the entire environment.
39 40
    def self.run(command = :process, configuration = Configuration.new)
      yield configuration if block_given?
41 42 43
      initializer = new configuration
      initializer.send(command)
      initializer
44
    end
45

46 47
    # Create a new Initializer instance that references the given Configuration
    # instance.
48 49
    def initialize(configuration)
      @configuration = configuration
50
      @loaded_plugins = Set.new
51
    end
52

53 54 55 56 57 58 59 60 61 62 63 64 65 66
    # 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
67
    # * #initialize_framework_settings
68 69
    # * #load_environment
    # * #load_plugins
70
    # * #load_observers
71
    # * #initialize_routing
72 73 74 75
    #
    # (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
76
    # Configuration instance.
77
    def process
78
      check_ruby_version
79
      load_rails_version
80
      set_load_path
81
      set_connection_adapters
82 83

      require_frameworks
84
      set_autoload_paths
85
      load_environment
86

87
      initialize_encoding
88 89 90
      initialize_database
      initialize_logger
      initialize_framework_logging
91
      initialize_framework_views
92 93 94
      initialize_dependency_mechanism
      initialize_breakpoints
      initialize_whiny_nils
95
      initialize_temporary_directories
96
      initialize_framework_settings
97

98 99 100 101
      # Support for legacy configuration style where the environment
      # could overwrite anything set from the defaults/global through
      # the individual base class configurations.
      load_environment
102

103
      add_support_load_paths
J
Jamis Buck 已提交
104 105

      load_plugins
106

107 108
      # Observers are loaded after plugins in case Observers or observed models are modified by plugins.
      load_observers
109 110 111

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

113 114
      # the framework is now fully initialized
      after_initialize
115
    end
116 117 118 119

    # 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.
120
    def check_ruby_version
121 122 123
      require 'ruby_version_check'
    end

124 125 126 127 128
    # Rails::VERSION should always be available.
    def load_rails_version
      require 'rails/version'
    end

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

137 138
    # Set the paths from which Rails will automatically load source files, and
    # the load_once paths.
139
    def set_autoload_paths
140
      Dependencies.load_paths = configuration.load_paths.uniq
141
      Dependencies.load_once_paths = configuration.load_once_paths.uniq
142

143 144 145
      extra = Dependencies.load_once_paths - Dependencies.load_paths
      unless extra.empty?
        abort <<-end_error
146
          load_once_paths must be a subset of the load_paths.
147 148 149
          Extra items in load_once_paths: #{extra * ','}
        end_error
      end
150

151 152
      # Freeze the arrays so future modifications will fail rather than do nothing mysteriously
      configuration.load_once_paths.freeze
153
    end
154

155 156 157 158
    # 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).
159
    def set_connection_adapters
160
      Object.const_set("RAILS_CONNECTION_ADAPTERS", configuration.connection_adapters) if configuration.connection_adapters
161
    end
162

163 164 165
    # Requires all frameworks specified by the Configuration#frameworks
    # list. By default, all frameworks (ActiveRecord, ActiveSupport,
    # ActionPack, ActionMailer, and ActionWebService) are loaded.
166 167 168
    def require_frameworks
      configuration.frameworks.each { |framework| require(framework.to_s) }
    end
169

170 171
    # Add the load paths used by support functions such as the info controller
    def add_support_load_paths
172
    end
173 174 175 176 177

    # 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']
178
    #
179 180 181
    # 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
182 183
    #
    # After all plugins are loaded, duplicates are removed from the load path.
184
    # Plugins are loaded in alphabetical order.
J
Jamis Buck 已提交
185
    def load_plugins
186
      find_plugins(configuration.plugin_paths).sort.each { |path| load_plugin path }
J
Jamis Buck 已提交
187 188 189
      $LOAD_PATH.uniq!
    end

190 191
    # Loads the environment specified by Configuration#environment_path, which
    # is typically one of development, testing, or production.
192 193 194
    def load_environment
      silence_warnings do
        config = configuration
195
        constants = self.class.constants
196
        eval(IO.read(configuration.environment_path), binding, configuration.environment_path)
197 198 199
        (self.class.constants - constants).each do |const|
          Object.const_set(const, self.class.const_get(const))
        end
200
      end
201
    end
202 203 204 205 206

    def load_observers
      ActiveRecord::Base.instantiate_observers
    end

207
    # This initialzation sets $KCODE to 'u' to enable the multibyte safe operations.
208
    # Plugin authors supporting other encodings should override this behaviour and
209 210 211 212 213
    # set the relevant +default_charset+ on ActionController::Base
    def initialize_encoding
      $KCODE='u'
    end

214 215 216 217
    # 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.
218 219 220 221 222
    def initialize_database
      return unless configuration.frameworks.include?(:active_record)
      ActiveRecord::Base.configurations = configuration.database_configuration
      ActiveRecord::Base.establish_connection
    end
223

224 225 226 227 228 229 230 231
    # 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+.
232
    def initialize_logger
233 234 235
      # if the environment has explicitly defined a logger, use it
      return if defined?(RAILS_DEFAULT_LOGGER)

236 237 238 239 240 241 242 243 244 245 246 247
      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
248
      end
249

250
      silence_warnings { Object.const_set "RAILS_DEFAULT_LOGGER", logger }
251
    end
252

253 254 255 256
    # 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+.
257
    def initialize_framework_logging
258 259 260
      for framework in ([ :active_record, :action_controller, :action_mailer ] & configuration.frameworks)
        framework.to_s.camelize.constantize.const_get("Base").logger ||= RAILS_DEFAULT_LOGGER
      end
261
    end
262

263 264 265 266
    # Sets the +template_root+ for ActionController::Base and ActionMailer::Base
    # (but only for those frameworks that are to be loaded). If the framework's
    # +template_root+ has already been set, it is not changed, otherwise it is
    # set to use Configuration#view_path.
267
    def initialize_framework_views
268 269 270
      for framework in ([ :action_controller, :action_mailer ] & configuration.frameworks)
        framework.to_s.camelize.constantize.const_get("Base").template_root ||= configuration.view_path
      end
271 272
    end

273 274 275
    # 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).
276 277
    def initialize_routing
      return unless configuration.frameworks.include?(:action_controller)
278
      ActionController::Routing.controller_paths = configuration.controller_paths
279 280
      ActionController::Routing::Routes.reload
    end
281

282 283
    # Sets the dependency loading mechanism based on the value of
    # Configuration#cache_classes.
284 285
    def initialize_dependency_mechanism
      Dependencies.mechanism = configuration.cache_classes ? :require : :load
286
    end
287

288 289
    # Sets the +BREAKPOINT_SERVER_PORT+ if Configuration#breakpoint_server
    # is true.
290 291
    def initialize_breakpoints
      silence_warnings { Object.const_set("BREAKPOINT_SERVER_PORT", 42531) if configuration.breakpoint_server }
292
    end
293 294 295

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

300 301 302 303
    def initialize_temporary_directories
      if configuration.frameworks.include?(:action_controller)
        session_path = "#{RAILS_ROOT}/tmp/sessions/"
        ActionController::Base.session_options[:tmpdir] = File.exist?(session_path) ? session_path : Dir::tmpdir
304

305 306 307 308 309 310 311
        cache_path = "#{RAILS_ROOT}/tmp/cache/"
        if File.exist?(cache_path)
          ActionController::Base.fragment_cache_store = :file_store, cache_path
        end
      end
    end

312
    # Initializes framework-specific settings for each of the loaded frameworks
313 314
    # (Configuration#frameworks). The available settings map to the accessors
    # on each of the corresponding Base classes.
315
    def initialize_framework_settings
316 317
      configuration.frameworks.each do |framework|
        base_class = framework.to_s.camelize.constantize.const_get("Base")
318

319 320 321
        configuration.send(framework).each do |setting, value|
          base_class.send("#{setting}=", value)
        end
322 323
      end
    end
324 325

    # Fires the user-supplied after_initialize block (Configuration#after_initialize)
326 327 328
    def after_initialize
      configuration.after_initialize_block.call if configuration.after_initialize_block
    end
329

330 331 332 333
    protected
      # Return a list of plugin paths within base_path.  A plugin path is
      # a directory that contains either a lib directory or an init.rb file.
      # This recurses into directories which are not plugin paths, so you
334
      # may organize your plugins within the plugin path.
335 336 337 338
      def find_plugins(*base_paths)
        base_paths.flatten.inject([]) do |plugins, base_path|
          Dir.glob(File.join(base_path, '*')).each do |path|
            if plugin_path?(path)
339
              plugins << path if plugin_enabled?(path)
340 341 342 343 344 345 346 347 348 349 350 351
            elsif File.directory?(path)
              plugins += find_plugins(path)
            end
          end
          plugins
        end
      end

      def plugin_path?(path)
        File.directory?(path) and (File.directory?(File.join(path, 'lib')) or File.file?(File.join(path, 'init.rb')))
      end

352 353 354 355
      def plugin_enabled?(path)
        configuration.plugins.empty? || configuration.plugins.include?(File.basename(path))
      end

356 357 358 359 360 361 362 363 364
      # Load the plugin at <tt>path</tt> unless already loaded.
      #
      # Each plugin is initialized:
      # * add its +lib+ directory, if present, to the beginning of the load path
      # * evaluate <tt>init.rb</tt> if present
      #
      # Returns <tt>true</tt> if the plugin is successfully loaded or
      # <tt>false</tt> if it is already loaded (similar to Kernel#require).
      # Raises <tt>LoadError</tt> if the plugin is not found.
365 366
      def load_plugin(directory)
        name = File.basename(directory)
367
        return false if loaded_plugins.include?(name)
368 369

        # Catch nonexistent and empty plugins.
370
        raise LoadError, "No such plugin: #{directory}" unless plugin_path?(directory)
371

372 373
        lib_path  = File.join(directory, 'lib')
        init_path = File.join(directory, 'init.rb')
374 375 376
        has_lib   = File.directory?(lib_path)
        has_init  = File.file?(init_path)

377 378 379
        # Add lib to load path *after* the application lib, to allow
        # application libraries to override plugin libraries.
        if has_lib
380
          application_lib_index = $LOAD_PATH.index(File.join(RAILS_ROOT, "lib")) || 0
381 382 383
          $LOAD_PATH.insert(application_lib_index + 1, lib_path)
        end

384
        # Allow plugins to reference the current configuration object
385
        config = configuration
386 387 388
	
        # Add to set of loaded plugins before 'name' collapsed in eval.
        loaded_plugins << name
389 390

        # Evaluate init.rb.
391
        silence_warnings { eval(IO.read(init_path), binding, init_path) } if has_init
392 393 394

        true
      end
395
  end
396

397 398 399 400 401 402
  # 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:
403 404 405
  #
  #   config = Rails::Configuration.new
  #   Rails::Initializer.run(:process, config)
406
  class Configuration
407 408
    # A stub for setting options on ActionController::Base
    attr_accessor :action_controller
409

410 411
    # A stub for setting options on ActionMailer::Base
    attr_accessor :action_mailer
412

413 414
    # A stub for setting options on ActionView::Base
    attr_accessor :action_view
415

416 417
    # A stub for setting options on ActionWebService::Base
    attr_accessor :action_web_service
418

419 420
    # A stub for setting options on ActiveRecord::Base
    attr_accessor :active_record
421

422 423
    # Whether or not to use the breakpoint server (boolean)
    attr_accessor :breakpoint_server
424

425 426 427
    # 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
428

429 430 431
    # 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.)
432
    attr_accessor :connection_adapters
433

434 435 436 437 438 439 440
    # 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
441

442 443 444 445 446
    # 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
447

448 449 450
    # 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
451

452
    # An array of paths from which Rails will automatically load from only once.
453
    # All elements of this array must also be in +load_paths+.
454
    attr_accessor :load_once_paths
455

456 457 458 459
    # 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
460

461 462 463
    # 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
464

465 466 467 468 469
    # 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
470

471 472
    # The root of the application's views. (Defaults to <tt>app/views</tt>.)
    attr_accessor :view_path
473

474 475 476
    # 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
477 478 479

    # The list of plugins to load. If this is set to <tt>[]</tt>, all plugins will be loaded.
    attr_accessor :plugins
480

481 482 483 484
    # The path to the root of the plugins directory. By default, it is in
    # <tt>vendor/plugins</tt>.
    attr_accessor :plugin_paths

485 486
    # Create a new Configuration instance, initialized with the default
    # values.
487
    def initialize
488 489
      self.frameworks                   = default_frameworks
      self.load_paths                   = default_load_paths
490
      self.load_once_paths              = default_load_once_paths
491 492 493 494 495 496 497
      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
498
      self.plugins                      = default_plugins
499
      self.plugin_paths                 = default_plugin_paths
500
      self.database_configuration_file  = default_database_configuration_file
501

502
      for framework in default_frameworks
503
        self.send("#{framework}=", Rails::OrderedOptions.new)
504
      end
505
    end
506

507 508 509
    # 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.
510
    def database_configuration
511
      YAML::load(ERB.new(IO.read(database_configuration_file)).result)
512
    end
513

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

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

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

533 534 535 536
    # Returns the block set in Configuration#after_initialize
    def after_initialize_block
      @after_initialize_block
    end
537

N
Nicholas Seckar 已提交
538 539
    # Add a preparation callback that will run before every request in development
    # mode, or before the first request in production.
540
    #
N
Nicholas Seckar 已提交
541 542 543 544
    # See Dispatcher#to_prepare.
    def to_prepare(&callback)
      Dispatcher.to_prepare(&callback)
    end
545

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

551 552 553 554 555 556 557 558 559 560 561 562
    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
563

564
    private
565 566 567 568
      def root_path
        ::RAILS_ROOT
      end

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

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

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

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

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

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

600 601
        paths.concat Dir["#{root_path}/vendor/plugins/*/lib/"]
        paths.concat builtin_directories
602
      end
603

604 605
      def default_load_once_paths
        plugin_root = "#{root_path}/vendor/plugins/"
606
        default_load_paths.select do |path|
607 608
          path[0, plugin_root.length] == plugin_root # No begins_with yet
        end
609
      end
610 611

      def default_log_path
612
        File.join(root_path, 'log', "#{environment}.log")
613
      end
614

615
      def default_log_level
616
        environment == 'production' ? :info : :debug
617
      end
618

619
      def default_database_configuration_file
620
        File.join(root_path, 'config', 'database.yml')
621
      end
622

623
      def default_view_path
624
        File.join(root_path, 'app', 'views')
625
      end
626

627
      def default_controller_paths
628 629 630
        paths = [ File.join(root_path, 'app', 'controllers'), File.join(root_path, 'components') ]
        paths.concat builtin_directories
        paths
631
      end
632

633 634 635
      def default_dependency_mechanism
        :load
      end
636

637 638 639
      def default_cache_classes
        false
      end
640

641 642 643
      def default_breakpoint_server
        false
      end
644

645 646 647
      def default_whiny_nils
        false
      end
648

649 650 651 652
      def default_plugins
        []
      end

653 654 655
      def default_plugin_paths
        ["#{root_path}/vendor/plugins"]
      end
656
  end
657
end
658

659
# Needs to be duplicated from Active Support since its needed before Active
660 661 662 663 664 665
# 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

666 667 668 669 670 671 672 673
    if pair = find_pair(key)
      pair.pop
      pair << value
    else
      self << [key, value]
    end
  end

674
  def [](key)
675 676
    pair = find_pair(key.to_sym)
    pair ? pair.last : nil
677 678 679 680 681 682 683 684 685
  end

  def method_missing(name, *args)
    if name.to_s =~ /(.*)=$/
      self[$1.to_sym] = args.first
    else
      self[name]
    end
  end
686 687 688 689 690 691

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