Germany/Projekte/Top 10 fuer Entwickler-2013/A8-Cross-Site Request Forgery (CSRF)

A5 Cross-Site Request Forgery (CSRF, XSRF, Session Riding)
Jeder, der einem Nutzer einer Webanwendung einen nicht beabsichtigten Request für diese Anwendung unterschieben kann. Hierfür kommt jede Website oder jede HTML-Quelle in Betracht, die der Nutzer verwendet. Durch Image-Tags, XSS oder andere Techniken löst das Opfer unbeabsichtigt einen gefälschten HTTP-Request für eine Anwendung aus. Falls der Nutzer authentisiert ist, wird dieser Angriff Erfolg haben. CSRF zielt auf Anwendungen, die es dem Angreifer erlauben, alle Details eines Requests für eine bestimmte Aktion vorherzusagen.

Da Browser Informationen zum Session-Management automatisch mitsenden, kann ein Angreifer gefälschte Requests auf bösartigen Websites hinterlegen, die von legitimen Requests nicht unterschieden werden können.

CSRF-Schwächen sind leicht durch Penetrationstests oder Quellcode-Analysen auffindbar. Der Angreifer kann unbemerkt das Opfer über dessen Browser dazu veranlassen, alle Daten zu ändern oder jede Funktion auszuführen, für die das spezifische Opfer berechtigt ist. Betrachten Sie den Geschäftswert der betroffenen Daten oder Funktionen. Es bleibt die Unsicherheit, ob der Nutzer die Aktion ausführen wollte.

Bedenken Sie mögliche Auswirkungen auf Ihre Reputation.

Die Anwendung erlaubt es einem Benutzer, einen zustandsändernden Request auszulösen, der kein geheimes Token beinhaltet, wie z.B.:


 *  http://example.com/app/transferFunds?amount=1500&amp;destinationAccount=4673243243 

Dadurch kann ein Angreifer einen Request erzeugen, der Geld vom Konto des Opfers auf das Konto des Angreifers transferiert. Diesen bettet er in einem Image-Tag oder einem Iframe ein und hinterlegt ihn in einer beliebigen Website.


 *  &lt;img src=&quot;http://example.com/app/transferFunds? amount=1500&amp;destinationAccount=attackersAcct#&quot; width=&quot;0&quot; height=&quot;0&quot; /&gt; 

Wenn das Opfer eine präparierte Seite besucht, während es z.B. bereits auf example.com authentisiert ist, wird der untergeschobene Request unbemerkt ausgeführt, da der Browser die aktuellen Session-Informationen automatisch mitsendet und somit unbeabsichtigt autorisiert.

Um CSRF zu verhindern, muss ein unvorhersagbarer Token im Body oder in der URL eines jeden HTTP-Requests eingebettet sein (und geprüft werden). Ein solcher Token sollte für mindestens jede Nutzer-Session, besser noch für jeden Request, einzigartig sein.
 * 1) Die bevorzugte Methode, ein solches Token unterzubringen ist ein Hidden-Input-Feld. Damit wird der Token-Wert im Body des HTTP-Requests und nicht im URL übertragen. Eine Übertragung im URL kann leichter ausgespäht werden.
 * 2) Ein solches Token kann auch direkt in den URL geschrieben oder als URL-Parameter übergeben werden. Jedoch birgt diese Vorgehensweise das Risiko, dass der URL dem Angreifer in die Hände fällt und somit das geheime Token kompromittiert ist. OWASPs  CSRF Guard kann genutzt werden, um automatisch solche Token in Java EE, .NET oder PHP Anwendungen einzubinden. OWASPs ESAPI / OWASP’s ESAPI (tbd!!) beinhaltet Token-Generatoren und Validatoren, die Entwickler einsetzen können, um ihre Transaktionen zu schützen.

= JAVA =

ESAPI
 /** (A) CSRF-Token erzeugen **/  private String csrfToken = resetCSRFToken; /** (B) In zu schützenden FORM-Seiten (B1), oder Urls (B2) das CSRF-Token als 'hidden field' hinzufügen, vgl ESAPI DefaultHTTPUtilities.java**/

/** (B1) das CSRF-Token als 'hidden field' in den GET-Request hinzufügen **/ example.jsp ... <%    String csrfTokenFieldName = org.owasp.esapi.HTTPUtilities.CSRF_TOKEN_NAME; String csrfToken = ESAPI.httpUtilities.getCSRFToken; %>  " value="<%csrfToken%>"/>  ... /** (B2) das CSRF-Token als URL Parameter in den GET-Request hinzufügen **/ final static String CSRF_TOKEN_NAME = "ctoken"; <%
 * user = ESAPI.authenticator.login(request, response);
 * String transferFundsHref = "/SwingSet/main?function=TransferFunds&solution";

%>  ' target="_blank">Transfer Funds  /** (C) Beim Empfang der Daten das CSRF-Token prüfen (GET-Request) **/ try {
 * ESAPI.httpUtilities.verifyCSRFToken(request);
 * fail;

} catch( Exception e ) {
 * // expected

} '''/** (D) Beim Abmelden bzw. Timeout die Session ungültig machen **/''' User user = ESAPI.authenticator.logout;

CSRF Guard

 * /** Parameter für CsrfGuard und Java EE-Filter im Deployment **/
 * /** Descriptor des Webserver-Containers definieren (web.xml-Datei) **/
 * Owasp.CsrfGuard.Config
 * WEB-INF/Owasp.CsrfGuard.properties

 
 * Owasp.CsrfGuard.Config.Print</param-name>
 * true</param-value>

</context-param>
 * <listener-class>org.owasp.csrfguard.CsrfGuardListener</listener-class>

 
 *  <filter-name>CSRFGuard</filter-name> 
 *  <filter-class>org.owasp.csrfguard.CsrfGuardFilter</filter-class> 

    <filter-mapping> 
 *  <filter-name>CSRFGuard</filter-name> 
 *  <url-pattern>/*</url-pattern> 

 </filter-mapping> 

vgl Owasp.CsrfGuard.Test Die Parameter in der Datei 'Owasp.CsrfGuard.properties' gemäß CSRFGuard_3_Configuration einstellen.

Tbd!!
Z.B. Keine Bookmarks auf ausgefüllte Eingabe-Formulare, Suchergebnisse; Frameworks funktionieren teilweise nicht für Web 2.0
 * {Soll so etwas rein?}


 * OWASP CSRF Prevention Cheat Sheet
 * OWASP CSRF Article
 * OWASP CSRFGuard - CSRF Defense Tool
 * OWASP CSRFGuard - CSRFGuard_3_Installation
 * OWASP CSRFGuard - CSRFGuard_3_Configuration
 * ESAPI Project Home Page
 * ESAPI HTTPUtilities Class with AntiCSRF Tokens
 * ESAPI-Java-Wiki
 * OWASP Testing Guide: Chapter on CSRF Testing
 * OWASP CSRFTester - CSRF Testing Tool


 * CWE Entry 352 on CSRF
 * John Melton's Weblog: The OWASP Top Ten and ESAPI –Part 5–

= Test =

tbd Text

tbd Text

(ganze Breite) Text

Text

Text

= Test LanguageFile 2010 en =

tbd Text

tbd Text

tbd (ganze Breite) Text

Text

Text

Text

= Test LanguageFile 2010 us =

tbd Text

tbd Text

tbd (ganze Breite) Text

Text

Text

Text

= Test LanguageFile 2010 Default =

tbd Text

tbd Text

tbd (ganze Breite) Text

Text

Text

Text

= Test LanguageFile 2013 Default =

tbd Text

tbd Text

tbd (ganze Breite) Text

Text

Text

Text

= Test LanguageFile Default =

tbd Text

tbd Text

tbd (ganze Breite) Text

Text

Text

Text