ghsa-j3rg-3rgm-537h
Vulnerability from github
Summary
Directus versions <=9.22.4 is vulnerable to Server-Side Request Forgery (SSRF) when importing a file from a remote web server (POST to /files/import
). An attacker can bypass the security controls that were implemented to patch vulnerability CVE-2022-23080 by performing a DNS rebinding attack and view sensitive data from internal servers or perform a local port scan (eg. can access internal metadata API for AWS at http://169.254.169.254
event if 169.254.169.254
is in the deny IP list).
Details
DNS rebinding attacks work by running a DNS name server that resolves two different IP addresses when a domain is resolved simultaneously. This type of attack can be exploited to bypass the IP address deny list validation that was added to /api/src/services/file.ts
for the function importOne
to mitigate the previous SSRF vulnerability CVE-2022-23080. The validation in /api/src/services/file.ts
first checks if the resolved IP address for a domain name does not a resolve to an IP address in the deny list:
```js let ip = resolvedUrl.hostname;
if (net.isIP(ip) === 0) {
try {
ip = (await lookupDNS(ip)).address;
} catch (err: any) {
logger.warn(err, Couldn't lookup the DNS for url ${importURL}
);
throw new ServiceUnavailableException(Couldn't fetch file from url "${importURL}"
, {
service: 'external-file',
});
}
}
if (env.IMPORT_IP_DENY_LIST.includes('0.0.0.0')) { const networkInterfaces = os.networkInterfaces();
for (const networkInfo of Object.values(networkInterfaces)) {
if (!networkInfo) continue;
for (const info of networkInfo) {
if (info.address === ip) {
logger.warn(`Requested URL ${importURL} resolves to localhost.`);
throw new ServiceUnavailableException(`Couldn't fetch file from url "${importURL}"`, {
service: 'external-file',
});
}
}
}
}
if (env.IMPORT_IP_DENY_LIST.includes(ip)) {
logger.warn(Requested URL ${importURL} resolves to a denied IP address.
);
throw new ServiceUnavailableException(Couldn't fetch file from url "${importURL}"
, {
service: 'external-file',
});
}
```
Once it validates that the resolved IP address is not in the deny list, then it uses axios
to GET
the url and saves the response content.
js
try {
fileResponse = await axios.get<Readable>(encodeURL(importURL), {
responseType: 'stream',
});
} catch (err: any) {
logger.warn(err, `Couldn't fetch file from url "${importURL}"`);
throw new ServiceUnavailableException(`Couldn't fetch file from url "${importURL}"`, {
service: 'external-file',
});
}
However, this validation check and fetching the web resource causes to DNS queries that enable a DNS rebinding attack. On the first DNS query, an attacker controlled name server can be configured to resolve to an external IP address that is not in the deny list to bypass the validation. Then when axios
is called, the name server resolves the domain name to a local IP address.
PoC
To demonstrate we will be using an online tool named rebinder. Rebinder randomly changes the IP address it resolves to depending on the subdomain. For an example, 7f000001.8efa468e.rbndr.us
can resolve to either 142.250.70.142
(google.com) or 127.0.0.1
. Sending multiple POST
requests to /files/import
using this domain will eventually cause a resolution to 142.250.70.142
first to bypass the validation then fetch the sensitive from an internal server when axios
is called.
The following screenshots show what it looks like when a successful attack occurs.
Downloading a file named secret.txt
from a webserver running from http://127.0.0.1/secret.txt
Receiving the request from the internal server. Note that the incoming connection is from 127.0.0.1.
After downloading the file it leaks the content of the secret file.
Impact
An attacker can exploit this vulnerability to access highly sensitive internal server and steal sensitive information. An example is on Cloud Environments that utilise internal APIs for managing machine and privileges. For an example, if directus
is hosted on AWS EC2 instance and has an IAM role assigned to the EC2 instance then an attacker can exploit this vulnerability to steal the AWS access keys to impersonate the EC2 instance using the AWS API.
{ "affected": [ { "package": { "ecosystem": "npm", "name": "directus" }, "ranges": [ { "events": [ { "introduced": "0" }, { "fixed": "9.23.0" } ], "type": "ECOSYSTEM" } ] } ], "aliases": [ "CVE-2023-26492" ], "database_specific": { "cwe_ids": [ "CWE-918" ], "github_reviewed": true, "github_reviewed_at": "2023-03-03T23:07:35Z", "nvd_published_at": "2023-03-03T22:15:00Z", "severity": "MODERATE" }, "details": "### Summary\nDirectus versions \u003c=9.22.4 is vulnerable to Server-Side Request Forgery (SSRF) when importing a file from a remote web server (POST to `/files/import`). An attacker can bypass the security controls that were implemented to patch vulnerability [CVE-2022-23080](https://security.snyk.io/vuln/SNYK-JS-DIRECTUS-2934713) by performing a [DNS rebinding attack](https://en.wikipedia.org/wiki/DNS_rebinding) and view sensitive data from internal servers or perform a local port scan (eg. can access internal metadata API for AWS at `http://169.254.169.254` event if `169.254.169.254` is in the deny IP list).\n\n### Details\nDNS rebinding attacks work by running a DNS name server that resolves two different IP addresses when a domain is resolved simultaneously. This type of attack can be exploited to bypass the IP address deny list validation that was added to [`/api/src/services/file.ts`](https://github.com/directus/directus/blob/main/api/src/services/files.ts) for the function `importOne` to mitigate the previous SSRF vulnerability [CVE-2022-23080](https://security.snyk.io/vuln/SNYK-JS-DIRECTUS-2934713). The validation in [`/api/src/services/file.ts`](https://github.com/directus/directus/blob/main/api/src/services/files.ts) first checks if the resolved IP address for a domain name does not a resolve to an IP address in the deny list:\n\n```js\nlet ip = resolvedUrl.hostname;\n\nif (net.isIP(ip) === 0) {\n try {\n ip = (await lookupDNS(ip)).address;\n } catch (err: any) {\n logger.warn(err, `Couldn\u0027t lookup the DNS for url ${importURL}`);\n throw new ServiceUnavailableException(`Couldn\u0027t fetch file from url \"${importURL}\"`, {\n service: \u0027external-file\u0027,\n });\n }\n}\n\nif (env.IMPORT_IP_DENY_LIST.includes(\u00270.0.0.0\u0027)) {\n const networkInterfaces = os.networkInterfaces();\n\n for (const networkInfo of Object.values(networkInterfaces)) {\n if (!networkInfo) continue;\n\n for (const info of networkInfo) {\n if (info.address === ip) {\n logger.warn(`Requested URL ${importURL} resolves to localhost.`);\n throw new ServiceUnavailableException(`Couldn\u0027t fetch file from url \"${importURL}\"`, {\n service: \u0027external-file\u0027,\n });\n }\n }\n }\n}\n\nif (env.IMPORT_IP_DENY_LIST.includes(ip)) {\n logger.warn(`Requested URL ${importURL} resolves to a denied IP address.`);\n throw new ServiceUnavailableException(`Couldn\u0027t fetch file from url \"${importURL}\"`, {\n service: \u0027external-file\u0027,\n });\n}\n```\n\nOnce it validates that the resolved IP address is not in the deny list, then it uses `axios` to `GET` the url and saves the response content.\n\n```js\ntry {\n fileResponse = await axios.get\u003cReadable\u003e(encodeURL(importURL), {\n responseType: \u0027stream\u0027,\n });\n} catch (err: any) {\n logger.warn(err, `Couldn\u0027t fetch file from url \"${importURL}\"`);\n throw new ServiceUnavailableException(`Couldn\u0027t fetch file from url \"${importURL}\"`, {\n service: \u0027external-file\u0027,\n });\n}\n```\n\nHowever, this validation check and fetching the web resource causes to DNS queries that enable a DNS rebinding attack. On the first DNS query, an attacker controlled name server can be configured to resolve to an external IP address that is not in the deny list to bypass the validation. Then when `axios` is called, the name server resolves the domain name to a local IP address.\n\n### PoC\nTo demonstrate we will be using an online tool named [rebinder](https://lock.cmpxchg8b.com/rebinder.html). Rebinder randomly changes the IP address it resolves to depending on the subdomain. For an example, `7f000001.8efa468e.rbndr.us` can resolve to either `142.250.70.142` (google.com) or **`127.0.0.1`**. Sending multiple `POST` requests to `/files/import` using this domain will eventually cause a resolution to `142.250.70.142` first to bypass the validation then fetch the sensitive from an internal server when `axios` is called.\n\nThe following screenshots show what it looks like when a successful attack occurs.\n\n*Downloading a file named `secret.txt` from a webserver running from `http://127.0.0.1/secret.txt`*\n![image](https://user-images.githubusercontent.com/6276577/218124035-26f7f0c3-47b3-424d-b4d4-bd3b47161983.png)\n\n*Receiving the request from the internal server. Note that the incoming connection is from **127.0.0.1**.*\n![image](https://user-images.githubusercontent.com/6276577/218124119-87b8d5d6-934d-4e07-be4d-066616a9a435.png)\n\n*After downloading the file it leaks the content of the secret file.*\n![image](https://user-images.githubusercontent.com/6276577/218122210-87b2e478-1081-4830-a9ea-e5d9f39bb129.png)\n\n### Impact\nAn attacker can exploit this vulnerability to access highly sensitive internal server and steal sensitive information. An example is on Cloud Environments that utilise internal APIs for managing machine and privileges. For an example, if `directus` is hosted on AWS EC2 instance and has an IAM role assigned to the EC2 instance then an attacker can exploit this vulnerability to steal the AWS access keys to impersonate the EC2 instance using the AWS API.\n", "id": "GHSA-j3rg-3rgm-537h", "modified": "2023-03-06T01:44:06Z", "published": "2023-03-03T23:07:35Z", "references": [ { "type": "WEB", "url": "https://github.com/directus/directus/security/advisories/GHSA-j3rg-3rgm-537h" }, { "type": "ADVISORY", "url": "https://nvd.nist.gov/vuln/detail/CVE-2023-26492" }, { "type": "WEB", "url": "https://github.com/directus/directus/commit/ff53d3e69a602d05342e15d9bb616884833ddbff" }, { "type": "PACKAGE", "url": "https://github.com/directus/directus" }, { "type": "WEB", "url": "https://github.com/directus/directus/releases/tag/v9.23.0" } ], "schema_version": "1.4.0", "severity": [ { "score": "CVSS:3.1/AV:N/AC:L/PR:L/UI:N/S:C/C:L/I:N/A:N", "type": "CVSS_V3" } ], "summary": "Directus vulnerable to Server-Side Request Forgery On File Import" }
Sightings
Author | Source | Type | Date |
---|
Nomenclature
- Seen: The vulnerability was mentioned, discussed, or seen somewhere by the user.
- Confirmed: The vulnerability is confirmed from an analyst perspective.
- Exploited: This vulnerability was exploited and seen by the user reporting the sighting.
- Patched: This vulnerability was successfully patched by the user reporting the sighting.
- Not exploited: This vulnerability was not exploited or seen by the user reporting the sighting.
- Not confirmed: The user expresses doubt about the veracity of the vulnerability.
- Not patched: This vulnerability was not successfully patched by the user reporting the sighting.