#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:
Andrew O'Neil 2007-10-26 02:23:55 +00:00
parent 7d0ddb9d33
commit dc1775169d
3 changed files with 50 additions and 1 deletions

View File

@ -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); 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 // First, try a handler method on the controller
if($this->hasMethod($funcName) || !$form) { if($this->hasMethod($funcName) || !$form) {

View File

@ -174,8 +174,22 @@ class Form extends ViewableData {
* @return FieldSet The form fields * @return FieldSet The form fields
*/ */
function 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; return $this->fields;
} }
}
/** /**
* Get a named field from this form's fields. * Get a named field from this form's fields.
@ -665,6 +679,27 @@ class Form extends ViewableData {
$this->hasDefaultAction = false; $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. * 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. * It checks for a call to the callfieldmethod action.
@ -689,6 +724,7 @@ class Form extends ViewableData {
self::$current_action = $action; self::$current_action = $action;
} }
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// TESTING HELPERS // TESTING HELPERS
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

View File

@ -110,6 +110,10 @@ ComplexTableField.prototype = {
} }
} }
if($('SecurityID')) {
popupLink = popupLink + '&SecurityID=' + $('SecurityID').value;
}
GB_OpenerObj = this; GB_OpenerObj = this;
// use same url to refresh the table after saving the popup, but use a generic rendering method // use same url to refresh the table after saving the popup, but use a generic rendering method
GB_RefreshLink = popupLink; GB_RefreshLink = popupLink;