2012-09-14 00:45:18 -04:00
require 'rack/session/abstract/id'
2011-06-29 09:38:09 -04:00
require 'action_controller/metal/exceptions'
2014-10-23 10:00:30 -04:00
require 'active_support/security_utils'
2010-01-31 21:32:28 -05:00
2007-09-22 22:32:55 -04:00
module ActionController #:nodoc:
2007-09-24 19:12:25 -04:00
class InvalidAuthenticityToken < ActionControllerError #:nodoc:
end
2007-09-22 22:32:55 -04:00
2013-12-12 22:41:14 -05:00
class InvalidCrossOriginRequest < ActionControllerError #:nodoc:
end
2010-08-26 17:03:30 -04:00
# Controller actions are protected from Cross-Site Request Forgery (CSRF) attacks
2014-08-08 17:25:18 -04:00
# by including a token in the rendered HTML for your application. This token is
2010-08-26 17:03:30 -04:00
# stored as a random string in the session, to which an attacker does not have
2012-06-07 14:33:38 -04:00
# access. When a request reaches your application, \Rails verifies the received
2012-06-14 13:07:17 -04:00
# token with the token in the session. Only HTML and JavaScript requests are checked,
# so this will not protect your XML API (presumably you'll have a different
2013-12-12 22:41:14 -05:00
# authentication scheme there anyway).
#
# GET requests are not protected since they don't have side effects like writing
# to the database and don't leak sensitive information. JavaScript requests are
# an exception: a third-party site can use a <script> tag to reference a JavaScript
# URL on your site. When your JavaScript response loads on their site, it executes.
# With carefully crafted JavaScript on their end, sensitive data in your JavaScript
# response may be extracted. To prevent this, only XmlHttpRequest (known as XHR or
# Ajax) requests are allowed to make GET requests for JavaScript responses.
2012-06-14 13:07:17 -04:00
#
# It's important to remember that XML or JSON requests are also affected and if
# you're building an API you'll need something like:
2012-06-07 14:19:49 -04:00
#
# class ApplicationController < ActionController::Base
2015-01-04 17:42:25 -05:00
# protect_from_forgery unless: -> { request.format.json? }
2012-06-07 14:19:49 -04:00
# end
2010-08-26 17:03:30 -04:00
#
# CSRF protection is turned on with the <tt>protect_from_forgery</tt> method,
2011-06-30 14:30:38 -04:00
# which checks the token and resets the session if it doesn't match what was expected.
# A call to this method is generated for new \Rails applications by default.
2010-08-26 17:03:30 -04:00
#
# The token parameter is named <tt>authenticity_token</tt> by default. The name and
# value of this token must be added to every layout that renders forms by including
2014-08-08 17:25:18 -04:00
# <tt>csrf_meta_tags</tt> in the HTML +head+.
2010-08-26 17:03:30 -04:00
#
# Learn more about CSRF attacks and securing your application in the
# {Ruby on Rails Security Guide}[http://guides.rubyonrails.org/security.html].
2007-09-22 22:32:55 -04:00
module RequestForgeryProtection
2009-05-28 12:35:36 -04:00
extend ActiveSupport :: Concern
2009-05-21 05:50:34 -04:00
2009-12-20 21:00:04 -05:00
include AbstractController :: Helpers
2010-06-02 16:56:41 -04:00
include AbstractController :: Callbacks
2009-05-21 05:50:34 -04:00
included do
2009-06-17 19:14:05 -04:00
# Sets the token parameter name for RequestForgery. Calling +protect_from_forgery+
# sets it to <tt>:authenticity_token</tt> by default.
2010-04-22 06:00:13 -04:00
config_accessor :request_forgery_protection_token
self . request_forgery_protection_token || = :authenticity_token
2009-05-21 05:50:34 -04:00
2013-02-21 12:16:43 -05:00
# Holds the class which implements the request forgery protection.
config_accessor :forgery_protection_strategy
self . forgery_protection_strategy = nil
2010-06-11 06:15:34 -04:00
# Controls whether request forgery protection is turned on or not. Turned off by default only in test mode.
2010-04-22 06:00:13 -04:00
config_accessor :allow_forgery_protection
self . allow_forgery_protection = true if allow_forgery_protection . nil?
2009-05-21 05:50:34 -04:00
2014-03-04 19:24:51 -05:00
# Controls whether a CSRF failure logs a warning. On by default.
config_accessor :log_warning_on_csrf_failure
self . log_warning_on_csrf_failure = true
2009-05-21 05:50:34 -04:00
helper_method :form_authenticity_token
helper_method :protect_against_forgery?
2007-09-22 22:32:55 -04:00
end
2009-12-20 21:00:04 -05:00
2007-09-22 22:32:55 -04:00
module ClassMethods
2014-07-24 12:57:13 -04:00
# Turn on request forgery protection. Bear in mind that GET and HEAD requests are not checked.
2007-09-24 19:12:25 -04:00
#
2013-12-12 22:41:14 -05:00
# class ApplicationController < ActionController::Base
# protect_from_forgery
# end
#
2007-09-24 13:02:02 -04:00
# class FooController < ApplicationController
2012-10-27 16:05:27 -04:00
# protect_from_forgery except: :index
2007-09-24 13:02:02 -04:00
#
2013-12-12 22:41:14 -05:00
# You can disable CSRF protection on controller by skipping the verification before_action:
2012-12-07 15:24:56 -05:00
# skip_before_action :verify_authenticity_token
2010-11-26 15:56:08 -05:00
#
2007-09-24 13:02:02 -04:00
# Valid Options:
#
2012-12-07 15:24:56 -05:00
# * <tt>:only/:except</tt> - Passed to the <tt>before_action</tt> call. Set which actions are verified.
2015-01-04 17:42:25 -05:00
# * <tt>:if/:unless</tt> - Passed to the <tt>before_action</tt> call. Set when actions are verified.
2012-09-13 05:07:37 -04:00
# * <tt>:with</tt> - Set the method to handle unverified request.
#
# Valid unverified request handling methods are:
# * <tt>:exception</tt> - Raises ActionController::InvalidAuthenticityToken exception.
# * <tt>:reset_session</tt> - Resets the session.
# * <tt>:null_session</tt> - Provides an empty session during request but doesn't reset it completely. Used as default if <tt>:with</tt> option is not specified.
2007-09-23 14:14:44 -04:00
def protect_from_forgery ( options = { } )
2013-02-21 12:16:43 -05:00
self . forgery_protection_strategy = protection_method_class ( options [ :with ] || :null_session )
2007-09-23 14:14:44 -04:00
self . request_forgery_protection_token || = :authenticity_token
2012-12-07 15:24:56 -05:00
prepend_before_action :verify_authenticity_token , options
2013-12-12 22:41:14 -05:00
append_after_action :verify_same_origin_request
2007-09-22 22:32:55 -04:00
end
2012-09-13 05:07:37 -04:00
private
2013-02-21 12:16:43 -05:00
def protection_method_class ( name )
2012-09-13 05:07:37 -04:00
ActionController :: RequestForgeryProtection :: ProtectionMethods . const_get ( name . to_s . classify )
rescue NameError
raise ArgumentError , 'Invalid request forgery protection method, use :null_session, :exception, or :reset_session'
end
end
module ProtectionMethods
2013-02-21 12:16:43 -05:00
class NullSession
def initialize ( controller )
@controller = controller
end
2012-09-13 05:07:37 -04:00
# This is the method that defines the application behavior when a request is found to be unverified.
def handle_unverified_request
2013-02-21 12:16:43 -05:00
request = @controller . request
2013-01-10 22:34:52 -05:00
request . session = NullSessionHash . new ( request . env )
2012-09-13 05:07:37 -04:00
request . env [ 'action_dispatch.request.flash_hash' ] = nil
request . env [ 'rack.session.options' ] = { skip : true }
request . env [ 'action_dispatch.cookies' ] = NullCookieJar . build ( request )
end
2013-02-21 12:16:43 -05:00
protected
2012-09-13 05:07:37 -04:00
class NullSessionHash < Rack :: Session :: Abstract :: SessionHash #:nodoc:
2013-01-10 22:34:52 -05:00
def initialize ( env )
super ( nil , env )
@data = { }
2012-09-13 05:07:37 -04:00
@loaded = true
end
2013-09-18 22:01:59 -04:00
# no-op
def destroy ; end
2012-09-13 05:07:37 -04:00
def exists?
true
end
end
class NullCookieJar < ActionDispatch :: Cookies :: CookieJar #:nodoc:
def self . build ( request )
2012-10-30 23:06:46 -04:00
key_generator = request . env [ ActionDispatch :: Cookies :: GENERATOR_KEY ]
host = request . host
secure = request . ssl?
2012-09-13 05:07:37 -04:00
2013-02-08 13:22:16 -05:00
new ( key_generator , host , secure , options_for_env ( { } ) )
2012-09-13 05:07:37 -04:00
end
def write ( * )
# nothing
end
end
end
2013-02-21 12:16:43 -05:00
class ResetSession
def initialize ( controller )
@controller = controller
end
2012-09-13 05:07:37 -04:00
def handle_unverified_request
2013-02-21 12:16:43 -05:00
@controller . reset_session
2012-09-13 05:07:37 -04:00
end
end
2013-02-21 12:16:43 -05:00
class Exception
def initialize ( controller )
@controller = controller
end
2012-09-13 05:07:37 -04:00
def handle_unverified_request
raise ActionController :: InvalidAuthenticityToken
end
end
2007-09-22 22:32:55 -04:00
end
protected
2013-12-12 22:41:14 -05:00
# The actual before_action that is used to verify the CSRF token.
# Don't override this directly. Provide your own forgery protection
# strategy instead. If you override, you'll disable same-origin
# `<script>` verification.
#
# Lean on the protect_from_forgery declaration to mark which actions are
# due for same-origin request verification. If protect_from_forgery is
# enabled on an action, this before_action flags its after_action to
# verify that JavaScript responses are for XHR requests, ensuring they
# follow the browser's same-origin policy.
def verify_authenticity_token
2013-12-17 18:02:04 -05:00
mark_for_same_origin_verification!
2013-12-12 22:41:14 -05:00
if ! verified_request?
2014-03-04 19:24:51 -05:00
if logger && log_warning_on_csrf_failure
logger . warn " Can't verify CSRF token authenticity "
end
2013-12-12 22:41:14 -05:00
handle_unverified_request
end
end
2013-02-21 12:16:43 -05:00
def handle_unverified_request
2013-02-21 12:25:26 -05:00
forgery_protection_strategy . new ( self ) . handle_unverified_request
2013-02-21 12:16:43 -05:00
end
2014-12-19 18:21:43 -05:00
#:nodoc:
2013-12-12 22:41:14 -05:00
CROSS_ORIGIN_JAVASCRIPT_WARNING = " Security warning: an embedded " \
" <script> tag on another site requested protected JavaScript. " \
" If you know what you're doing, go ahead and disable forgery " \
" protection on this action to permit cross-origin JavaScript embedding. "
private_constant :CROSS_ORIGIN_JAVASCRIPT_WARNING
# If `verify_authenticity_token` was run (indicating that we have
# forgery protection enabled for this request) then also verify that
# we aren't serving an unauthorized cross-origin response.
def verify_same_origin_request
if marked_for_same_origin_verification? && non_xhr_javascript_response?
logger . warn CROSS_ORIGIN_JAVASCRIPT_WARNING if logger
raise ActionController :: InvalidCrossOriginRequest , CROSS_ORIGIN_JAVASCRIPT_WARNING
2011-05-09 11:23:41 -04:00
end
2011-01-04 19:36:07 -05:00
end
2013-12-17 18:02:04 -05:00
# GET requests are checked for cross-origin JavaScript after rendering.
def mark_for_same_origin_verification!
@marked_for_same_origin_verification = request . get?
end
2013-12-12 22:41:14 -05:00
# If the `verify_authenticity_token` before_action ran, verify that
# JavaScript responses are only served to same-origin GET requests.
def marked_for_same_origin_verification?
2013-12-17 18:02:04 -05:00
@marked_for_same_origin_verification || = false
2013-12-12 22:41:14 -05:00
end
# Check for cross-origin JavaScript responses.
def non_xhr_javascript_response?
content_type =~ %r( \ Atext/javascript ) && ! request . xhr?
end
2014-08-19 17:29:26 -04:00
AUTHENTICITY_TOKEN_LENGTH = 32
2011-05-23 19:22:33 -04:00
# Returns true or false if a request is verified. Checks:
2007-09-22 22:32:55 -04:00
#
2013-01-22 16:01:57 -05:00
# * is it a GET or HEAD request? Gets should be safe and idempotent
2008-11-20 17:06:19 -05:00
# * Does the form_authenticity_token match the given token value from the params?
2011-01-04 19:36:07 -05:00
# * Does the X-CSRF-Token header match the form_authenticity_token
2007-09-22 22:32:55 -04:00
def verified_request?
2013-01-22 16:01:57 -05:00
! protect_against_forgery? || request . get? || request . head? ||
2014-08-19 17:29:26 -04:00
valid_authenticity_token? ( session , form_authenticity_param ) ||
valid_authenticity_token? ( session , request . headers [ 'X-CSRF-Token' ] )
2007-09-22 22:32:55 -04:00
end
2009-10-28 03:12:35 -04:00
2009-07-25 11:03:58 -04:00
# Sets the token value for the current session.
2007-09-23 14:14:44 -04:00
def form_authenticity_token
2014-08-19 17:29:26 -04:00
masked_authenticity_token ( session )
end
# Creates a masked version of the authenticity token that varies
# on each request. The masking is used to mitigate SSL attacks
# like BREACH.
def masked_authenticity_token ( session )
one_time_pad = SecureRandom . random_bytes ( AUTHENTICITY_TOKEN_LENGTH )
encrypted_csrf_token = xor_byte_strings ( one_time_pad , real_csrf_token ( session ) )
masked_token = one_time_pad + encrypted_csrf_token
Base64 . strict_encode64 ( masked_token )
end
# Checks the client's masked token to see if it matches the
# session token. Essentially the inverse of
# +masked_authenticity_token+.
def valid_authenticity_token? ( session , encoded_masked_token )
return false if encoded_masked_token . nil? || encoded_masked_token . empty?
begin
masked_token = Base64 . strict_decode64 ( encoded_masked_token )
rescue ArgumentError # encoded_masked_token is invalid Base64
return false
end
# See if it's actually a masked token or not. In order to
# deploy this code, we should be able to handle any unmasked
# tokens that we've issued without error.
if masked_token . length == AUTHENTICITY_TOKEN_LENGTH
# This is actually an unmasked token. This is expected if
# you have just upgraded to masked tokens, but should stop
# happening shortly after installing this gem
compare_with_real_token masked_token , session
elsif masked_token . length == AUTHENTICITY_TOKEN_LENGTH * 2
# Split the token into the one-time pad and the encrypted
# value and decrypt it
one_time_pad = masked_token [ 0 ... AUTHENTICITY_TOKEN_LENGTH ]
encrypted_csrf_token = masked_token [ AUTHENTICITY_TOKEN_LENGTH .. - 1 ]
csrf_token = xor_byte_strings ( one_time_pad , encrypted_csrf_token )
compare_with_real_token csrf_token , session
else
false # Token is malformed
end
end
def compare_with_real_token ( token , session )
2014-10-23 10:00:30 -04:00
ActiveSupport :: SecurityUtils . secure_compare ( token , real_csrf_token ( session ) )
2014-08-19 17:29:26 -04:00
end
def real_csrf_token ( session )
session [ :_csrf_token ] || = SecureRandom . base64 ( AUTHENTICITY_TOKEN_LENGTH )
Base64 . strict_decode64 ( session [ :_csrf_token ] )
end
def xor_byte_strings ( s1 , s2 )
s1 . bytes . zip ( s2 . bytes ) . map { | ( c1 , c2 ) | c1 ^ c2 } . pack ( 'c*' )
2007-09-22 22:32:55 -04:00
end
2008-11-20 17:06:19 -05:00
2009-11-18 02:36:48 -05:00
# The form's authenticity parameter. Override to provide your own.
def form_authenticity_param
params [ request_forgery_protection_token ]
end
2013-05-07 22:29:40 -04:00
# Checks if the controller allows forgery protection.
2007-09-28 11:55:45 -04:00
def protect_against_forgery?
2010-04-22 06:00:13 -04:00
allow_forgery_protection
2007-09-28 11:55:45 -04:00
end
2007-09-22 22:32:55 -04:00
end
2007-12-09 20:15:30 -05:00
end