IMF
- Layout for this exercise:
data:image/s3,"s3://crabby-images/d3f8e/d3f8eb230d4bf8f673409040488f5eedbbc1efb4" alt=""
1 - INTRODUCTION
- The goal of this exercise is to develop a hacking process for the vulnerable machine IMF.
- According to the author's description IMF contains a number of flags starting off easy and getting harder as the hacking process progresses, so that each flag gives a hint for the next flag.
- IMF can be downloaded from here:
https://www.vulnhub.com/entry/imf-1,162/
- Once downloaded and extracted with VirtualBox:
data:image/s3,"s3://crabby-images/d0b3d/d0b3d40fa743e11d9f7a6d93aa2f21fae87d53dc" alt=""
2 - ENUMERATION
- Using netdiscover to find the IP that corresponds to IMF:
data:image/s3,"s3://crabby-images/0dcd3/0dcd34c2dded1bb0854c06ff06519cdc09629f83" alt=""
data:image/s3,"s3://crabby-images/69324/6932451fc44c98c181264b9b2d34136508e6362b" alt=""
- Scanning with Nmap:
data:image/s3,"s3://crabby-images/d171f/d171fdeb4302c20de64ced697715450f5a17c109" alt=""
- So there is just one service available, the web server at port 80. Let's connect directly via a browser:
data:image/s3,"s3://crabby-images/bee3d/bee3dd0601cecd31389386f4df7ca5f0d3953269" alt=""
2 - FLAG 1
- Examining the source of the script contact.php we find a reference to flag1:
data:image/s3,"s3://crabby-images/6dd15/6dd15992e0857d332f8e107b33e5c056a624d942" alt=""
- Storing and decoding with base64:
data:image/s3,"s3://crabby-images/17727/17727c698b439a7d50d940e246d3ed80ba39647f" alt=""
data:image/s3,"s3://crabby-images/3feb1/3feb1097d0c9924d831b91df5095377bbf67b9a6" alt=""
- So we have the 1st flag: allthefiles
3 - FLAG 2
- Also, at the same script contact.php we find 3 lines of characters ending up with ==, what gives a hint that it might be another encoded base64 string:
data:image/s3,"s3://crabby-images/d6dc6/d6dc616169d1223eecfe5f9cdea7606f8911400b" alt=""
- Storing the characters and decoding with base64, we find a recursive reference to flag2:
data:image/s3,"s3://crabby-images/7506d/7506de1431f13a395b2d61704a2be1d87de3ccd0" alt=""
data:image/s3,"s3://crabby-images/1f198/1f198afa1173766295815ebb9e40811d82611ed4" alt=""
- Storing and decoding with base64:
data:image/s3,"s3://crabby-images/fc049/fc049d507cfc71e9724aeebe1bfb625b7346c577" alt=""
data:image/s3,"s3://crabby-images/75e1c/75e1c48ea1dfc20632c21d31eeecf452a9e2eccc" alt=""
- So we have the 2nd flag: imfadministrator
4 - FLAG 3
- When trying the string achieved at the 2nd flag as a folder for the web server we get a login form:
data:image/s3,"s3://crabby-images/ba9c1/ba9c10ce722f9f1dc40c8b912377089b64d1ced3" alt=""
- Viewing the source:
data:image/s3,"s3://crabby-images/1b606/1b606ed296eea726ecb4a6b2ab43c19e06f5f268" alt=""
- Somebody called Roger gives us a message about SQL and a "hard-coded password".
- Having a look at the Contact Us tab we are offered 3 different potential usernames:
data:image/s3,"s3://crabby-images/f9e7f/f9e7f79886b2130776876fa881eb699dade710b1" alt=""
- Let's study the different responses from the server when using rmichaels, akeith or estone as usernames and any other random password:
- For rmichaels we have an answer of "invalid password":
data:image/s3,"s3://crabby-images/1b661/1b661e67d84d06d84c0957aca4c7ef25d448aad7" alt=""
data:image/s3,"s3://crabby-images/095f1/095f13dda82217251c5fb03c7964e17169e568c0" alt=""
- For akeith we have an anwser of "invalid username":
data:image/s3,"s3://crabby-images/be962/be962c965ec0a3200972d3bab305aac29e1b44d3" alt=""
data:image/s3,"s3://crabby-images/13f94/13f943c028be502562fed74dc246ba1aabc514ae" alt=""
- For estone we have an anwser of "invalid username":
data:image/s3,"s3://crabby-images/126cb/126cbe06acb4e860d40fd1e629765931b732a969" alt=""
data:image/s3,"s3://crabby-images/22060/22060bc657e5568f015601d27eda5341dc634c29" alt=""
- We conclude that rmichaels is a valid username, although we don't know yet the password.
- However, at this point we know that the server is using the language PHP (contact.php), and Burp can help us to analyze what is happening when entering credentials at the login form.
- Let's use Burp to intercept the login process using rmichaels as username and an arbitrary password:
data:image/s3,"s3://crabby-images/1f2e2/1f2e2ca01881c40a6a53f3ed656df8bf90b70214" alt=""
data:image/s3,"s3://crabby-images/75c50/75c50f68d7926ba96aa4bc4fc893d20763a13e22" alt=""
- First thing to do is to store the cookie value for later usage:
data:image/s3,"s3://crabby-images/8b852/8b8527e752f86a693ebe826a338661c917fc4e5a" alt=""
- Now, let's review the characteristics of the PHP strmcp() function.
- strmcp() compares two strings, returning 0 if they are equal; however, it also returns 0 if one the arguments is a string and the other one an array, like explained here:
http://danuxx.blogspot.com/2013/03/unauthorized-access-bypassing-php-strcmp.html
https://marcosvalle.github.io/ctf/php/2016/05/12/php-comparison-vlun.html
- We can take advantage of that circumstance by modifying the parameter pass to pass[] (an array):
data:image/s3,"s3://crabby-images/9537f/9537f9405a927368d13c9ab4af8e811b356682a2" alt=""
data:image/s3,"s3://crabby-images/13b2d/13b2d9b762fd4ca2698ffb48639945c57ed430ad" alt=""
- Forwarding, the authentication is bypassed:
data:image/s3,"s3://crabby-images/fadd8/fadd8553adfa75e921189bf4c105f69b76926fde" alt=""
data:image/s3,"s3://crabby-images/50cb7/50cb79797c3e141bec83064bbbd420f7c86aed20" alt=""
- Storing and decoding we get the flag3: continueTOcms
data:image/s3,"s3://crabby-images/1a839/1a839ecc8aa18652b8870e65152dd2a58c55695b" alt=""
data:image/s3,"s3://crabby-images/a82a6/a82a6540deb50ff5227386058114e7d7c612e753" alt=""
5 - FLAG 4
- Following the advice at flag3 let's continue by clicking IMF CMS (after disabling Burp's interception):
data:image/s3,"s3://crabby-images/f7a1f/f7a1f5f61359a6f1a4d1f404af519a27457fb2fb" alt=""
data:image/s3,"s3://crabby-images/b89d0/b89d0d06b99d048098a823ac07227b29cce96c55" alt=""
- Enumerating the potential databases with sqlmap and option --dbs:
data:image/s3,"s3://crabby-images/23c06/23c06e7fe5da61f8d4c1ae5a5c5fc2def848107a" alt=""
data:image/s3,"s3://crabby-images/a7afa/a7afa0b8292de552a1fdd73fc27dd31815684e36" alt=""
- Searching for tables at database admin:
data:image/s3,"s3://crabby-images/46e56/46e569772e4f824ee2544652d69241026081e36e" alt=""
data:image/s3,"s3://crabby-images/38b5b/38b5bd19fcb68dafc6528101667035e78efbf96b" alt=""
- Dumping the content of admin -> pages:
data:image/s3,"s3://crabby-images/fadbc/fadbca48f717f688107b824f75ee15fa339a9377" alt=""
data:image/s3,"s3://crabby-images/6be5b/6be5befad286444e66c5be2b411fa45735ed4ed8" alt=""
- The result of the dump shows that under pages there is another hidden folder called tutorials-incomplete. Let's check it:
data:image/s3,"s3://crabby-images/7121d/7121d30da4408a0bc5dfbc2ae4a733f6b6a3e5bb" alt=""
- At the lower left side of the picture we can find one QR code. Isolating and decoding it with xing we get the flag4:
data:image/s3,"s3://crabby-images/d6d4c/d6d4c57f68b1cd494b92882f7edbd2626215995d" alt=""
data:image/s3,"s3://crabby-images/e2c75/e2c75e2dfec47c47027353b764ca6f49c52a52d9" alt=""
data:image/s3,"s3://crabby-images/84069/84069a97e29c16541461745f5b4ce006ef017b95" alt=""
- Storing and decoding flag4:
data:image/s3,"s3://crabby-images/8eadc/8eadc3bd653d31efc077a8488d0739d397e14f88" alt=""
data:image/s3,"s3://crabby-images/16e3b/16e3b19d0ee8e9f6c0c051b977546793b9dc4e9b" alt=""
- The script uploadr942.php leads us to a very interesting uploading page:
data:image/s3,"s3://crabby-images/0b74f/0b74f4b62d77879e92d376a243e3f37636110f40" alt=""
6 - FLAG 5
- Let's try some uploads to learn whether there is any type of restriction of filter about the format of the uploading files.
- First, we learn that uploading a PHP file is not allowed:
data:image/s3,"s3://crabby-images/611ea/611ea72a2dae181d355c4b0d1e056d1677e96532" alt=""
data:image/s3,"s3://crabby-images/ea040/ea040d6a7272368cb6bb517c2dd26ebb9a0296d6" alt=""
data:image/s3,"s3://crabby-images/15af3/15af3dad1d50adc0360d45b9d5b2fdb0743dbcc7" alt=""
- However, the same script adding the string GIF89 and renaming with the .gif extension is accepted:
data:image/s3,"s3://crabby-images/57f66/57f66d53b15ef5514705b60115f1e4d3d9916f54" alt=""
data:image/s3,"s3://crabby-images/7d118/7d118f1bf737f6665849dad3138cf95692932280" alt=""
data:image/s3,"s3://crabby-images/8feb4/8feb4e3608e2678e4cb3419bd71362da1fd2cbcb" alt=""
- Viewing the source, there is a string that refers to the successfully loaded file:
data:image/s3,"s3://crabby-images/52b64/52b6447781378cbbd85b95e4227f33ce24e89188" alt=""
data:image/s3,"s3://crabby-images/32209/32209a2c32758cb5c6477ee426a7cacb084741a5" alt=""
- Trying to visit the /uploads folder we notice that it is forbidden, but it means that at least we know of its existence:
data:image/s3,"s3://crabby-images/c7ca8/c7ca8aefcd184f74ec1261b7f57fa9a8cf5d7d9b" alt=""
- We are able to confirm that the file has been successfully uploaded to /uploads:
data:image/s3,"s3://crabby-images/891ad/891ad6a88cd7a7722a1203e53f341ee49633e565" alt=""
- Now, let's try to execute some commands remotely from the brower.
- For instance, who is our user:
data:image/s3,"s3://crabby-images/ff1e5/ff1e5ddc7483c1e627e61d2b5f12d2db24967865" alt=""
- Where we are:
data:image/s3,"s3://crabby-images/58fb7/58fb7050dd5f7fd5e6e47fe5f7c98261dd0ab3cd" alt=""
- Listing the content of the folder /uploads we detect the presence of the text file flag5_abc123def.txt:
data:image/s3,"s3://crabby-images/d5f08/d5f0839fb6797c9465ee8ec88442f746eb096496" alt=""
- By the way, the rest of the content corresponds to other uploaded files during previous trials, as well as the current valid test.gif script.
- Reading flag5_abc123def.txt we find the flag5:
data:image/s3,"s3://crabby-images/4f11d/4f11dd6ad9acd757e8abe354fbb3f11941e6e335" alt=""
- Storing and decoding with base64, we find that the flag 5 is agentservices:
data:image/s3,"s3://crabby-images/a3297/a3297f4d818267c0d4d0eccb3e900ad842d49554" alt=""
data:image/s3,"s3://crabby-images/2321a/2321a84fb425b5e451fa26fbc18af4df2a841e8d" alt=""
7 - FLAG 6
7.1 - Locating the agent service
- Now, the flag 5 agentservices should be our passport to the final root access.
- Searching for some agent service, it seems that it works at port tcp/7788:
data:image/s3,"s3://crabby-images/3e1db/3e1db1c29a92382bf716f2d2f7c8bd01a02aad80" alt=""
- Locating the executable agent:
data:image/s3,"s3://crabby-images/e95eb/e95eb6d6c06d1afa60897c8cc0c996e0977d5ecf" alt=""
- Checking permissions for agent:
data:image/s3,"s3://crabby-images/e2eb6/e2eb6405ffa8c405ea1c3abc107b57a2df1659c5" alt=""
- Running it, and Agent ID is required:
data:image/s3,"s3://crabby-images/8a90a/8a90ab23cec280413cf1b1797f010681327af1c8" alt=""
7.2 - Port knocing
- Looking for more interesting content at /usr/local/bin we find an access_codes file:
data:image/s3,"s3://crabby-images/e5f9b/e5f9b96a992eeee2863f64649909f9d4e41806fc" alt=""
- Reading access_codes, it seems like a port knock sequence:
data:image/s3,"s3://crabby-images/b94fa/b94fa163d03ba4f65856c139e1022f37aca9d406" alt=""
- While port 7788 is filtered by default ...
data:image/s3,"s3://crabby-images/e67a6/e67a6563059809a429de904cd63da83205b1b2c0" alt=""
- ... after executing command knock (following the information provided by access_codes) the port 7788 is open:
data:image/s3,"s3://crabby-images/883a3/883a30babcc38f32d283b586799566ccbbd081d7" alt=""
data:image/s3,"s3://crabby-images/95fe7/95fe72a235fe82fc1ff318288b6251a4354dec3c" alt=""
7.3 - Running the agent service
-Setting a Netcat listening session on port 7788:
data:image/s3,"s3://crabby-images/51e50/51e5022dc4f6d07f8f856086f66eb3df50f32e10" alt=""
- Connecting with Netcat from Kali to IMF's port 7788, where agent service is running:
data:image/s3,"s3://crabby-images/70a76/70a764a73e813ceee435e019f258a638a09b5fa8" alt=""
- Let's transfer agent to our local machine:
data:image/s3,"s3://crabby-images/732cf/732cf4bfc99eeee0220b71d9f2eed535e70ee4ed" alt=""
data:image/s3,"s3://crabby-images/b35e5/b35e5f58afbbe3bf3ba2d0600bac1ec131827409" alt=""
- The transfer is successful:
data:image/s3,"s3://crabby-images/e7dd1/e7dd165d0fca7d741f861fd59125e423626d175c" alt=""
- What type of file is agent:
data:image/s3,"s3://crabby-images/5dde1/5dde18621fcc27bfd78817bbccba376a2603708a" alt=""
- Giving execution permissions and running locally agent:
data:image/s3,"s3://crabby-images/7e957/7e957f89ff00698f8cd74e7e7a5bdfba0337c5cf" alt=""
- Entering an invalid Agent ID like abcde:
data:image/s3,"s3://crabby-images/39118/39118d2ee7021421ac953cf728b0d4e1223ab869" alt=""
- Let's ltrace our program, entering an arbitrary ID:
data:image/s3,"s3://crabby-images/c54cf/c54cfd8ace0f8ded93d5a2cefb22000081fff903" alt=""
- It is noticeable that the function strncmp compares string abcde with string 48093572.
- Shall this be the valid Agent ID? The answer is yes:
data:image/s3,"s3://crabby-images/22a10/22a10fc738bfb44827e84884fd2df093ac1c513c" alt=""
7.4 - Getting a remote limited shell with weevely
- weevely is is a stealth PHP web shell that simulate telnet-like connection. It is an essential tool for web application post exploitation, and can be used as stealth backdoor or as a web shell to manage legit web accounts, even free hosted ones.
https://tools.kali.org/maintaining-access/weevely
- Using weevely let's generate a PHP script called prueba with the password secreto:
data:image/s3,"s3://crabby-images/c5c99/c5c993799bce75ab97bf3143ea28615456cffaf4" alt=""
data:image/s3,"s3://crabby-images/b7d76/b7d76a103f1ed8b1f0b2e80e0c58eba8701fdff8" alt=""
- Reading prueba:
data:image/s3,"s3://crabby-images/f97ef/f97efcc4ae5967641e49be5b6ac6d96750b5bcc7" alt=""
data:image/s3,"s3://crabby-images/72308/72308233e9507cd6f4d26f3906a477a9bf6d4091" alt=""
- Adding GIF89 to the top of the script and renaming with the extension .gif:
data:image/s3,"s3://crabby-images/89ceb/89cebef77de66441a014794e7142cd4154c77e06" alt=""
data:image/s3,"s3://crabby-images/68057/680570d92bfa56fb9bf945153c10dfbd0288238f" alt=""
- Now, it is time to upload prueba.gif to IMF:
data:image/s3,"s3://crabby-images/0f999/0f999da4117fb2ddf7f3cea77763fc18b5d6479a" alt=""
data:image/s3,"s3://crabby-images/6da3b/6da3b1c316c531ea6bc2f585c2b576e0ab653378" alt=""
- The upload is successful, and the server sends back a code referring to the uploaded script:
data:image/s3,"s3://crabby-images/c8b1d/c8b1d4de31defb7c40a3374e9432a69c185e6bc2" alt=""
- Launching weevely from Kali, we achieve a low privileged remote shell at IMF:
data:image/s3,"s3://crabby-images/d514f/d514fb569c7a3235c0393083f4fcf4ce077f2dbb" alt=""
7.5 - Privilege Escalation
- The final step will to exploit the executable agent with the goal of getting a root shell will be done in short.