PT-2026-42037 · Npm · @Apify/Actors-Mcp-Server
Publicado
2026-05-19
·
Atualizado
2026-05-19
·
CVE-2026-46341
CVSS v3.1
6.1
Média
| Vetor | AV:N/AC:L/PR:N/UI:R/S:C/C:L/I:L/A:N |
Summary
The
fetch-apify-docs tool validates URLs against a domain allowlist using String.startsWith() instead of proper URL hostname comparison. This allows bypass via attacker-controlled subdomains (e.g., https://docs.apify.com.evil.com/), enabling the tool to fetch and return arbitrary web content to the LLM.Details
Vulnerable component
src/tools/common/fetch apify docs.ts, line 51:typescript
const isAllowedDomain = ALLOWED DOC DOMAINS.some((domain) => url.startsWith(domain));src/const.ts, lines 167-170:typescript
export const ALLOWED DOC DOMAINS = [
'https://docs.apify.com',
'https://crawlee.dev',
] as const;How the bypass works
String.startsWith('https://docs.apify.com') matches any string beginning with that prefix, including:https://docs.apify.com.evil.com/payload- attacker-controlled subdomainhttps://docs.apify.com@evil.com/payload- userinfo component in URL (browser behavior varies, butfetch()in Node.js may follow this)https://docs.apify.com.evil.com:8080/path- custom port on attacker domain
All of these pass the
startsWith check because they begin with the exact string https://docs.apify.com.The fetched content is returned to the LLM
After the allowlist check passes, the tool fetches the URL and returns the full page content as markdown (
fetch apify docs.ts:69-103):typescript
const response = await fetch(url);
// ...
const html = await response.text();
markdown = htmlToMarkdown(html);
// ...
return buildMCPResponse({ texts: [`Fetched content from ${url}:
${markdown}`], ... });The HTML is converted to markdown and returned verbatim to the LLM. This creates a prompt injection vector - the attacker's page can contain instructions that the LLM may follow.
While tools like
get-html-skeleton have no domain allowlist at all - it accepts any URL. The fetch-apify-docs tool was clearly intended to be more restricted (documentation-only), but the startsWith check defeats that intent.PoC
json
{
"method": "tools/call",
"params": {
"name": "fetch-apify-docs",
"arguments": {
"url": "https://docs.apify.com.evil.com/prompt-injection-payload"
}
}
}The URL passes the
startsWith('https://docs.apify.com') check, fetches the attacker's page, and returns its content to the LLM.Impact
- Prompt injection via fetched content: Attacker hosts a page at
docs.apify.com.evil.comcontaining LLM instructions. When the tool fetches and returns this content, the LLM may follow the injected instructions. - Security boundary violation: The allowlist was explicitly designed to restrict fetching to trusted documentation domains. The bypass defeats this intent.
- SSRF (limited): The tool can fetch from attacker-controlled servers, though the primary risk is the content returned to the LLM rather than network access.
- Account compromise via meta.apifyToken: Injected prompt instructions can direct the LLM to include a specific
meta.apifyToken(the server's per-request token feature) in subsequentcall-actorinvocations, redirecting billable operations to a victim's account or accessing their private Actors
Correção
RCE
Encontrou algum problema na descrição? Tem algo a acrescentar? Fique à vontade para nos escrever 👾
Identificadores relacionados
Produtos afetados
@Apify/Actors-Mcp-Server