Jun 9, 2026·4 min read·7 visits
Unauthenticated remote attackers can hijack active phoenix_storybook sessions by supplying a victim's PubSub topic via URL parameters, leading to potential cross-session state exposure.
A security vulnerability in the Elixir package phoenix_storybook (versions 0.4.0 up to 1.1.0) allows unauthenticated remote attackers to perform cross-session PubSub topic injection. By manipulating URL parameters, an attacker can hijack the real-time communications channel, enabling them to capture user state and control parameters from active sessions.
The Elixir package phoenix_storybook is an interactive catalog designed to showcase and test Phoenix components. To maintain isolated styling scopes and avoid JavaScript pollution, the application loads component previews within nested <iframe> elements.
To synchronize the UI states (such as configuration variations, color modes, and themes) between the parent playground LiveView and the nested iframe LiveView, the application uses an Elixir-native Phoenix.PubSub server (PhoenixStorybook.PubSub).
The component iframe exposes an attack surface by reading communication parameters directly from the client. Unauthenticated users can exploit this behavior because the application fails to validate the ownership or integrity of these parameters, leading to an authorization bypass.
The root cause is a CWE-639 (Authorization Bypass Through User-Controlled Key) flaw. When the parent LiveView (StoryLive) mounts, it establishes a session-specific PubSub topic based on its process identifier (PID):
playground_topic = "playground-#{inspect(self())}".
The parent then references this topic in the query parameters when rendering the iframe:
/storybook/iframe/<story_name>?topic=playground-#PID<0.1234.0>.
Inside the iframe, the ComponentIframeLive LiveView extracts this parameter during the handle_params/3 lifecycle stage and immediately broadcasts its own process identifier (PID) to that topic without verifying session ownership. Any client can supply an arbitrary topic string, forcing the iframe process to broadcast itself to that channel.
The vulnerability was fixed in commit 6ee03f1c738d4436dde1b066cf65c80663d489f5 by replacing the plain text topic parameter with a cryptographically signed token generated via Phoenix.Token.
Prior to the patch, the application parsed the raw topic from the URL params and executed PubSub.broadcast!/3:
# VULNERABLE APPROACH
if params["topic"] do
PubSub.broadcast!(
PhoenixStorybook.PubSub,
params["topic"],
{:component_iframe_pid, self()}
)
endThe updated implementation extracts a playground_token parameter and verifies it against the application's secure endpoint:
# PATCHED APPROACH
topic = verified_playground_topic(socket, params["playground_token"])
if topic do
PubSub.broadcast!(
PhoenixStorybook.PubSub,
topic,
{:component_iframe_pid, self()}
)
endThis cryptographic verification ensures that only tokens signed by the parent application (using the server's secret_key_base) can specify a target communication channel.
Exploitation relies on predicting or intercepting the target victim's parent LiveView PID, which acts as the PubSub topic. In Elixir/BEAM systems, process identifiers are allocated sequentially, making them predictable in environments with low process churn.
An attacker crafts a URL targeting the storybook's iframe endpoint, appending the victim's estimated topic string:
https://example.com/storybook/iframe/component?topic=playground-%23PID%3C0.1432.0%3E.
When the attacker loads this URL, their iframe LiveView broadcasts its process identifier (PID) to the victim's topic. The victim's parent LiveView accepts this spoofed PID. As a result, when the victim interacts with the UI, their private states, mock payloads, or theme selections are sent directly to the attacker's iframe process, triggering a cross-session information exposure.
The impact is classified as Low (CVSS 2.3) due to several restrictive prerequisites. The attack does not lead to direct Remote Code Execution (RCE) or complete system compromise. Instead, the consequences are restricted to cross-session information disclosure and unauthorized UI state manipulation.
An attacker can capture private mock variables, custom input attributes, and user-configured states intended only for the victim's display. No persistent changes are made to the database, and there is no direct impact on service availability.
To address this vulnerability, administrators should upgrade phoenix_storybook to version 1.1.0 or higher. This version deprecates the plain topic parameter in favor of signed tokens.
If upgrading immediately is not feasible, implement a manual code patch. Define a cryptographic salt and apply Phoenix.Token.verify/4 on incoming request parameters in lib/phoenix_storybook/live/story/component_iframe_live.ex.
Ensure that your application's secret_key_base is securely configured and rotated regularly, as the patch's cryptographic integrity depends entirely on the secrecy of this key.
CVSS:4.0/AV:N/AC:L/AT:P/PR:N/UI:P/VC:L/VI:L/VA:N/SC:N/SI:N/SA:N| Product | Affected Versions | Fixed Version |
|---|---|---|
phoenix_storybook phenixdigital | >= 0.4.0, < 1.1.0 | 1.1.0 |
| Attribute | Detail |
|---|---|
| CWE ID | CWE-639 |
| Attack Vector | Network |
| CVSS Score | 2.3 (Low) |
| EPSS Score | 0.00054 |
| Exploit Status | No active public exploits |
| KEV Status | Not listed |
The system uses actor-controlled keys to access resources, allowing an attacker to bypass authorization by providing a key belonging to another user.
An unauthenticated remote code execution (RCE) vulnerability exists in phoenix_storybook versions 0.5.0 through 1.0.x due to improper input sanitization during HEEx template generation. By sending crafted WebSocket messages, an attacker can escape HTML attribute boundaries and execute arbitrary Elixir code.
An unauthenticated Denial-of-Service (DoS) vulnerability exists in phoenix_storybook versions 0.2.0 through 1.0.11 due to allocation of resources without limits (CWE-770). The application dynamically converts user-supplied parameter keys to atoms, leading to BEAM Atom Table exhaustion and immediate virtual machine crash.
An improper authorization vulnerability in the unreleased development master branch of Dex allows clients to bypass the AllowedConnectors access control list using the token-exchange endpoint.
CVE-2024-29203 identifies a cross-site scripting (XSS) vulnerability in the content ingestion and parsing mechanics of TinyMCE rich text editor. Due to a failure to enforce sandbox attributes on dynamic iframe elements and safely handle legacy embed objects, unauthenticated attackers can inject malicious elements that execute scripts within the context of the parent application session.
A technical breakdown of the OS command injection vulnerability in the shell-quote NPM package (CVE-2026-9277 / GHSA-w7jw-789q-3m8p). The bug resides in the character-by-character backslash-escaping logic applied to the .op field of object-tokens within the quote() function, which fails to match and escape line terminators due to a regex matching oversight in JavaScript. This allows unauthenticated remote attackers to execute arbitrary shell commands if they can control inputs processed by this library.
A high-severity memory corruption vulnerability exists in the V8 JavaScript engine of Google Chrome before versions 149.0.7827.102/103. The flaw arises from an incorrect bounds-check elimination during JIT compilation by the TurboFan optimizer, allowing remote attackers to achieve out-of-bounds read and write access inside the sandboxed renderer process.