2010-01-31 21:32:28 -05:00
|
|
|
require 'active_support/core_ext/class/attribute'
|
2011-06-29 09:38:09 -04:00
|
|
|
require 'action_controller/metal/exceptions'
|
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
|
|
|
|
2010-08-26 17:03:30 -04:00
|
|
|
# Controller actions are protected from Cross-Site Request Forgery (CSRF) attacks
|
|
|
|
# by including a token in the rendered html for your application. This token is
|
|
|
|
# stored as a random string in the session, to which an attacker does not have
|
2011-06-30 14:30:38 -04:00
|
|
|
# access. When a request reaches your application, \Rails verifies the received
|
|
|
|
# token with the token in the session. Only HTML and JavaScript requests are checked,
|
2010-08-26 17:03:30 -04:00
|
|
|
# so this will not protect your XML API (presumably you'll have a different
|
|
|
|
# authentication scheme there anyway). Also, GET requests are not protected as these
|
|
|
|
# should be idempotent.
|
|
|
|
#
|
|
|
|
# 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
|
2010-09-11 05:04:19 -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
|
|
|
|
2012-03-09 11:33:06 -05:00
|
|
|
# Controls how unverified request will be handled
|
|
|
|
config_accessor :request_forgery_protection_method
|
|
|
|
self.request_forgery_protection_method ||= :reset_session
|
|
|
|
|
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
|
|
|
|
|
|
|
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
|
2008-02-06 13:50:48 -05:00
|
|
|
# Turn on request forgery protection. Bear in mind that only non-GET, HTML/JavaScript requests are checked.
|
2007-09-24 19:12:25 -04:00
|
|
|
#
|
2007-09-24 13:02:02 -04:00
|
|
|
# Example:
|
|
|
|
#
|
|
|
|
# class FooController < ApplicationController
|
|
|
|
# protect_from_forgery :except => :index
|
|
|
|
#
|
2010-11-26 15:56:08 -05:00
|
|
|
# You can disable csrf protection on controller-by-controller basis:
|
|
|
|
#
|
|
|
|
# skip_before_filter :verify_authenticity_token
|
|
|
|
#
|
|
|
|
# It can also be disabled for specific controller actions:
|
|
|
|
#
|
|
|
|
# skip_before_filter :verify_authenticity_token, :except => [:create]
|
2007-09-24 13:02:02 -04:00
|
|
|
#
|
|
|
|
# Valid Options:
|
|
|
|
#
|
2011-05-23 19:22:33 -04:00
|
|
|
# * <tt>:only/:except</tt> - Passed to the <tt>before_filter</tt> call. Set which actions are verified.
|
2012-03-09 11:33:06 -05:00
|
|
|
# * <tt>:with</tt> - Set the method to handle unverified request. Valid values: <tt>:exception</tt> and <tt>:reset_session</tt> (default).
|
2007-09-23 14:14:44 -04:00
|
|
|
def protect_from_forgery(options = {})
|
|
|
|
self.request_forgery_protection_token ||= :authenticity_token
|
2012-03-09 11:33:06 -05:00
|
|
|
self.request_forgery_protection_method = options.delete(:with) if options.key?(:with)
|
2011-02-22 14:59:13 -05:00
|
|
|
prepend_before_filter :verify_authenticity_token, options
|
2007-09-22 22:32:55 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
protected
|
2011-05-23 19:22:33 -04:00
|
|
|
# The actual before_filter that is used. Modify this to change how you handle unverified requests.
|
2007-09-23 14:14:44 -04:00
|
|
|
def verify_authenticity_token
|
2011-05-09 11:23:41 -04:00
|
|
|
unless verified_request?
|
2012-01-05 13:12:58 -05:00
|
|
|
logger.warn "Can't verify CSRF token authenticity" if logger
|
2011-05-09 11:23:41 -04:00
|
|
|
handle_unverified_request
|
|
|
|
end
|
2011-01-04 19:36:07 -05:00
|
|
|
end
|
|
|
|
|
2011-07-24 06:21:42 -04:00
|
|
|
# This is the method that defines the application behavior when a request is found to be unverified.
|
2012-03-09 11:33:06 -05:00
|
|
|
# By default, \Rails uses <tt>request_forgery_protection_method</tt> when it finds an unverified request:
|
|
|
|
#
|
|
|
|
# * <tt>:reset_session</tt> - Resets the session.
|
|
|
|
# * <tt>:exception</tt>: - Raises ActionController::InvalidAuthenticityToken exception.
|
2011-01-04 19:36:07 -05:00
|
|
|
def handle_unverified_request
|
2012-03-09 11:33:06 -05:00
|
|
|
case request_forgery_protection_method
|
|
|
|
when :exception
|
|
|
|
raise ActionController::InvalidAuthenticityToken
|
|
|
|
when :reset_session
|
|
|
|
reset_session
|
|
|
|
else
|
|
|
|
raise ArgumentError, 'Invalid request forgery protection method, use :exception or :reset_session'
|
|
|
|
end
|
2007-09-22 22:32:55 -04:00
|
|
|
end
|
2009-10-28 03:12:35 -04:00
|
|
|
|
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
|
|
|
#
|
|
|
|
# * is it a GET 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?
|
2011-01-04 19:36:07 -05:00
|
|
|
!protect_against_forgery? || request.get? ||
|
|
|
|
form_authenticity_token == params[request_forgery_protection_token] ||
|
|
|
|
form_authenticity_token == 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
|
2011-05-23 07:02:06 -04:00
|
|
|
session[:_csrf_token] ||= SecureRandom.base64(32)
|
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
|
|
|
|
|
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
|