mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 14:05:37 +02:00
#941 - Security flaw: SS prone to CSRF attack
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@43876 467b73ca-7a2a-4603-9d3b-597d59a354a9
This commit is contained in:
parent
7d0ddb9d33
commit
dc1775169d
@ -156,6 +156,15 @@ class Controller extends ViewableData {
|
||||
user_error("No action button has been clicked in this form executon, and no default has been allowed", E_USER_ERROR);
|
||||
}
|
||||
|
||||
// Protection against CSRF attacks
|
||||
if($form->securityEnabled()) {
|
||||
$securityID = Session::get('SecurityID');
|
||||
|
||||
if(!$securityID || !isset($this->requestParams['SecurityID']) || $securityID != $this->requestParams['SecurityID']) {
|
||||
trigger_error("Security ID doesn't match, possible CRSF attack.", E_USER_ERROR);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// First, try a handler method on the controller
|
||||
if($this->hasMethod($funcName) || !$form) {
|
||||
|
@ -174,8 +174,22 @@ class Form extends ViewableData {
|
||||
* @return FieldSet The form fields
|
||||
*/
|
||||
function Fields() {
|
||||
if($this->securityEnabled()) {
|
||||
if(Session::get('SecurityID')) {
|
||||
$securityID = Session::get('SecurityID');
|
||||
} else {
|
||||
$securityID = rand();
|
||||
Session::set('SecurityID', $securityID);
|
||||
}
|
||||
|
||||
$fieldsClone = clone $this->fields;
|
||||
$fieldsClone->push(new HiddenField('SecurityID', 'SecurityID', $securityID));
|
||||
|
||||
return $fieldsClone;
|
||||
} else {
|
||||
return $this->fields;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Get a named field from this form's fields.
|
||||
@ -665,6 +679,27 @@ class Form extends ViewableData {
|
||||
$this->hasDefaultAction = false;
|
||||
}
|
||||
|
||||
private $security = true;
|
||||
|
||||
/**
|
||||
* Disable the requirement of a SecurityID in the Form. This security protects
|
||||
* against CSRF attacks, but you should disable this if you don't want to tie
|
||||
* a form to a session - eg a search form.
|
||||
*/
|
||||
function disableSecurity() {
|
||||
$this->security = false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns true if security is enabled - that is if the SecurityID
|
||||
* should be included and checked on this form.
|
||||
*
|
||||
* @return bool
|
||||
*/
|
||||
function securityEnabled() {
|
||||
return $this->security;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the name of a field, if that's the only field that the current controller is interested in.
|
||||
* It checks for a call to the callfieldmethod action.
|
||||
@ -689,6 +724,7 @@ class Form extends ViewableData {
|
||||
self::$current_action = $action;
|
||||
}
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||
// TESTING HELPERS
|
||||
//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||
|
@ -110,6 +110,10 @@ ComplexTableField.prototype = {
|
||||
}
|
||||
}
|
||||
|
||||
if($('SecurityID')) {
|
||||
popupLink = popupLink + '&SecurityID=' + $('SecurityID').value;
|
||||
}
|
||||
|
||||
GB_OpenerObj = this;
|
||||
// use same url to refresh the table after saving the popup, but use a generic rendering method
|
||||
GB_RefreshLink = popupLink;
|
||||
|
Loading…
Reference in New Issue
Block a user