HTTP Request/Response Lifecycle
Summary
Browser
The user types in: https://hamax97.github.io/home
:
-
Hostname resolution:
- If the browser has in its cache the ip address for
hamax97.github.io
, it uses it; - Otherwise, it issues a DNS lookup.
- If the browser has in its cache the ip address for
-
TODO: when and how does the TLS happen?
-
Once the browser has the ip address it creates a TCP connection with the web server. It then starts speaking the HTTP protocol. To test this locally, start your web application and run:
telnet localhost 3000 # Trying 127.0.0.1... # Connected to localhost. # Escape character is '^]'. > GET /articles HTTP/1.1 > Host: localhost # you need two end-of-lines above, but not this line.
Or you can try to a real web page:
printf "%s\r\n" \ "GET /questions HTTP/1.1" \ "Host: stackoverflow.com" \ "" | nc -v stackoverflow.com 443
Using the headers
Connection: Keep-Alive
andKeep-Alive: timeout=5, max=1000
you can manage the lifecyle of the connection.
DNS lookup
TODO: From my notes, create page: How DNS works?
The ip address is returned to the browser if found: 51.52.53.91
.
Web server
- Useful to handle hundreds of thousands of connections at a time.
- Useful to serve static files.
- Useful to work as reverse proxy.
- Useful to handle SSL handshakes.
- NGINX for example.
App server
Useful for:
- Communication with the web framework, for example, Rails.
- To hand over complex requests to the web framework, the ones that are not just serving static content.
- To handle thousands of connections.
- To handle SSL handshakes.
Examples: PUMA / Guinicorn / …
How an app server communicates with Rails?
There are multiple ways in which this could be done in Ruby. Read until the end to see what is the standard option servers use, Rack.
-
Rails could register a block with the web server:
server.on_request do |request, response| request.path # => "/hello" request.headers # => { host: ... } response.status = 200 response.body = "Hello World" end
-
The server could call a method on Rails:
MyApp.handle_request( http_method, # => "GET" http_path, # => "/hello", http_headers # => #<Headers @host=...> ) # => #<Response @status=200 @body="Hello World">
-
The server could place the request information in environment variables:
ENV["REQUEST_INFO"] ENV["PATH_INFO"] ENV["HTTP_HOST"] puts "Status: 200 OK" puts puts "Hello World"
-
The standard option, Rack: A unified API to present a way for web servers to communicate with Ruby web frameworks.
Rack
Rack is a protocol. It states that a web application is an object (commonly called app
) that:
- Responds to the
call
message. - The
call
message receives theenv
hash.-
This hash looks like:
env = { "REQUEST_METHOD" => "GET", "PATH_INFO" => "/hello", "HTTP_HOST" => "hamax97.github.io", # ... }
-
-
The
call
message returns an array with three things in order (a tuple):status
,headers
, andbody
. Like this:status, headers, body = app.call(env) status # => 200 headers # => { "Content-Type" => "text/plain" } body # => ["Hello World"]
For technical reasons the
body
is aneach
able object, not just a plain string.TODO: Find out why
body
needs to be aneach
able object. Perhaps for streaming responses?
Rackup
Interface (wrapper) for running Rack applications. It provides:
- A way to add middlewares to your application.
- A wrapper around multiple application servers, for example, PUMA or WEBrick.
How to use?
Install:
gem install rackup
This is a very simple Rack app:
# app.rb
class HelloWorld
def call(env)
if env["PATH_INFO"] == "/hello"
[200, { "content-type" => "text/plain" }, ["Hello World"]]
elsif env["PATH_INFO"] == "/"
[301, { "location" => "/hello" }, []]
else
[404, { "content-type" => "text/plain" }, ["Not Found"]]
end
end
end
- Note there’s no subclassing from Rack or something like that. Remember, Rack is just a protocol/specification.
- Also, note that it would be very hard to maintain this if statement in larger applications.
Create a config.ru
file:
# config.rb
require_relative "app"
run HelloWorld.new
Run:
rackup
- This command will read the
config.ru
file and start the server.
Middlewares
How about extracting the redirect functionality from the HelloWorld
app so that it can be
reused?
class Redirect
def initialize(app, from:, to:)
@app = app
@from = from
@to = to
end
def call(env)
if env["PATH_INFO"] == @from
[301, { "location" => @to }, []]
else
@app.call(env)
end
end
end
class HelloWorld
def call(env)
if env["PATH_INFO"] == "/hello"
[200, { "content-type" => "text/plain" }, ["Hello World"]]
else
[404, { "content-type" => "text/plain" }, ["Not Found"]]
end
end
end
And the config.ru
:
require_relative "app"
run Redirect.new(
HelloWorld.new,
from: "/",
to: "/hello"
)
- Now you can reuse the
Redirect
functionality. This is called a middleware.
The middleware is not a concept in the Rack specification itself, rather, it’s a useful technique that has been broadly adopted.
Note, though, that it would get really complicated to nest the HelloWorld.new
inside more and more
functionalities. For this, Rackup provides use
in its DSL:
require_relative "app"
use Redirect, from: "/", to: "/hello"
run HelloWorld.new
Rack provides some useful middlewares.
App - Rails
Rails.application
is a Rack app
Rails generated a config.ru
file when you scaffolded your application. It looks like:
# This file is used by Rack-based servers to start the application.
require_relative "config/environment"
run Rails.application
Rails.application.load_server
Rails.application
must be an object that follows the Rack specification (a Rack app). Let’s try using the
Rails’ console:
Run:
bin/rails console
Then:
> env = Rack::MockRequest.env_for("http://localhost:3000/articles")
> env
# =>
# {"rack.version"=>[1, 3],
# "rack.input"=>#<StringIO:0x00007f1d6e414a50>,
# ...
# "REQUEST_METHOD"=>"GET",
# "SERVER_NAME"=>"localhost",
# "SERVER_PORT"=>"3000",
# "QUERY_STRING"=>"",
# "PATH_INFO"=>"/articles",
# ...}
> status, headers, body = Rails.application.call(env)
# Started GET "/articles" for at 2023-06-16 12:34:00 -0500
# Processing by ArticlesController#index as HTML
# Rendering layout layouts/application.html.erb
# Rendering articles/index.html.erb within layouts/application
# ...
# Completed 200 OK in 5ms (Views: 4.3ms | ActiveRecord: 0.2ms | Allocations: 2943)
> status
# => 200
> headers
# =>
# {"X-Frame-Options"=>"SAMEORIGIN",
# "X-XSS-Protection"=>"0",
# ...
# "Set-Cookie"=>
# "_learn_rails_session=...; path=/; HttpOnly; SameSite=Lax",
# ...
# "Server-Timing"=>
# "start_processing.action_controller;dur=0.07, sql.active_record;dur=0.35, instantiation.active_record;dur=0.09, render_template.action_view;dur=1.88, render_layout.action_view;dur=3.85, process_action.action_controller;dur=4.90"}
> puts body.join("")
# <!DOCTYPE html>
# <html>
# <head>
# <title>LearnRails</title>
# <meta name="viewport"
# ...
How about middlewares?
In the config.ru
file we don’t see any use
keyword. So, where are the middlewares?
Rails handles middlewares differently. To see the list of middlwares used run:
bin/rails middleware
# use ActionDispatch::HostAuthorization
# use Rack::Sendfile
# use ActionDispatch::Static
# use ActionDispatch::Executor
# use ActionDispatch::ServerTiming
# use ActiveSupport::Cache::Strategy::LocalCache::Middleware
# use Rack::Runtime
# use Rack::MethodOverride
# ...
If you want you can remove middlewares if you don’t need them, or you can add your own.
Go to your config/application.rb
file:
# Disable cookies middleware.
config.middleware.delete ActionDispatch::Cookies
config.middleware.delete ActionDispatch::Session::CookieStore
config.middleware.delete ActionDispatch::Flash
# Add your own middleware.
config.middleware.use CaptchaEverywhere
How does a request get to a controller’s action? - The Router
The last output line of the command bin/rails middleware
is something like run LearnRails::Application.routes
:
bin/rails middleware
# use ...
# run LearnRails::Application.routes
LearnRails::Application.routes
should be a Rack app:
> env = Rack::MockRequest.env_for("http://localhost:3000/articles")
> status, headers, body = LearnRails::Application.routes.call(env)
# Processing by ArticlesController#index as HTML
# ...
# Completed 200 OK in 94ms (Views: 69.9ms | ActiveRecord: 2.0ms | Allocations: 56523)
It is! It matches the url path against a set of rules in your config/routes.rb
:
Rails.application.routes.draw do
root "articles#index"
get "/articles" => "articles#index"
get "/articles/new" => "articles#new"
post "/articles" => "articles#create"
end
Each of these routes expands to: ArticlesController.action(:index)
or :new
or :create
instead of
:index
.
ArticlesController.action
returns a lambda that accepts the env
hash. Remember lambdas respond
to .call
, therefore, this lambda returned is a Rack app.
Want to look at Rail’s source code? Look at the
action
class method here.
Therefore, this Rails app could be thought as the following Rack app:
class LearnRails
def call(env)
verb = env["REQUEST_METHOD"]
path = env["PATH_INFO"]
if verb == "GET" && path == "/articles"
ArticlesController.action(:index).call(env)
elsif verb == "POST" && path == "/articles"
ArticlesController.action(:create).call(env)
elsif # ...
else
[404, {"content-type": "text-plain", ...}, ["Not Found"]]
end
end
end
For more details on how the Rails Router works, see RailsConf 2018: Re-graphing The Mental Model of The Rails Router.
How to make the Router point to whatever app you want?
-
You can match your route to a Rack app instead of specifying the controller and action in a string:
Rails.application.routes.draw do get "/articles" => HelloWorld.new get "/other-thing" => ->(env) { [200, {"content-type": "text-plain"}, ["Hello World!"]] } # redirect(...) returns a Rack app! get "/" => redirect("/articles") end
-
You can mount a Sinatra app in your Rails app:
Rails.application.routes.draw do # Sidekiq's web app was built with Sinatra. mount Sidekiq::Web, at: "/sidekiq" end
-
You can use
ArticlesController.action(:your_action)
directly. This is NOT recommended, though. It skips performance optimizations and the auto loader:Rails.application.routes.draw do get "/articles" => ArticlesController.action(:index) get "/articles/new" => ArticlesController.action(:new) post "/articles" => ArticlesController.action(:create) end