mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 12:05:37 +00:00
361 lines
12 KiB
Markdown
361 lines
12 KiB
Markdown
|
# Security
|
||
|
|
||
|
## Introduction
|
||
|
|
||
|
This page details notes on how to ensure that we develop secure SilverStripe applications. See [security](/topics/security) for
|
||
|
the Silverstripe-class as a starting-point for most security-related functionality.
|
||
|
|
||
|
See our [contributing guidelines](/misc/contributing#reporting-security-issues) on how
|
||
|
to report security issues.
|
||
|
|
||
|
## SQL Injection
|
||
|
|
||
|
The [coding-conventions](/misc/coding-conventions) help guard against SQL injection attacks but still require developer
|
||
|
dilligence: ensure that any variable you insert into a filter / sort / join clause has been escaped.
|
||
|
|
||
|
See [http://shiflett.org/articles/sql-injection](http://shiflett.org/articles/sql-injection).
|
||
|
|
||
|
### Automatic escaping
|
||
|
|
||
|
Silverstripe automatically runs [addslashes()](http://php.net/addslashes) in DataObject::write() wherever possible. Data
|
||
|
is escaped when saving back to the database, not when writing to object-properties.
|
||
|
|
||
|
* DataObject::get_by_id()
|
||
|
* DataObject::update()
|
||
|
* DataObject::castedUpdate()
|
||
|
* DataObject->Property = 'val', DataObject->setField('Property','val')
|
||
|
* DataObject::write()
|
||
|
* Form->saveInto()
|
||
|
* FormField->saveInto()
|
||
|
* DBField->saveInto()
|
||
|
|
||
|
Note: It is NOT good practice to "be sure" and convert the data passed to the functions below manually. This might
|
||
|
result in *double escaping* and alters the actually saved data (e.g. by adding slashes to your content).
|
||
|
|
||
|
### Manual escaping
|
||
|
|
||
|
As a rule of thumb, whenever you're creating raw queries (or just chunks of SQL), you need to take care of escaping
|
||
|
yourself. See [coding-conventions](/misc/coding-conventions) and [datamodel](/topics/datamodel) for ways to cast and convert
|
||
|
your data.
|
||
|
|
||
|
* SQLQuery
|
||
|
* DataObject::buildSQL()
|
||
|
* DB::query()
|
||
|
* Director::urlParams()
|
||
|
* Controller->requestParams, Controller->urlParams
|
||
|
* GET/POST data passed to a Form-method
|
||
|
|
||
|
Example:
|
||
|
|
||
|
:::php
|
||
|
class MyForm extends Form {
|
||
|
function save($RAW_data, $form) {
|
||
|
$SQL_data = Convert::raw2sql($RAW_data); // works recursively on an array
|
||
|
$objs = DataObject::get('Player', "Name = '{$SQL_data[name]}'");
|
||
|
// ...
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
* FormField->Value()
|
||
|
* URLParams passed to a Controller-method
|
||
|
|
||
|
Example:
|
||
|
|
||
|
:::php
|
||
|
class MyController extends Controller {
|
||
|
function myurlaction($RAW_urlParams) {
|
||
|
$SQL_urlParams = Convert::raw2sql($RAW_urlParams); // works recursively on an array
|
||
|
$objs = DataObject::get('Player', "Name = '{$SQL_data[OtherID]}'");
|
||
|
// ...
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
As a rule of thumb, you should escape your data **as close to querying as possible**.
|
||
|
This means if you've got a chain of functions passing data through, escaping should happen at the end of the chain.
|
||
|
|
||
|
:::php
|
||
|
class MyController extends Controller {
|
||
|
/**
|
||
|
|
||
|
* @param array $RAW_data All names in an indexed array (not SQL-safe)
|
||
|
*/
|
||
|
function saveAllNames($RAW_data) {
|
||
|
// $SQL_data = Convert::raw2sql($RAW_data); // premature escaping
|
||
|
foreach($RAW_data as $item) $this->saveName($item);
|
||
|
}
|
||
|
|
||
|
function saveName($RAW_name) {
|
||
|
$SQL_name = Convert::raw2sql($RAW_name);
|
||
|
DB::query("UPDATE Player SET Name = '{$SQL_name}'");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
This might not be applicable in all cases - especially if you are building an API thats likely to be customized. If
|
||
|
you're passing unescaped data, make sure to be explicit about it by writing *phpdoc*-documentation and *prefixing* your
|
||
|
variables ($RAW_data instead of $data).
|
||
|
|
||
|
|
||
|
|
||
|
## XSS (Cross-Site-Scripting)
|
||
|
|
||
|
SilverStripe helps you guard any output against clientside attacks initiated by malicious user input, commonly known as
|
||
|
XSS (Cross-Site-Scripting). With some basic guidelines, you can ensure your output is safe for a specific use case (e.g.
|
||
|
displaying a blog post in HTML from a trusted author, or escaping a search parameter from an untrusted visitor before
|
||
|
redisplaying it).
|
||
|
|
||
|
Note: SilverStripe templates do not remove tags, please use [strip_tags()](http://php.net/strip_tags) for this purpose
|
||
|
or [sanitize](http://htmlpurifier.org/) it correctly.
|
||
|
|
||
|
See [http://shiflett.org/articles/foiling-cross-site-attacks](http://shiflett.org/articles/foiling-cross-site-attacks)
|
||
|
for in-depth information about "Cross-Site-Scripting".
|
||
|
|
||
|
### Escaping model properties
|
||
|
|
||
|
`[api:SSViewer]` (the SilverStripe template engine) automatically takes care of escaping HTML tags from specific
|
||
|
object-properties by [casting](/topics/datamodel#casting) its string value into a `[api:DBField]` object.
|
||
|
|
||
|
PHP:
|
||
|
|
||
|
:::php
|
||
|
class MyObject extends DataObject {
|
||
|
public static $db = array(
|
||
|
'MyEscapedValue' => 'Text', // Example value: <b>not bold</b>
|
||
|
'MyUnescapedValue' => 'HTMLText' // Example value: <b>bold</b>
|
||
|
);
|
||
|
}
|
||
|
|
||
|
|
||
|
Template:
|
||
|
|
||
|
:::php
|
||
|
<ul>
|
||
|
<li>$MyEscapedValue</li> // output: <b>not bold<b>
|
||
|
<li>$MyUnescapedValue</li> // output: <b>bold</b>
|
||
|
</ul>
|
||
|
|
||
|
|
||
|
The example below assumes that data wasn't properly filtered when saving to the database, but are escaped before
|
||
|
outputting through SSViewer.
|
||
|
|
||
|
### Overriding default escaping in templates
|
||
|
|
||
|
You can force escaping on a casted value/object by using an [escape type](/topics/datamodel) method in your template, e.g.
|
||
|
"XML" or "ATT".
|
||
|
|
||
|
Template (see above):
|
||
|
|
||
|
:::php
|
||
|
<ul>
|
||
|
// output: <a href="#" title="foo & &#quot;bar"">foo & "bar"</a>
|
||
|
<li><a href="#" title="$Title.ATT">$Title</a></li>
|
||
|
<li>$MyEscapedValue</li> // output: <b>not bold<b>
|
||
|
<li>$MyUnescapedValue</li> // output: <b>bold</b>
|
||
|
<li>$MyUnescapedValue.XML</li> // output: <b>bold<b>
|
||
|
</ul>
|
||
|
|
||
|
|
||
|
### Escaping custom attributes and getters
|
||
|
|
||
|
Every object attribute or getter method used for template purposes should have its escape type defined through the
|
||
|
static *$casting* array. Caution: Casting only applies when using values in a template, not in PHP.
|
||
|
|
||
|
PHP:
|
||
|
|
||
|
:::php
|
||
|
class MyObject extends DataObject {
|
||
|
public $Title = '<b>not bold</b>'; // will be escaped due to Text casting
|
||
|
|
||
|
$casting = array(
|
||
|
"Title" => "Text", // forcing a casting
|
||
|
'TitleWithHTMLSuffix' => 'HTMLText' // optional, as HTMLText is the default casting
|
||
|
);
|
||
|
|
||
|
function TitleWithHTMLSuffix($suffix) {
|
||
|
// $this->Title is not casted in PHP
|
||
|
return $this->Title . '<small>(' . $suffix. ')</small>';
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
Template:
|
||
|
|
||
|
:::php
|
||
|
<ul>
|
||
|
<li>$Title</li> // output: <b>not bold<b>
|
||
|
<li>$Title.RAW</li> // output: <b>not bold</b>
|
||
|
<li>$TitleWithHTMLSuffix</li> // output: <b>not bold</b>: <small>(...)</small>
|
||
|
</ul>
|
||
|
|
||
|
|
||
|
Note: Avoid generating HTML by string concatenation in PHP wherever possible to minimize risk and separate your
|
||
|
presentation from business logic.
|
||
|
|
||
|
### Manual escaping in PHP
|
||
|
|
||
|
When using *customise()* or *renderWith()* calls in your controller, or otherwise forcing a custom context for your
|
||
|
template, you'll need to take care of casting and escaping yourself in PHP.
|
||
|
|
||
|
The `[api:Convert]` class has utilities for this, mainly *Convert::raw2xml()* and *Convert::raw2att()* (which is
|
||
|
also used by *XML* and *ATT* in template code).
|
||
|
|
||
|
PHP:
|
||
|
|
||
|
:::php
|
||
|
class MyController extends Controller {
|
||
|
function search($request) {
|
||
|
$htmlTitle = '<p>Your results for:' . Convert::raw2xml($request->getVar('Query')) . '</p>';
|
||
|
return $this->customise(array(
|
||
|
'Query' => DBField::create('Text', $request->getVar('Query')),
|
||
|
'HTMLTitle' => DBField::create('HTMLText', $htmlTitle)
|
||
|
));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
Template:
|
||
|
|
||
|
:::php
|
||
|
<h2 title="Searching for $Query.ATT">$HTMLTitle</h2>
|
||
|
|
||
|
|
||
|
Whenever you insert a variable into an HTML attribute within a template, use $VarName.ATT, no not $VarName.
|
||
|
|
||
|
You can also use the built-in casting in PHP by using the *obj()* wrapper, see [datamodel](/topics/datamodel) .
|
||
|
|
||
|
### Escaping URLs
|
||
|
|
||
|
Whenever you are generating a URL that contains querystring components based on user data, use urlencode() to escape the
|
||
|
user data, not *Convert::raw2att()*. Use raw ampersands in your URL, and cast the URL as a "Text" DBField:
|
||
|
|
||
|
PHP:
|
||
|
|
||
|
:::php
|
||
|
class MyController extends Controller {
|
||
|
function search($request) {
|
||
|
$rssRelativeLink = "/rss?Query=" . urlencode($_REQUEST['query']) . "&sortOrder=asc";
|
||
|
$rssLink = Controller::join_links($this->Link(), $rssRelativeLink);
|
||
|
return $this->customise(array(
|
||
|
"RSSLink" => DBField::create("Text", $rssLink),
|
||
|
));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
Template:
|
||
|
|
||
|
:::php
|
||
|
<a href="$RSSLink.ATT">RSS feed</a>
|
||
|
|
||
|
|
||
|
Some rules of thumb:
|
||
|
|
||
|
* Don't concatenate URLs in a template. It only works in extremely simple cases that usually contain bugs.
|
||
|
* Use *Controller::join_links()* to concatenate URLs. It deals with querystrings and other such edge cases.
|
||
|
|
||
|
|
||
|
## Cross-Site Request Forgery (CSRF)
|
||
|
|
||
|
SilverStripe has built-in countermeasures against this type of identity theft for all form submissions. A form object
|
||
|
will automatically contain a *SecurityID* parameter which is generated as a secure hash on the server, connected to the
|
||
|
currently active session of the user. If this form is submitted without this parameter, or if the parameter doesn't
|
||
|
match the hash stored in the users session, the request is discarded.
|
||
|
|
||
|
If you know what you're doing, you can disable this behaviour:
|
||
|
|
||
|
:::php
|
||
|
$myForm->disableSecurityToken();
|
||
|
|
||
|
|
||
|
See
|
||
|
[http://shiflett.org/articles/cross-site-request-forgeries](http://shiflett.org/articles/cross-site-request-forgeries)
|
||
|
|
||
|
|
||
|
|
||
|
## Casting user input
|
||
|
|
||
|
When working with `$_GET`, `$_POST` or `Director::urlParams` variables, and you know your variable has to be of a
|
||
|
certain type, like an integer, then it's essential to cast it as one. *Why?* To be sure that any processing of your
|
||
|
given variable is done safely, with the assumption that it's an integer.
|
||
|
|
||
|
To cast the variable as an integer, place `(int)` or `(integer)` before the variable.
|
||
|
|
||
|
For example: a page with the URL paramaters *mysite.com/home/add/1* requires that ''Director::urlParams['ID']'' be an
|
||
|
integer. We cast it by adding `(int)` - ''(int)Director::urlParams['ID']''. If a value other than an integer is
|
||
|
passed, such as *mysite.com/home/add/dfsdfdsfd*, then it returns 0.
|
||
|
|
||
|
Below is an example with different ways you would use this casting technique:
|
||
|
|
||
|
:::php
|
||
|
function CaseStudies() {
|
||
|
|
||
|
// cast an ID from URL parameters e.g. (mysite.com/home/action/ID)
|
||
|
$anotherID = (int)Director::urlParams['ID'];
|
||
|
|
||
|
// perform a calculation, the prerequisite being $anotherID must be an integer
|
||
|
$calc = $anotherID + (5 - 2) / 2;
|
||
|
|
||
|
// cast the 'category' GET variable as an integer
|
||
|
$categoryID = (int)$_GET['category'];
|
||
|
|
||
|
// perform a get_by_id, ensure the ID is an integer before querying
|
||
|
return DataObject::get_by_id('CaseStudy', $categoryID);
|
||
|
}
|
||
|
|
||
|
|
||
|
The same technique can be employed anywhere in your PHP code you know something must be of a certain type. A list of PHP
|
||
|
cast types can be found here:
|
||
|
|
||
|
* `(int)`, `(integer)` - cast to integer
|
||
|
* `(bool)`, `(boolean)` - cast to boolean
|
||
|
* `(float)`, `(double)`, `(real)` - cast to float
|
||
|
* `(string)` - cast to string
|
||
|
* `(array)` - cast to array
|
||
|
* `(object)` - cast to object
|
||
|
|
||
|
Note that there is also a 'SilverStripe' way of casting fields on a class, this is a different type of casting to the
|
||
|
standard PHP way. See [casting](/topics/datamodel#casting).
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
## Filesystem
|
||
|
|
||
|
### Don't allow script-execution in /assets
|
||
|
|
||
|
As all uploaded files are stored by default on the /assets-directory, you should disallow script-execution for this
|
||
|
folder. This is just an additional security-measure to making sure you avoid directory-traversal, check for filesize and
|
||
|
disallow certain filetypes.
|
||
|
|
||
|
Example configuration for Apache2:
|
||
|
|
||
|
|
||
|
<VirtualHost *:80>
|
||
|
...
|
||
|
<LocationMatch assets/>
|
||
|
php_flag engine off
|
||
|
Options -ExecCGI -Includes -Indexes
|
||
|
</LocationMatch>
|
||
|
</VirtualHost>
|
||
|
|
||
|
|
||
|
If you are using shared hosting or in a situation where you cannot alter your Vhost definitions, you can use a .htaccess
|
||
|
file in the assets directory. This requires PHP to be loaded as an Apache module (not CGI or FastCGI).
|
||
|
|
||
|
**/assets/.htaccess**
|
||
|
|
||
|
|
||
|
php_flag engine off
|
||
|
Options -ExecCGI -Includes -Indexes
|
||
|
|
||
|
|
||
|
## Related
|
||
|
|
||
|
* [http://silverstripe.org/security-releases/](http://silverstripe.org/security-releases/)
|
||
|
|
||
|
## Links
|
||
|
|
||
|
* [Best-practices for securing MySQL (securityfocus.com)](http://www.securityfocus.com/infocus/1726)
|