GraphStrike is a suite of tools that enables Cobalt Strike’s HTTPS Beacon to use Microsoft Graph API for C2 communications. All Beacon traffic will be transmitted via two files created in the attacker’s SharePoint site, and all communications from Beacon will route
Why?
Threat intelligence has been released regarding several different APTs leveraging Microsoft Graph API and other Microsoft services for offensive campaigns:
- BLUELIGHT – APT37/InkySquid/ScarCruft
- Graphite – APT28/Fancy Bear
- Graphican – APT15/Nickel/The Flea
- SiestaGraph – UNKNOWN
Threat actors continue to leverage legitimate services for illegitimate purposes.
Utilizing a high-reputation domain like graph.microsoft.com for C2 communications is extremely effective and desirable, but often complicated and prohibitive from a time and effort standpoint.
Most C2 frameworks do not support methods to fetch or rotate access tokens, which makes them unable to use Graph API.
This can make it difficult for red teams to replicate these techniques, and deprives defenders of a chance to observe and develop signatures for this kind of activity.
GraphStrike seeks to ease that burden and provide a reliable and repeatable process to leverage Microsoft Graph API while keeping the familiarity and reliability of the Cobalt Strike user experience.
Is This An External C2?
Not technically, no. Having previously built a true External C2 using Graph API (which sent Beacon traffic as Microsoft Teams messages), the burden of having to develop, maintain, and integrate a custom implant that meets the External C2 specification and gets the job done is all too familiar.
GraphStrike instead leverages an open source User Defined Reflective Loader(UDRL) called AceLdr by Kyle Avery (adapted as ‘GraphLdr’ in this project) to hook the WinINet library calls that Beacon normally makes and manipulate them as neccessary in order to use Graph API.
There is no custom implant or additional process to speak of, just the Beacon process with a couple of hooked Windows API’s.
On the server side there is a Python3 program that translates Cobalt Strike Team Server traffic into Graph API traffic and vice-versa.
Features
GraphStrike supports almost all normal Cobalt Strike activities to include:
- Use of Proxychains through a Cobalt Strike SOCKS proxy (though it is very slow…)
- Upload/Download of large files
- BOFs, execute-assembly, etc.
This also includes GraphStrike integration of the sleep, exit, and remove commands to match GraphStrike Server sleep times with Beacon as well as delete files in SharePoint when a Beacon is exited or removed.
GraphStrike additionally incorporates all of the features and functionality of the original AceLdr, with some additional API’s made to utilize call stack spoofing as well.
Requirements
GraphStrike requires the following before you get started:
- A Microsoft Azure tenant with a SharePoint/O365 license assigned + site created. The default site is fine.
- An Azure account with Global Administrator permissions in that tenant.
- Python 3.8-3.11Note #4 (and additional dependencies that will be installed during the setup process)
Firewall Rules
- Ensure that each machine that the Cobalt Strike client runs on is able to connect to the Cobalt Strike team server machine on ports 443 and 5000.
Setup
Make note of the following before proceeding with the setup process:
1. Certain components utilize relative paths to locate other assets. Please change directories as instructed below.
2. The Cobalt Strike profile may only be edited BEFORE step 5 in the below setup processNote #1.
On The Machine That Will Run The Cobalt Strike Team Server:
- Clone the repo.
- From the repo directory, run
sudo setup/install_dependencies.sh
to install required system dependencies. - Run
python3 -m venv virtual
and thensource virtual/bin/activate
to create and then enter the virtual environment. - Change to the setup directory and run
pip3 install -r requirements.txt
. - Run
./provisioner.py new
and complete the setup process. - Start the Cobalt Strike team server using graphstrike.profile as the malleable C2 profile.
- Start a Cobalt Strike client instance (you can do this on a client machine, or on the TS box and kill it afterwards) and create a Cobalt Strike HTTPS listener on port 443 with
graph.microsoft.com
as the HTTPS Hosts and HTTPS Host(Stager) fields. - Change back to the primary repo directory and run the GraphStrike Server using
./GraphStrike.py
.
On ALL Machines That Will Run The Cobalt Strike Client:
- Copy the GraphStrike/client directory to the client machine from the TS machine. This must be done only AFTER completing provisioning!
- Import GraphStrike.cna to Cobalt Strike using the Script Manager.
- Create Cobalt Strike payloads, whether that be raw shellcode or compiled artifacts using the Artifact Kit or an alternate payload generation framework. Artifact Kit users see below!
- Profit.
Artifact Kit Users
Due to the size of GraphLdr, users of the Artifact Kit will need to re-compile it with specific options in order for GraphStrike to be compatible with Artifact Kit generated payloads.
Specifically, the ‘Stage Size’ and ‘RDLL Size’ fields need to be specified so as to use the 100K RDLL size. Two examples of working syntax are provided below:
./build.sh pipe VirtualAlloc 505029 100 false false none /opt/cobaltstrike/artifacts
./build.sh peek HeapAlloc 492376 100 false true indirect /opt/cobaltstrike/artifacts
Cleanup
On The Machine That Is Running The TS + GraphStrike Server:
- Stop the GraphStrike server
- Change back to the setup directory and run
./provisioner.py delete
to remove created Azure assets.
Notes
In no particular order, here are a few suggestions and observations to help use GraphStrike to it’s full potential.
- The profile included with GraphStrike is very minimalistic; this is by design. Changing any of the EXISTING fields in the profile may/will break GraphStrike! You should be able to add additional profile language/behaviour to other sections that are not already defined (.e.g customize pipe name, injection behaviour, etc). Any edits to the profile MUST be made prior to running the provisioner!
- The Azure application that is used for C2 communications by both Beacon and the GraphStrike Server is rate-limited to 1200 requests/min. The GraphStrike Server uses 120/min as a baseline to function.
- The lower a Beacon’s sleep time is the more requests it will make; additionally, each Beacon created using GraphStrike is going to be using some of that 1200/min limit.
- Going interactive with a Beacon is doable, but going interactive with more than one Beacon probably isn’t. If you run into rate limiting issues, consider increasing the sleep time for your beacons, decreasing the number of Beacon’s you have running, or both.
- The lower a Beacon’s sleep time is the more requests it will make; additionally, each Beacon created using GraphStrike is going to be using some of that 1200/min limit.
- While the GraphStrike Server’s sleep time changes on a per-Beacon basis according to issued sleep commands, what this really means is that the GraphStrike Server will sleep for the specified time before checking in with the TS for tasking.
- It does NOT mean that Beacon will immediately receive and process the tasking as soon as it is retrieved from the TS by the GraphStrike server.
- Beacon will sleep the specified time before reaching out to SharePoint to retrieve TS tasking, but due to the nature of async C2 this will not be in lockstep with when the GraphStrike Server uploads it.
- It does NOT mean that Beacon will immediately receive and process the tasking as soon as it is retrieved from the TS by the GraphStrike server.
- If a Beacon dies without having exited gracefully (AV, it crashes, etc), the Beacon will appear to still be calling into the TS, and the fact that it is dead will only become apparent once you issue it a command.
- What is really connecting to the TS / making it appear that the Beacon is still calling in is the GraphStrike server, so this really isn’t a reflection on the health of a Beacon. Such is the nature of async C2.
- GraphStrike works on a 1:1:1 model; 1 SharePoint site is associated with 1 GraphStrike server which is associated with 1 TS. You’ll have issues if you try connecting two TS/GraphStrike servers to a single SharePoint site.
- You can of course connect multiple Cobalt Strike clients to a single TS / GraphStrike server, each client just needs a copy of the ‘client’ folder produced by the provisioning process.
- There is a known issue regarding compatibility of the az utility used by GraphStrike and Python 3.12.
- I’d recommend you review the documentation for AceLdr, as all of the notes from that project apply here as well.
Limitations
The following limitations exist in GraphStrike:
- Only x64 Beacons are supported.
- Staged Beacons are not supported.
- GraphStrike is only compatible with the WinINet library; the new WinHTTP library option for Beacons is not supported.
- No support for issuing a sleep command via Beacon’s right-click menu. Sleep beacons using the command line option instead.
- GraphStrike is only supported on Linux instances of Cobalt Strike. Windows support is certainly possible to implement, and is really just a matter of changing around some paths within the Python files and Aggressor script.