An asynchronous proxy to proxy HTTP traffic through AWS API Gateway and rotate IP address on each request.
- DOUBLETAP
This is a mitmproxy addon that allows you to dynamically proxy HTTP traffic through AWS API Gateway in order to rotate IP address on each request.
Essentially, it's a fully weaponized version of the underlying concept which tools such as FireProx and the IP_Rotate Burp extension have implemented albeit with a lot major improvements:
- Written in Python 3
- Fully asynchronous (which is extremely important for this particular implementation to work efficiently)
- Works just like a regular proxy
- Dynamically creates, stages and deploys AWS API Gateway endpoints for each new domain across multiple AWS regions concurrently and transparently redirects traffic.
When it comes to rotating IPs, there are a lot of ways of doing the same thing. This approach offers major benefits over the traditional methods but obviously there are also some cons.
-
Much easier to setup and use (you literally just need an AWS account, that's it).
-
Cost-wise it's pretty much free unless you go above 1 million requests a month. See the AWS API Gateway pricing page for details.
-
You have a much greater "pool of IPs" compared to the other approaches. The "pool" is even bigger when using multiple AWS regions (which DOUBLETAP does).
-
Connection speeds are extremely fast as the "proxying" is just an HTTP request to an AWS endpoint. There is basically no network overhead compared to other approaches.
-
You're "proxying" through a highly reputable and trusted (trust is obviously subjective) entity as supposed to a random service on the internet that may or may not be maliciously modifying/intercepting your traffic.
-
The IPs that the end service sees are in the AWS network range which are generally trusted.
-
You can only proxy HTTP, HTTP/2 and Websocket traffic. Not arbitrary TCP (currently DOUBLETAP only supports HTTP due to some mitmproxy limitations)
-
Can be somewhat easily detected by looking for a specific header that cannot be removed (see the Defense & Detection section for more details)
-
Does not work against other services hosted on AWS API Gateway
-
It can take up to ~30 seconds to receive back a response when issuing a request to a new domain/URL. Subsequent requests to the same domain/URL will have normal response times (see the limitations section for more details)
mitmproxy exposes an addon system which allows you to create components of any complexity that interact with it's proxy engine.
When you first fire up DOUBLETAP, it'll query AWS API Gateway to see if there's already an existing API called "DOUBLETAP" (by default) which was previously setup by the tool, and if so pulls down a list of each API endpoint and the domains they proxy traffic to so that it doesn't create them again.
The real "magic" comes into play when you send an HTTP request through the proxy. DOUBLETAP works by hooking the mitmproxy request
event, which fires every time the proxy receives a HTTP request, it then performs the following actions:
- Constructs the root URL from the URL you requested
- Checks an internal dictionary structure to see if it already setup an API Gateway endpoint that proxies traffic to that domain
- If not, it'll concurrently create, stage and deploy a new API Geteway endpoint across multiple AWS regions (10 by default) to proxy traffic to the domain you requested (this makes the IP rotation rate even higher).
- During the API Gateway endpoint creation, it'll remap the
X-Forwarded-For
header: this allows us to specify an IP of our choosing that the end server will see in that header. - DOUBLETAP can detect when the API is fully staged (usually takes anywhere between 10-30 seconds) and will only allow the requests to proceed once the API endpoints are ready across all regions.
- It'll then pick at random an API Gateway endpoint URL from the ones it created
- Generate a fake IP for the
X-Forwarded-for
header and change the User Agent of the request - Finally it'll seamlessly redirect the modified request to the chosen API Gateway endpoint URL which will then proxy the request to the actual target.
The result is that the server you're hitting will see a truly unique IP on almost each request. Additionally it will not give away your real IP address through the X-Forwarded-For
header as it's supplied a bogus IP.
What's super important to note here is that all of this happens asynchronously in the background, meaning the proxy does not block every time it has to interact with AWS and/or on each HTTP request.
A fundamental limitation of this technique/implementation is that whenever you request a URL to a new domain that DOUBLETAP hasn't created an API Gateway endpoint for previously, it takes anywhere between 10-30 seconds before the Gateway endpoint is staged and ready to accept traffic.
Practically speaking, this means an HTTP request to a new domain/URL will just sit there doing nothing for up to 30ish seconds until you receive back any data. Obviously, subsequent requests to that same domain/URL will not have this issue and you'll receive back the response instantly.
As far as I'm aware, there really isn't a way around this. Additionally, AWS doesn't provide a reliable way to determine whether an API Gateway endpoint has finished staging or not. DOUBLETAP handles this by spinning up background AsyncIO tasks that perform an HTTP request to the endpoint URLs and do a signature check on the response looking for specific status codes and data that I've found through testing mean the API is still staging.
To help alleviate this limitation, see the section on the prestage
and allowlist
options.
Additionally:
- HTTP/2 requests are not supported. mitmproxy doesn't have the ability to redirect HTTP/2 connections (yet). However, AWS API Gateway does support HTTP/2 and Websocket connections so mitmproxy just needs to catch up.
- Incredibly, if the end service you're trying access legitimately uses AWS API Gateway, the proxying won't work. It's like a cloud Judo move. Thankfully, not a lot of things use AWS API Gateway as I've only ran into this once in a year or so of using this tool.
- The new SprayingToolkit update is built to support proxying everything through DOUBLETAP. No more IP blacklisting on password sprays π
- ENTROPICFORESIGHT is built to support proxying everything through DOUBLETAP. No more API keys and/or rate limiting when trying to scrape OSINT data π
- Scraping with Headless Browsers (Note: You're going to want to use the
allowlist
and/or theprestage
options if you do this. See this section.) - Anything that can benefit from a new IP on each request π make the possibilities flow through you.
- The underlying technique can be detected by looking for the
x-amz-apigw-id
header which is sent on each request through AWS API Gateway. There is no way to avoid this. (See the Defense & Detection section for more details) - While the IP on each request does change most of the time there is always a slight possibility it doesn't as this isn't a "legit" or predictable feature of AWS API Gateway. Either way, your real IP won't ever be revealed.
- By default, a random IP is generated and sent along in the
X-Forwarded-For
header using the Python Faker library.
While the IP address does change on each connection, there are some things that are "baked" into how AWS API Gateway works that can be used to detect this technique.
Note: all of the below is from my current understanding of how things work which is subject to change π. I'm by no means an AWS expert. Feel free to reach out if something is inaccurate.
The most effective way of identifying the underlying technique of this tool (this is a non-fragile detection) is to look for the x-amz-apigw-id
header in HTTP requests. This header contains a Base64 encoded value which identifies the API Gateway being used and there isn't a way to overwrite it/remove it (unlike the X-Forwarded-For
header). This can be used to identify the AWS account which has deployed the specific gateway, so you could give the header value to AWS in order to file an abuse complaint if needed.
You probably shouldn't be receiving HTTP requests from AWS API Gateway anyway so I feel pretty confident in saying this is safe way of blocking/detecting this technique. Creating an IDS/IPS rule looking for that header should be pretty trivial.
If you're using AWS API Gateway legitimately to host your service in the first place, you're implicitly safe from this technique as proxying to another service hosted on AWS API Gateway won't work! I call this Cloud Judo.
Finally, by default a fake IP is generated and sent along in the X-Forwarded-For
header using the Python Faker library. You could check this IP to make sure it's valid and correlate it with WHOIS data. Or just take a look at the Faker library closer to see if there's a way of predicting the IPs it generates.
docker build github.com/Porchetta-Industries/DOUBLETAP
git clone https://github.com/Porchetta-Industries/DOUBLETAP && cd DOUBLETAP
pip3 install -r requirements.txt
DOUBLETAP needs AWS credentials in order to interact with AWS API Gateway. By default it'll look for the AWS_ACCESS_KEY_ID
and AWS_SECRET_ACCESS_KEY
environment variables containing the AWS access key and AWS secret key respectively. If it doesn't find those environment variables, it'll try to grab the access and secret key from the ~/.aws/credentials
file which is setup when you install the AWS CLI utility.
The proxy will bind on all interfaces on port 8080
by default.
docker run -p 8080:8080 -e AWS_ACCESS_KEY_ID=$AWS_ACCESS_KEY -e AWS_SECRET_ACCESS_KEY=$AWS_SECRET_KEY --rm -it $IMAGE_ID
Optionally you can put the environment variables in a .env
file and use that:
# Print out the .env file
$ cat .env
AWS_ACCESS_KEY_ID=<my_aws_access_key_id>
AWS_SECRET_ACCESS_KEY=<my_aws_secret_access_key>
# Run DOUBLETAP and pass it the .env file
$ docker run -p 8080:8080 --env-file .env --rm -it $IMAGE_ID
You can also start DOUBLETAP directly using the mitmdump
utility which comes with the mitmproxy
Python package. This is useful if you want to avoid using the docker image.
mitmdump --no-http2 -k -s doubletap.py
There are a few options you can pass to DOUBLETAP in order to customize the proxying behavior, prestage proxies and/or allow only certain domains/URLs to be proxied. To pass option(s) to the proxy you have to use the --set
flag followed by option=value
. (This is just how mitmproxy addons work)
Examples:
# Using mitmdump directly
$ mitmdump --no-http2 -k -s doubletap.py --set allowlist='.*slashdot.org,.*google.com' --set cleanup=true --set prestage='https://www.slashdot.org'
# Using the docker image
$ docker run -p 8080:8080 --env-file .env --rm -it $IMAGE_ID --set allowlist='.*slashdot.org' --set cleanup=true --set prestage=~/urls.txt
The allowlist
option accepts a file containing regexes (one per line) or a comma separated list of regex(s). If this option is given, DOUBLETAP will only proxy requests if the root URL or domain matches one of the regexes.
This is extremely useful when proxying headless browsers through DOUBLETAP as webpages tend to load a bunch of resources from third-party domains. Because of the ~30 seconds you have to wait for each new URL/domain, it would take forever to load a webpage through a headless browser without this option.
As you might think, the cleanup
option will destroy all existing proxies in AWS so you have a "clean slate" to start with.
Only accepts true
, defaults to false
.
Note: it can take up to a minute for the resource to get destroyed on AWS's side. Unfortunately there's no way of checking this from DOUBLETAP.
The prestage
option accepts a file of root URLs (one per line) or a comma separated list of root URLs that will be "prestaged" before the proxy starts up.
Practically speaking, prestaging URLs sets up the proxies in AWS before hand so you don't have to wait those ~30 seconds before getting back a response when you start proxying traffic.
This really comes down to what you're trying to do/tool you're using. Generally, most tools have HTTP proxy support. You can also use ProxyChains to "force" something to use a proxy.
Here's some commands I recommend trying to test the proxy is working and to get an idea how things work:
Note: please read the limitations section. First time you request a URL/domain it can take up to 30 seconds to receive back a response. Depending on the tool, you might need to set a higher timeout threshold in order to accommodate this.
curl --insecure -x http://127.0.0.1:8080 -v https://ifconfig.me/all
http --verify no -v --proxy http://127.0.0.1:8080 https://ifconfig.me/all
If you run one of the above commands, you should see a new IP in the response on each request π₯π
As of v1.5.0, WitnessMe supports proxies. You can combine WitnessMe with DOUBLETAP to screenshot webpages while rotating IPs! Super useful to assess attack surfaces on Red Teams.
In the following example our target is contoso.com
. We've done some OSINT already and gathered a list of subdomains that we want to screenshot.
In one terminal window start DOUBLETAP:
docker run -p 8080:8080 --env-file .env --rm -it $IMAGE_ID --set allowlist='.*contoso.com'
In another terminal window, start WitnessMe and give it the HTTP_PROXY
environment variable pointing to DOUBLETAP:
HTTP_PROXY=http://127.0.0.1:8080 witnessme --timeout 40 screenshot /my_contoso_subdomains.txt
The above command will force WitnessMe to proxy all traffic through DOUBLETAP and sets the timeout to 40 seconds so the connection doesn't time out before the proxy is setup in AWS.
To Do
Implement "domain/URL pre-loads"Implement "domain/URL allow/deny" support with regexes- Allow customization of how DOUBLETAP chooses the API Gateway proxy URL (e.g. Round-Robin as supposed to at random)
- Allow customization of User-Agent replacement.
- Allow customization of how the bogus IP in the
X-Forwarded-For
header is generated Expose a "cleanup" command to remove stages from API GatewayBetter logging