PT-2026-47628 · Pypi · Windows-Mcp

Published

2026-05-21

·

Updated

2026-05-21

·

CVE-2026-48989

CVSS v4.0

8.9

High

VectorAV:N/AC:L/AT:N/PR:N/UI:N/VC:H/VI:H/VA:H/SC:N/SI:N/SA:N/E:P
HTTP transports expose unauthenticated PowerShell control with wildcard CORS
There is an issue in the SSE and Streamable HTTP transport modes. The default stdio mode is not affected, but the documented HTTP modes expose the MCP control plane without authentication and add wildcard CORS handling around it. The same server exposes the PowerShell tool, which executes caller-controlled commands as the Windows user running Windows-MCP.
Relevant source:
  • src/windows mcp/ main .py:37-42: http middleware() installs OptionsMiddleware and CORSMiddleware with allow origins=["*"], allow methods=["*"], and allow headers=["*"].
  • src/windows mcp/ main .py:45-72: OptionsMiddleware responds to every OPTIONS request with wildcard Access-Control-Allow-Origin, Access-Control-Allow-Methods, and Access-Control-Allow-Headers.
  • src/windows mcp/ main .py:75-113: build mcp() constructs FastMCP(name="windows-mcp", ...) without an auth provider.
  • src/windows mcp/ main .py:139-151: both sse and streamable-http call mcp.run(...) with that middleware and no application-level auth/security settings.
  • src/windows mcp/tools/shell.py:10-24: registers the PowerShell tool and passes caller-controlled command to PowerShellExecutor.execute command.
  • src/windows mcp/desktop/powershell.py:176-204: executes that command through PowerShell -EncodedCommand.
  • README.md:421-424 and 433-434: documents the HTTP transports and describes Streamable HTTP as network-accessible HTTP streaming.
In an affected configuration, a client that can reach http://localhost:8000/mcp can initialize an MCP session and invoke tools/call for PowerShell. The issue is not just that PowerShell is powerful; it is that the HTTP control plane around that tool is unauthenticated and configured with wildcard CORS.

Root cause

The HTTP transport entry points compose two independent design decisions that fail-open together: the FastMCP instance is built without any authentication provider, and the middleware stack installs blanket wildcard CORS (allow origins=*, allow methods=*, allow headers=*) that explicitly permits cross-origin browsers and any non-browser HTTP client to reach the MCP control plane. Either one alone would be a partial defense; together, the unauthenticated control plane is reachable from arbitrary origins, with no host-validation, no token check, and no DNS-rebinding mitigation between an attacker's request and the registered PowerShell tool. The structural fix is to require an auth provider (token, mTLS, or local-only secret handshake) on the HTTP transports and to scope CORS to a specific operator-configured allowlist rather than applying wildcard policy to a tool surface that includes shell execution.

Auth boundary violated

Boundary: Network trust domain (an unauthenticated remote/cross-origin caller is treated as if it were a trusted local MCP client with rights to invoke privileged shell tools).
Respected at: stdio transport path (src/windows mcp/ main .py stdio branch) — the default transport relies on parent-process pipe ownership for caller identity, which is a real OS-level boundary.
Violated at: src/windows mcp/ main .py:139-151 (SSE and Streamable HTTP branches call mcp.run(...) with the wildcard-CORS middleware installed at :37-42 and no auth provider attached at :75-113). The boundary is silently dropped: there is no code path between the inbound HTTP request and tools/call for PowerShell (src/windows mcp/tools/shell.py:10-24src/windows mcp/desktop/powershell.py:176-204) that asserts caller identity or origin.
Minimal protocol proof from a matching FastMCP 3.2.4 harness with the same middleware posture:
text
$ curl -i -s -X OPTIONS 'http://127.0.0.1:18123/mcp/' 
 -H 'Origin: https://attacker.example' 
 -H 'Access-Control-Request-Method: POST' 
 -H 'Access-Control-Request-Headers: content-type,mcp-session-id'

HTTP/1.1 200 OK
access-control-allow-origin: *
access-control-allow-methods: *
access-control-allow-headers: *

$ curl -i -s 'http://127.0.0.1:18123/mcp' 
 -H 'Origin: https://attacker.example' 
 -H 'Content-Type: application/json' 
 -H 'Accept: application/json, text/event-stream' 
 --data '{"jsonrpc":"2.0","id":1,"method":"initialize","params":{"protocolVersion":"2025-06-18","capabilities":{},"clientInfo":{"name":"evil-page","version":"1"}}}'

HTTP/1.1 200 OK
content-type: text/event-stream
mcp-session-id: c67be0098b7643eb961b2fd0185ee043
access-control-allow-origin: *

$ curl -i -s 'http://127.0.0.1:18123/mcp' 
 -H 'Origin: https://attacker.example' 
 -H 'Mcp-Session-Id: c67be0098b7643eb961b2fd0185ee043' 
 -H 'Content-Type: application/json' 
 -H 'Accept: application/json, text/event-stream' 
 --data '{"jsonrpc":"2.0","id":2,"method":"tools/call","params":{"name":"PowerShell","arguments":{"command":"calc.exe","timeout":30}}}'

HTTP/1.1 200 OK
content-type: text/event-stream
mcp-session-id: c67be0098b7643eb961b2fd0185ee043
access-control-allow-origin: *

event: message
data: {"jsonrpc":"2.0","id":2,"result":{"content":[{"type":"text","text":"executed: calc.exe"}],"structuredContent":{"result":"executed: calc.exe"},"isError":false}}

Impact

For affected HTTP-transport deployments, successful exploitation gives arbitrary PowerShell execution as the user running Windows-MCP. There is an important browser caveat: current Chrome/Edge Local Network Access / Private Network Access behavior may block or prompt for public-site-to-localhost requests because this middleware does not return Access-Control-Allow-Private-Network: true. The exposure still applies to same-origin/private-origin contexts, browsers or apps without that enforcement, user-approved local-network prompts, browser extensions, and non-browser HTTP clients.
Suggested fix: require authentication for HTTP transports, remove wildcard CORS from MCP control endpoints, restrict origins to explicit trusted clients, and enable/propagate transport security settings such as host validation. If unauthenticated HTTP is retained for development, I would make it an explicit unsafe flag and add regression tests for cross-origin OPTIONS, initialize, and tools/call.

Fix

Missing Authentication

Found an issue in the description? Have something to add? Feel free to write us 👾

Weakness Enumeration

Related Identifiers

CVE-2026-48989
GHSA-VRXG-GM77-7Q5G

Affected Products

Windows-Mcp