jupyterlab-git extension: Stored XSS leading to RCE
Description
Overview
Amazon Web Services (AWS) Security has identified a stored cross-site scripting (XSS) issue in the jupyterlab-git JupyterLab extension that can lead to remote code execution (RCE). The issue exists in the PlainTextDiff.ts component, where the createHeader() method passes Git filenames directly to innerHTML without sanitization when rendering diffs for renamed files in commit history. This allows an adversary to craft a filename containing arbitrary HTML/JavaScript that executes when another user views the rename diff in the Git History tab.
The issue can be leveraged through the rename history view in the JupyterLab Git panel. An adversary creates a file with a crafted filename containing a JavaScript payload (e.g., .py), renames the file in a subsequent commit, and pushes to a shared repository. When a victim clones the repository, navigates to the Git History tab, clicks the rename commit, and then clicks the renamed file to view the diff, the unsanitized filename renders via innerHTML, executing arbitrary JavaScript in the victim's browser session. The injected JavaScript reads the xsrf cookie, opens a JupyterLab terminal via POST /api/terminals, connects via WebSocket, and executes arbitrary shell commands — achieving full RCE. An adversary can leverage this to exfiltrate secrets or credentials from the victim's environment.
Scope of impact
We discovered this issue during internal security testing. The issue is present in the default configuration of JupyterLab when the jupyterlab-git extension is installed.
The attack requires:
- The adversary to have commit access to a Git repository that the victim has cloned
- The victim to navigate to the Git History tab, click the rename commit, and click the renamed file to view the diff
The issue could allow an actor who has access to a shared Git repository to execute arbitrary JavaScript in another user's JupyterLab environment by committing a file with a crafted filename, potentially leading to remote code execution with access to user code, data, environment variables, and credentials.
Proof of concept
The issue exists in the createHeader() method where filenames from rename history are passed directly to innerHTML without sanitization:
[1] https://github.com/jupyterlab/jupyterlab-git/blob/main/src/components/diff/PlainTextDiff.ts#L214
Attack flow:
- An adversary creates a file with a crafted filename containing a JavaScript payload, e.g., .py
- The adversary renames the file in a subsequent commit and pushes both commits to a shared Git repository
- The victim clones or pulls the repository and navigates to the Git History tab in JupyterLab
- The victim clicks the rename commit, then clicks the renamed file to view the diff
- The createHeader() method constructs a diff header using string concatenation with the unsanitized filename and assigns the result to innerHTML
- The injected JavaScript executes in the victim's browser session, reads the _xsrf cookie, sends a POST request to /api/terminals to open a JupyterLab terminal, connects via WebSocket, and executes arbitrary shell commands
Proof-of-concept mitigation
The issue can be mitigated by replacing innerHTML with textContent for filename rendering in the createHeader() method of PlainTextDiff.ts. Alternatively, proper HTML sanitization (escaping <, >, &, ", ') can be applied before inserting user-controlled filenames into the DOM.
AI Insight
LLM-synthesized narrative grounded in this CVE's description and references.
Affected products
1Patches
Vulnerability mechanics
Root cause
"The createHeader() method in PlainTextDiff.ts passes Git filenames directly to innerHTML without sanitization, enabling stored XSS."
Attack vector
An adversary with commit access to a shared Git repository creates a file whose filename contains a JavaScript payload (e.g., `<img src=x onerror=eval(atob("..."))>.py`), then renames the file in a subsequent commit and pushes both commits [ref_id=1]. When a victim clones or pulls the repository, opens the Git History tab, clicks the rename commit, and clicks the renamed file to view the diff, the unsanitized filename is rendered via `innerHTML`, executing the injected JavaScript in the victim's browser session [ref_id=2]. The payload reads the `_xsrf` cookie, opens a JupyterLab terminal via `POST /api/terminals`, connects over WebSocket, and executes arbitrary shell commands, achieving full remote code execution [ref_id=1].
Affected code
The vulnerability resides in the `PlainTextDiff.ts` component of the jupyterlab-git extension, specifically in the `createHeader()` method at line 214. This method constructs a diff header by concatenating Git filenames directly into `innerHTML` without any sanitization when rendering diffs for renamed files in commit history.
What the fix does
The advisory recommends replacing `innerHTML` with `textContent` for filename rendering in the `createHeader()` method of `PlainTextDiff.ts`, or applying proper HTML sanitization (escaping `<`, `>`, `&`, `"`, `'`) before inserting user-controlled filenames into the DOM [ref_id=1]. No patch diff is included in the bundle, so the exact code change is not shown, but the guidance is to avoid interpreting filenames as HTML markup.
Preconditions
- authThe adversary must have commit access to a Git repository that the victim has cloned
- inputThe victim must navigate to the Git History tab, click the rename commit, and click the renamed file to view the diff
- configThe jupyterlab-git extension must be installed in the victim's JupyterLab environment
Generated on Jun 19, 2026. Inputs: CWE entries + fix-commit diffs from this CVE's patches. Citations validated against bundle.
References
2News mentions
0No linked articles in our index yet.