2012-09-01 17:25:58 -04:00
The Rails Initialization Process
================================
2010-03-27 21:02:59 -04:00
2012-05-22 11:32:12 -04:00
This guide explains the internals of the initialization process in Rails
as of Rails 4. It is an extremely in-depth guide and recommended for advanced Rails developers.
2010-03-27 21:02:59 -04:00
2012-11-29 17:25:02 -05:00
After reading this guide, you will know:
2012-12-07 12:50:09 -05:00
* How to use `rails server` .
2010-03-27 21:02:59 -04:00
2012-09-01 17:25:58 -04:00
--------------------------------------------------------------------------------
2010-03-27 21:02:59 -04:00
2012-06-10 20:55:12 -04:00
This guide goes through every method call that is
2012-06-17 14:11:22 -04:00
required to boot up the Ruby on Rails stack for a default Rails 4
application, explaining each part in detail along the way. For this
guide, we will be focusing on what happens when you execute +rails
server+ to boot your app.
2010-12-14 23:08:08 -05:00
NOTE: Paths in this guide are relative to Rails or a Rails application unless otherwise specified.
2010-03-27 21:02:59 -04:00
2012-09-03 21:21:24 -04:00
TIP: If you want to follow along while browsing the Rails [source
code](https://github.com/rails/rails), we recommend that you use the `t`
2012-08-11 02:19:51 -04:00
key binding to open the file finder inside GitHub and find files
2012-06-17 14:14:19 -04:00
quickly.
2012-09-01 17:25:58 -04:00
Launch!
-------
2010-03-27 21:02:59 -04:00
2013-01-06 18:13:47 -05:00
Now we finally boot and initialize the app. It all starts with your app's
`bin/rails` executable. A Rails application is usually started by running
`rails console` or `rails server` .
2010-03-27 21:02:59 -04:00
2012-09-01 21:37:59 -04:00
### `bin/rails`
2010-12-14 23:08:08 -05:00
2011-11-19 21:36:50 -05:00
This file is as follows:
2010-12-14 23:08:08 -05:00
2012-09-01 17:08:06 -04:00
```ruby
2013-01-06 18:13:47 -05:00
#!/usr/bin/env ruby
2011-06-04 15:28:21 -04:00
APP_PATH = File.expand_path('../../config/application', __FILE__ )
require File.expand_path('../../config/boot', __FILE__ )
require 'rails/commands'
2012-09-01 17:08:06 -04:00
```
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
The `APP_PATH` constant will be used later in `rails/commands` . The `config/boot` file referenced here is the `config/boot.rb` file in our application which is responsible for loading Bundler and setting it up.
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
### `config/boot.rb`
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
`config/boot.rb` contains:
2010-03-27 21:02:59 -04:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
# Set up gems listed in the Gemfile.
2012-05-22 11:48:46 -04:00
ENV['BUNDLE_GEMFILE'] ||= File.expand_path('../../Gemfile', __FILE__ )
require 'bundler/setup' if File.exists?(ENV['BUNDLE_GEMFILE'])
2012-09-01 17:08:06 -04:00
```
2010-03-27 21:02:59 -04:00
2012-09-01 21:37:59 -04:00
In a standard Rails application, there's a `Gemfile` which declares all
dependencies of the application. `config/boot.rb` sets
`ENV['BUNDLE_GEMFILE']` to the location of this file. If the Gemfile
exists, `bundler/setup` is then required.
2012-05-22 11:48:46 -04:00
The gems that a Rails 4 application depends on are as follows:
TODO: change these when the Rails 4 release is near.
2010-12-14 23:08:08 -05:00
* abstract (1.0.0)
2012-05-26 16:09:54 -04:00
* actionmailer (4.0.0.beta)
* actionpack (4.0.0.beta)
* activemodel (4.0.0.beta)
* activerecord (4.0.0.beta)
* activesupport (4.0.0.beta)
2011-05-25 19:13:49 -04:00
* arel (2.0.7)
2010-12-14 23:08:08 -05:00
* builder (3.0.0)
* bundler (1.0.6)
* erubis (2.6.6)
* i18n (0.5.0)
* mail (2.2.12)
* mime-types (1.16)
* polyglot (0.3.1)
* rack (1.2.1)
* rack-cache (0.5.3)
* rack-mount (0.6.13)
* rack-test (0.5.6)
2012-05-26 16:09:54 -04:00
* rails (4.0.0.beta)
* railties (4.0.0.beta)
2010-12-14 23:08:08 -05:00
* rake (0.8.7)
* sqlite3-ruby (1.3.2)
* thor (0.14.6)
* treetop (1.4.9)
* tzinfo (0.3.23)
2012-09-01 21:37:59 -04:00
### `rails/commands.rb`
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
Once `config/boot.rb` has finished, the next file that is required is `rails/commands` which will execute a command based on the arguments passed in. In this case, the `ARGV` array simply contains `server` which is extracted into the `command` variable using these lines:
2010-12-14 23:08:08 -05:00
2012-09-01 17:08:06 -04:00
```ruby
2012-05-22 11:55:59 -04:00
ARGV < < '--help' if ARGV.empty?
2011-06-04 15:28:21 -04:00
aliases = {
"g" => "generate",
2012-05-26 16:09:54 -04:00
"d" => "destroy",
2011-06-04 15:28:21 -04:00
"c" => "console",
"s" => "server",
2011-08-14 10:45:22 -04:00
"db" => "dbconsole",
"r" => "runner"
2011-06-04 15:28:21 -04:00
}
2010-12-14 23:08:08 -05:00
2011-06-04 15:28:21 -04:00
command = ARGV.shift
command = aliases[command] || command
2012-09-01 17:08:06 -04:00
```
2010-12-14 23:08:08 -05:00
2012-05-22 11:55:59 -04:00
TIP: As you can see, an empty ARGV list will make Rails show the help
snippet.
2012-09-01 21:37:59 -04:00
If we used `s` rather than `server` , Rails will use the `aliases` defined in the file and match them to their respective commands. With the `server` command, Rails will run this code:
2010-12-14 23:08:08 -05:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
when 'server'
# Change to the application's path if there is no config.ru file in current dir.
2013-01-06 18:13:47 -05:00
# This allows us to run `rails server` from other directories, but still get
2011-06-04 15:28:21 -04:00
# the main config.ru and properly set the tmp directory.
Dir.chdir(File.expand_path('../../', APP_PATH)) unless File.exists?(File.expand_path("config.ru"))
require 'rails/commands/server'
2013-01-05 17:57:03 -05:00
Rails::Server.new.tap do |server|
2011-06-04 15:28:21 -04:00
# We need to require application after the server sets environment,
# otherwise the --environment option given to the server won't propagate.
require APP_PATH
Dir.chdir(Rails.application.root)
server.start
2013-01-05 17:57:03 -05:00
end
2012-09-01 17:08:06 -04:00
```
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
This file will change into the root of the directory (a path two directories back from `APP_PATH` which points at `config/application.rb` ), but only if the `config.ru` file isn't found. This then requires `rails/commands/server` which sets up the `Rails::Server` class.
2012-06-09 15:20:47 -04:00
2012-09-01 17:08:06 -04:00
```ruby
2012-06-09 15:20:47 -04:00
require 'fileutils'
require 'optparse'
require 'action_dispatch'
module Rails
class Server < ::Rack::Server
2012-09-01 17:08:06 -04:00
```
2012-06-09 15:20:47 -04:00
2012-09-01 21:37:59 -04:00
`fileutils` and `optparse` are standard Ruby libraries which provide helper functions for working with files and parsing options.
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
### `actionpack/lib/action_dispatch.rb`
2010-12-14 23:08:08 -05:00
2012-06-17 14:30:20 -04:00
Action Dispatch is the routing component of the Rails framework.
It adds functionalities like routing, session, and common middlewares.
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
### `rails/commands/server.rb`
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
The `Rails::Server` class is defined in this file as inheriting from `Rack::Server` . When `Rails::Server.new` is called, this calls the `initialize` method in `rails/commands/server.rb` :
2010-03-27 21:02:59 -04:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
def initialize(*)
super
set_environment
end
2012-09-01 17:08:06 -04:00
```
2010-03-27 21:02:59 -04:00
2012-09-01 21:37:59 -04:00
Firstly, `super` is called which calls the `initialize` method on `Rack::Server` .
2010-03-27 21:02:59 -04:00
2012-09-01 21:37:59 -04:00
### Rack: `lib/rack/server.rb`
2010-03-27 21:02:59 -04:00
2012-09-01 21:37:59 -04:00
`Rack::Server` is responsible for providing a common server interface for all Rack-based applications, which Rails is now a part of.
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
The `initialize` method in `Rack::Server` simply sets a couple of variables:
2010-12-14 23:08:08 -05:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
def initialize(options = nil)
@options = options
@app = options[:app] if options && options[:app]
end
2012-09-01 17:08:06 -04:00
```
2010-03-27 21:02:59 -04:00
2012-09-01 21:37:59 -04:00
In this case, `options` will be `nil` so nothing happens in this method.
2010-03-27 21:02:59 -04:00
2012-09-01 21:37:59 -04:00
After `super` has finished in `Rack::Server` , we jump back to `rails/commands/server.rb` . At this point, `set_environment` is called within the context of the `Rails::Server` object and this method doesn't appear to do much at first glance:
2010-03-27 21:02:59 -04:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
def set_environment
ENV["RAILS_ENV"] ||= options[:environment]
end
2012-09-01 17:08:06 -04:00
```
2010-03-31 05:50:13 -04:00
2012-09-01 21:37:59 -04:00
In fact, the `options` method here does quite a lot. This method is defined in `Rack::Server` like this:
2010-03-31 05:50:13 -04:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
def options
@options ||= parse_options(ARGV)
end
2012-09-01 17:08:06 -04:00
```
2010-03-27 21:02:59 -04:00
2012-09-01 21:37:59 -04:00
Then `parse_options` is defined like this:
2010-03-27 21:02:59 -04:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
def parse_options(args)
options = default_options
2010-03-31 05:50:13 -04:00
2011-06-04 15:28:21 -04:00
# Don't evaluate CGI ISINDEX parameters.
# http://hoohoo.ncsa.uiuc.edu/cgi/cl.html
args.clear if ENV.include?("REQUEST_METHOD")
2010-03-27 21:02:59 -04:00
2011-06-04 15:28:21 -04:00
options.merge! opt_parser.parse! args
options[:config] = ::File.expand_path(options[:config])
ENV["RACK_ENV"] = options[:environment]
options
end
2012-09-01 17:08:06 -04:00
```
2010-03-27 21:02:59 -04:00
2012-09-01 21:37:59 -04:00
With the `default_options` set to this:
2010-03-27 21:02:59 -04:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
def default_options
{
:environment => ENV['RACK_ENV'] || "development",
:pid => nil,
:Port => 9292,
:Host => "0.0.0.0",
:AccessLog => [],
:config => "config.ru"
}
end
2012-09-01 17:08:06 -04:00
```
2010-03-27 21:02:59 -04:00
2012-09-01 21:37:59 -04:00
There is no `REQUEST_METHOD` key in `ENV` so we can skip over that line. The next line merges in the options from `opt_parser` which is defined plainly in `Rack::Server`
2010-03-27 21:02:59 -04:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
def opt_parser
Options.new
end
2012-09-01 17:08:06 -04:00
```
2010-03-27 21:02:59 -04:00
2012-09-06 22:26:59 -04:00
The class **is** defined in `Rack::Server` , but is overwritten in `Rails::Server` to take different arguments. Its `parse!` method begins like this:
2010-03-27 21:02:59 -04:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
def parse!(args)
args, options = args.dup, {}
2010-12-14 23:08:08 -05:00
2011-08-05 15:31:54 -04:00
opt_parser = OptionParser.new do |opts|
opts.banner = "Usage: rails server [mongrel, thin, etc] [options]"
opts.on("-p", "--port=port", Integer,
"Runs Rails on the specified port.", "Default: 3000") { |v| options[:Port] = v }
2010-03-31 05:50:13 -04:00
...
2012-09-01 17:08:06 -04:00
```
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
This method will set up keys for the `options` which Rails will then be
able to use to determine how its server should run. After `initialize`
has finished, we jump back into `rails/server` where `APP_PATH` (which was
2012-06-10 21:20:40 -04:00
set earlier) is required.
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
### `config/application`
2012-06-10 21:26:54 -04:00
2012-09-01 21:37:59 -04:00
When `require APP_PATH` is executed, `config/application.rb` is loaded.
2012-08-11 02:19:51 -04:00
This file exists in your app and it's free for you to change based
2012-06-17 14:30:20 -04:00
on your needs.
2012-06-10 21:26:54 -04:00
2012-09-01 21:37:59 -04:00
### `Rails::Server#start`
2010-12-14 23:08:08 -05:00
2012-10-09 21:19:02 -04:00
After `config/application` is loaded, `server.start` is called. This method is defined like this:
2010-12-14 23:08:08 -05:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
def start
2012-05-23 11:45:48 -04:00
url = "#{options[:SSLEnable] ? 'https' : 'http'}://#{options[:Host]}:#{options[:Port]}"
2011-06-04 15:28:21 -04:00
puts "=> Booting #{ActiveSupport::Inflector.demodulize(server)}"
2012-05-23 11:45:48 -04:00
puts "=> Rails #{Rails.version} application starting in #{Rails.env} on #{url}"
2013-03-18 13:42:24 -04:00
puts "=> Run `rails server -h` for more startup options"
2011-06-04 15:28:21 -04:00
trap(:INT) { exit }
puts "=> Ctrl-C to shutdown server" unless options[:daemonize]
#Create required tmp directories if not found
%w(cache pids sessions sockets).each do |dir_to_make|
FileUtils.mkdir_p(Rails.root.join('tmp', dir_to_make))
2010-03-31 05:50:13 -04:00
end
2011-06-04 15:28:21 -04:00
2012-05-23 11:45:48 -04:00
unless options[:daemonize]
wrapped_app # touch the app so the logger is set up
console = ActiveSupport::Logger.new($stdout)
console.formatter = Rails.logger.formatter
Rails.logger.extend(ActiveSupport::Logger.broadcast(console))
end
2011-06-04 15:28:21 -04:00
super
ensure
# The '-h' option calls exit before @options is set.
# If we call 'options' with it unset, we get double help banners.
puts 'Exiting' unless @options && options[:daemonize]
end
2012-09-01 17:08:06 -04:00
```
2010-03-27 21:02:59 -04:00
2012-05-23 11:45:48 -04:00
This is where the first output of the Rails initialization happens. This
2012-09-01 21:37:59 -04:00
method creates a trap for `INT` signals, so if you `CTRL-C` the server,
2012-05-23 11:45:48 -04:00
it will exit the process. As we can see from the code here, it will
2012-09-01 21:37:59 -04:00
create the `tmp/cache` , `tmp/pids` , `tmp/sessions` and `tmp/sockets`
directories. It then calls `wrapped_app` which is responsible for
2012-06-10 21:45:03 -04:00
creating the Rack app, before creating and assigning an
2012-09-01 21:37:59 -04:00
instance of `ActiveSupport::Logger` .
2012-05-23 11:45:48 -04:00
2012-09-01 21:37:59 -04:00
The `super` method will call `Rack::Server.start` which begins its definition like this:
2010-03-27 21:02:59 -04:00
2012-09-01 17:08:06 -04:00
```ruby
2012-05-23 11:53:08 -04:00
def start & blk
2011-06-04 15:28:21 -04:00
if options[:warn]
$-w = true
end
2010-12-14 23:08:08 -05:00
2011-06-04 15:28:21 -04:00
if includes = options[:include]
$LOAD_PATH.unshift(*includes)
end
2010-12-14 23:08:08 -05:00
2011-06-04 15:28:21 -04:00
if library = options[:require]
require library
end
2010-12-14 23:08:08 -05:00
2011-06-04 15:28:21 -04:00
if options[:debug]
$DEBUG = true
require 'pp'
p options[:server]
pp wrapped_app
pp app
end
2010-03-27 21:02:59 -04:00
2012-05-23 11:53:08 -04:00
check_pid! if options[:pid]
2010-03-27 21:02:59 -04:00
2012-05-23 11:53:08 -04:00
# Touch the wrapped app, so that the config.ru is loaded before
# daemonization (i.e. before chdir, etc).
wrapped_app
daemonize_app if options[:daemonize]
write_pid if options[:pid]
trap(:INT) do
if server.respond_to?(:shutdown)
server.shutdown
else
exit
end
end
server.run wrapped_app, options, & blk
end
2012-09-01 17:08:06 -04:00
```
2010-03-27 21:02:59 -04:00
2012-09-01 21:37:59 -04:00
The interesting part for a Rails app is the last line, `server.run` . Here we encounter the `wrapped_app` method again, which this time
2012-06-10 21:45:03 -04:00
we're going to explore more (even though it was executed before, and
2012-08-11 02:19:51 -04:00
thus memorized by now).
2010-03-27 21:02:59 -04:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
@wrapped_app ||= build_app app
2012-09-01 17:08:06 -04:00
```
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
The `app` method here is defined like so:
2010-12-14 23:08:08 -05:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
def app
@app ||= begin
if !::File.exist? options[:config]
abort "configuration #{options[:config]} not found"
2010-03-27 21:02:59 -04:00
end
2011-06-04 15:28:21 -04:00
app, options = Rack::Builder.parse_file(self.options[:config], opt_parser)
self.options.merge! options
app
2010-12-14 23:08:08 -05:00
end
2011-06-04 15:28:21 -04:00
end
2012-09-01 17:08:06 -04:00
```
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
The `options[:config]` value defaults to `config.ru` which contains this:
2010-12-14 23:08:08 -05:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
# This file is used by Rack-based servers to start the application.
2010-12-14 23:08:08 -05:00
2011-06-04 15:28:21 -04:00
require ::File.expand_path('../config/environment', __FILE__ )
2012-05-23 12:01:21 -04:00
run < %= app_const %>
2012-09-01 17:08:06 -04:00
```
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
The `Rack::Builder.parse_file` method here takes the content from this `config.ru` file and parses it using this code:
2010-12-14 23:08:08 -05:00
2012-09-01 17:08:06 -04:00
```ruby
2012-09-03 21:21:24 -04:00
app = eval "Rack::Builder.new {( " + cfgfile + "\n )}.to_app",
2010-12-14 23:08:08 -05:00
TOPLEVEL_BINDING, config
2012-09-01 17:08:06 -04:00
```
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
The `initialize` method of `Rack::Builder` will take the block here and execute it within an instance of `Rack::Builder` . This is where the majority of the initialization process of Rails happens. The `require` line for `config/environment.rb` in `config.ru` is the first to run:
2010-12-14 23:08:08 -05:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
require ::File.expand_path('../config/environment', __FILE__ )
2012-09-01 17:08:06 -04:00
```
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
### `config/environment.rb`
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
This file is the common file required by `config.ru` (`rails server`) and Passenger. This is where these two ways to run the server meet; everything before this point has been Rack and Rails setup.
2010-03-27 21:02:59 -04:00
2012-09-01 21:37:59 -04:00
This file begins with requiring `config/application.rb` .
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
### `config/application.rb`
2010-12-14 23:08:08 -05:00
2012-09-06 22:26:59 -04:00
This file requires `config/boot.rb` , but only if it hasn't been required before, which would be the case in `rails server` but **wouldn't** be the case with Passenger.
2010-12-14 23:08:08 -05:00
2011-04-13 20:49:14 -04:00
Then the fun begins!
2010-12-22 23:35:59 -05:00
2012-09-01 17:25:58 -04:00
Loading Rails
-------------
2010-12-22 23:35:59 -05:00
2012-09-01 21:37:59 -04:00
The next line in `config/application.rb` is:
2010-12-14 23:08:08 -05:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
require 'rails/all'
2012-09-01 17:08:06 -04:00
```
2010-12-14 23:08:08 -05:00
2012-09-01 21:37:59 -04:00
### `railties/lib/rails/all.rb`
2010-12-14 23:08:08 -05:00
2012-06-17 14:30:20 -04:00
This file is responsible for requiring all the individual frameworks of Rails:
2010-12-14 23:08:08 -05:00
2012-09-01 17:08:06 -04:00
```ruby
2011-06-04 15:28:21 -04:00
require "rails"
2010-12-14 23:08:08 -05:00
2011-06-04 15:28:21 -04:00
%w(
2010-12-14 23:08:08 -05:00
active_record
action_controller
action_mailer
rails/test_unit
2012-10-18 00:03:24 -04:00
sprockets
2011-06-04 15:28:21 -04:00
).each do |framework|
begin
require "#{framework}/railtie"
rescue LoadError
2010-03-27 21:02:59 -04:00
end
2011-06-04 15:28:21 -04:00
end
2012-09-01 17:08:06 -04:00
```
2010-03-27 21:02:59 -04:00
2012-06-10 21:45:03 -04:00
This is where all the Rails frameworks are loaded and thus made
2012-08-11 02:19:51 -04:00
available to the application. We won't go into detail of what happens
2012-06-10 21:45:03 -04:00
inside each of those frameworks, but you're encouraged to try and
explore them on your own.
2010-12-14 23:08:08 -05:00
2012-06-10 21:45:03 -04:00
For now, just keep in mind that common functionality like Rails engines,
2012-08-11 02:19:51 -04:00
I18n and Rails configuration is all being defined here.
2012-06-13 11:34:48 -04:00
2012-09-01 21:37:59 -04:00
### Back to `config/environment.rb`
2012-06-13 11:34:48 -04:00
2012-09-01 21:37:59 -04:00
When `config/application.rb` has finished loading Rails, and defined
your application namespace, you go back to `config/environment.rb` ,
2012-06-13 11:34:48 -04:00
where your application is initialized. For example, if you application was called
2012-09-01 21:37:59 -04:00
`Blog` , here you would find `Blog::Application.initialize!` , which is
defined in `rails/application.rb`
2012-06-13 11:48:45 -04:00
2012-09-01 21:37:59 -04:00
### `railties/lib/rails/application.rb`
2012-06-13 11:48:45 -04:00
2012-09-01 21:37:59 -04:00
The `initialize!` method looks like this:
2012-06-13 11:48:45 -04:00
2012-09-01 17:08:06 -04:00
```ruby
2012-06-13 11:48:45 -04:00
def initialize!(group=:default) #:nodoc:
raise "Application has been already initialized." if @initialized
run_initializers(group, self)
@initialized = true
self
end
2012-09-01 17:08:06 -04:00
```
2012-06-13 11:48:45 -04:00
As you can see, you can only initialize an app once. This is also where the initializers are run.
TODO: review this
The initializers code itself is tricky. What Rails is doing here is it
2012-09-01 21:37:59 -04:00
traverses all the class ancestors looking for an `initializers` method,
sorting them and running them. For example, the `Engine` class will make
all the engines available by providing the `initializers` method.
2012-06-13 11:48:45 -04:00
2012-09-01 21:37:59 -04:00
After this is done we go back to `Rack::Server`
2012-06-13 11:48:45 -04:00
2012-09-01 17:25:58 -04:00
### Rack: lib/rack/server.rb
2012-06-13 11:59:14 -04:00
2012-09-01 21:37:59 -04:00
Last time we left when the `app` method was being defined:
2012-06-13 11:59:14 -04:00
2012-09-01 17:08:06 -04:00
```ruby
2012-06-13 11:59:14 -04:00
def app
@app ||= begin
if !::File.exist? options[:config]
abort "configuration #{options[:config]} not found"
end
app, options = Rack::Builder.parse_file(self.options[:config], opt_parser)
self.options.merge! options
app
end
end
2012-09-01 17:08:06 -04:00
```
2012-06-13 11:59:14 -04:00
2012-09-01 21:37:59 -04:00
At this point `app` is the Rails app itself (a middleware), and what
2012-06-13 11:59:14 -04:00
happens next is Rack will call all the provided middlewares:
2012-09-01 17:08:06 -04:00
```ruby
2012-06-13 11:59:14 -04:00
def build_app(app)
middleware[options[:environment]].reverse_each do |middleware|
middleware = middleware.call(self) if middleware.respond_to?(:call)
next unless middleware
klass = middleware.shift
app = klass.new(app, *middleware)
end
app
end
2012-09-01 17:08:06 -04:00
```
2012-06-13 11:59:14 -04:00
2012-09-01 21:37:59 -04:00
Remember, `build_app` was called (by wrapped_app) in the last line of `Server#start` .
2012-06-13 11:59:14 -04:00
Here's how it looked like when we left:
2012-09-01 17:08:06 -04:00
```ruby
2012-06-13 11:59:14 -04:00
server.run wrapped_app, options, & blk
2012-09-01 17:08:06 -04:00
```
2012-06-13 11:59:14 -04:00
2012-09-01 21:37:59 -04:00
At this point, the implementation of `server.run` will depend on the
2012-06-14 12:05:10 -04:00
server you're using. For example, if you were using Mongrel, here's what
2012-09-01 21:37:59 -04:00
the `run` method would look like:
2012-06-14 12:05:10 -04:00
2012-09-01 17:08:06 -04:00
```ruby
2012-06-14 12:05:10 -04:00
def self.run(app, options={})
server = ::Mongrel::HttpServer.new(
options[:Host] || '0.0.0.0',
options[:Port] || 8080,
options[:num_processors] || 950,
options[:throttle] || 0,
options[:timeout] || 60)
# Acts like Rack::URLMap, utilizing Mongrel's own path finding methods.
# Use is similar to #run , replacing the app argument with a hash of
# { path=>app, ... } or an instance of Rack::URLMap.
if options[:map]
if app.is_a? Hash
app.each do |path, appl|
path = '/'+path unless path[0] == ?/
server.register(path, Rack::Handler::Mongrel.new(appl))
end
elsif app.is_a? URLMap
app.instance_variable_get(:@mapping).each do |(host, path, appl)|
next if !host.nil? & & !options[:Host].nil? & & options[:Host] != host
path = '/'+path unless path[0] == ?/
server.register(path, Rack::Handler::Mongrel.new(appl))
end
else
raise ArgumentError, "first argument should be a Hash or URLMap"
end
else
server.register('/', Rack::Handler::Mongrel.new(app))
end
yield server if block_given?
server.run.join
end
2012-09-01 17:08:06 -04:00
```
2012-06-14 12:05:10 -04:00
2012-08-11 02:19:51 -04:00
We won't dig into the server configuration itself, but this is
2012-06-14 12:05:10 -04:00
the last piece of our journey in the Rails initialization process.
2012-08-11 02:19:51 -04:00
This high level overview will help you understand when your code is
2012-06-14 12:05:10 -04:00
executed and how, and overall become a better Rails developer. If you
still want to know more, the Rails source code itself is probably the
best place to go next.