Back to Blog
critical SEVERITY10 min read

CVE-2025-7783: Critical form-data Unsafe Randomness Vulnerability Fixed

A critical vulnerability (CVE-2025-7783) was discovered in the widely-used `form-data` npm package, where an unsafe random function was used to generate multipart boundary strings, making them predictable and potentially exploitable by attackers. The fix upgrades `form-data` to patched versions (2.5.4, 3.0.4, and 4.0.4) across all supported major releases. Developers using any version of `form-data` prior to these patches should upgrade immediately to protect their applications from boundary pre

O
By orbisai0security
โ€ขApril 15, 2026

CVE-2025-7783: Critical form-data Unsafe Randomness Vulnerability Fixed

Severity: ๐Ÿ”ด CRITICAL | Package: form-data | Fixed In: 2.5.4, 3.0.4, 4.0.4


Introduction

If your Node.js application submits forms, uploads files, or communicates with APIs using multipart data, there is a very good chance you depend on the form-data package โ€” either directly or as a transitive dependency. With tens of millions of weekly downloads on npm, form-data is one of the most quietly essential libraries in the JavaScript ecosystem.

On the surface, it does something deceptively simple: it constructs multipart/form-data payloads, the same format your browser uses when you upload a file through an HTML form. But buried in that simplicity was a critical security flaw โ€” the library was using an unsafe, non-cryptographic random function to generate the boundary strings that separate multipart fields.

This vulnerability, tracked as CVE-2025-7783, has been rated CRITICAL in severity. In this post, we'll break down exactly what went wrong, how it could be exploited, and what you need to do to protect your applications.


Background: What Is form-data?

The form-data npm package is a Node.js implementation of the multipart/form-data specification defined in RFC 7578. It is used to programmatically construct form submissions โ€” particularly for file uploads โ€” in environments where a browser is not present.

You'll find it as a direct or indirect dependency in packages like:
- axios
- node-fetch
- superagent
- request (deprecated but still widely used)
- Countless internal API clients and SDKs

A typical usage looks like this:

const FormData = require('form-data');
const fs = require('fs');

const form = new FormData();
form.append('username', 'alice');
form.append('avatar', fs.createReadStream('/path/to/avatar.png'));

// The resulting HTTP body looks something like:
// --abc123randomBoundary456xyz
// Content-Disposition: form-data; name="username"
//
// alice
// --abc123randomBoundary456xyz
// Content-Disposition: form-data; name="avatar"; filename="avatar.png"
// Content-Type: image/png
//
// <binary data>
// --abc123randomBoundary456xyz--

That abc123randomBoundary456xyz string is the boundary โ€” a delimiter that tells the server where one field ends and the next begins. The security of multipart parsing depends heavily on this boundary being unpredictable.


The Vulnerability Explained

What Went Wrong: Predictable Boundaries

The core of CVE-2025-7783 is straightforward but serious: form-data was using Math.random() to generate multipart boundary strings instead of a cryptographically secure random number generator (CSPRNG).

Here's a simplified illustration of what the vulnerable code likely looked like:

// โŒ VULNERABLE: Using Math.random() โ€” NOT cryptographically secure
function generateBoundary() {
  let boundary = '--------------------------';
  for (let i = 0; i < 24; i++) {
    boundary += Math.floor(Math.random() * 10).toString(10);
  }
  return boundary;
}

And what a secure implementation should look like:

// โœ… SECURE: Using crypto.randomBytes() โ€” cryptographically secure
const crypto = require('crypto');

function generateBoundary() {
  return `----FormBoundary${crypto.randomBytes(16).toString('hex')}`;
}

Why Does Math.random() Matter Here?

Math.random() is designed for statistical randomness โ€” things like shuffling a deck of cards in a game or picking a random color for a UI element. It is explicitly not designed for security purposes.

Here's why it falls short:

Property Math.random() crypto.randomBytes()
Cryptographically secure โŒ No โœ… Yes
Seed is unpredictable โŒ No โœ… Yes
Output is predictable given state โœ… Yes (bad!) โŒ No
Suitable for security tokens โŒ No โœ… Yes

The V8 JavaScript engine (used by Node.js) implements Math.random() using an algorithm called xorshift128+. While it produces numbers that look random, a determined attacker who can observe a few outputs from Math.random() can reconstruct the internal state of the PRNG and predict all future outputs.

This is not theoretical โ€” tools like Practical Predictability of Math.random() demonstrate exactly this attack.

Real-World Attack Scenarios

Scenario 1: Boundary Injection / Pollution

If an attacker can predict the boundary string before a request is sent, they may be able to craft malicious input data that contains the boundary string. When the server parses the multipart body, it would interpret the injected boundary as a real delimiter, potentially:

  • Splitting a single field into multiple fields
  • Injecting additional form fields the application never intended to send
  • Overwriting legitimate field values (e.g., injecting admin=true after a role=user field)
# Attacker knows the boundary will be: ----FormBoundary1234567890

# Malicious input in a "comment" field:
Hello World
----FormBoundary1234567890
Content-Disposition: form-data; name="role"

admin
----FormBoundary1234567890--

# Server parses this as TWO fields: "comment" AND "role=admin"

Scenario 2: Security Token Exposure via Timing Attacks

In applications where the boundary is logged, transmitted over a side channel, or otherwise observable, an attacker could use the predictability of Math.random() to correlate requests, de-anonymize users, or link together supposedly independent sessions.

Scenario 3: CSRF Token Bypass in Multipart Forms

Some CSRF protection schemes rely on the unpredictability of multipart boundaries as an additional layer of security (the "secret" is that only the legitimate origin can construct a valid multipart body). A predictable boundary undermines this assumption entirely.

Scenario 4: File Upload Manipulation

In systems that process file uploads, a predictable boundary could allow an attacker to manipulate the parsing of a file's binary content โ€” particularly if the file itself is attacker-controlled โ€” causing the server to misinterpret where the file data ends and metadata begins.


The Fix

What Changed

The maintainers of form-data released patched versions across all actively supported major version lines:

Major Version Vulnerable Patched
2.x < 2.5.4 2.5.4
3.x < 3.0.4 3.0.4
4.x < 4.0.4 4.0.4

The fix replaces the use of Math.random() with Node.js's built-in crypto module, specifically crypto.randomBytes(), which draws entropy from the operating system's CSPRNG (/dev/urandom on Linux/macOS, BCryptGenRandom on Windows).

Before vs. After

Before (Vulnerable):

// Using Math.random() โ€” output is predictable
FormData.prototype._generateBoundary = function() {
  var boundary = '--------------------------';
  for (var i = 0; i < 24; i++) {
    boundary += Math.floor(Math.random() * 10).toString(10);
  }
  this._boundary = boundary;
};

After (Secure):

// Using crypto.randomBytes() โ€” output is cryptographically unpredictable
const crypto = require('crypto');

FormData.prototype._generateBoundary = function() {
  this._boundary = `----FormBoundary${crypto.randomBytes(16).toString('hex')}`;
};

With crypto.randomBytes(16), the boundary contains 128 bits of cryptographically secure entropy. An attacker would need to make 2ยนยฒโธ guesses on average to predict it โ€” effectively impossible with current technology.

How This Was Applied in the PR

The fix was applied by updating webui/package.json and the corresponding webui/yarn.lock to pin form-data to the patched versions. Because form-data is often a transitive dependency (pulled in by other packages), the yarn.lock update ensures that all resolution paths โ€” regardless of which parent package requested form-data โ€” resolve to the secure version.

// webui/package.json (conceptual โ€” ensuring patched versions)
{
  "resolutions": {
    "form-data": "^4.0.4"
  }
}

Using resolutions (in Yarn) or overrides (in npm 8+) is a powerful technique to force all transitive dependencies to use a specific patched version of a package, even when the direct dependency hasn't updated their own package.json yet.


Prevention & Best Practices

1. Never Use Math.random() for Security Purposes

This is the most important takeaway. Math.random() has exactly one appropriate use case in security contexts: none.

Use Case Correct Tool
Random tokens, IDs, boundaries crypto.randomBytes()
UUIDs crypto.randomUUID() (Node 14.17+)
Secure passwords crypto.randomBytes() + encoding
Random numbers in range (secure) crypto.randomInt() (Node 14.10+)
Shuffling for games/UI Math.random() โœ…
// โœ… Generating a secure random token in Node.js
const crypto = require('crypto');

// 32 bytes = 256 bits of entropy, hex encoded
const secureToken = crypto.randomBytes(32).toString('hex');

// Or as a UUID
const secureUUID = crypto.randomUUID();

// Or a secure random integer in range [0, 100)
const secureInt = crypto.randomInt(100);

2. Audit Your Dependencies Regularly

CVE-2025-7783 affects a package that most developers never directly interact with โ€” it's a transitive dependency. This is why automated dependency scanning is non-negotiable.

Recommended tools:

# npm audit (built-in)
npm audit
npm audit fix

# Snyk
npx snyk test

# Trivy (used to detect this CVE)
trivy fs --scanners vuln .

# GitHub Dependabot
# Enable in .github/dependabot.yml

3. Use Lock Files and Dependency Pinning

Always commit your yarn.lock or package-lock.json. These files ensure that everyone on your team โ€” and your CI/CD pipeline โ€” installs the exact same versions of every dependency, including transitive ones.

When a vulnerability is found in a transitive dependency, use resolution overrides to force the patched version:

// package.json โ€” Yarn resolutions
{
  "resolutions": {
    "form-data": ">=4.0.4"
  }
}
// package.json โ€” npm overrides (npm 8+)
{
  "overrides": {
    "form-data": ">=4.0.4"
  }
}

4. Implement a Vulnerability Management Process

Don't wait for vulnerabilities to be reported to you โ€” build the process into your development lifecycle:

  1. Shift Left: Run npm audit or equivalent in your CI pipeline on every pull request
  2. Monitor: Subscribe to security advisories for your key dependencies (GitHub Advisory Database, Snyk, OSS Index)
  3. Respond: Define an SLA for patching based on severity (e.g., Critical: 24-48 hours, High: 1 week)
  4. Automate: Use tools like Dependabot, Renovate, or OrbisAI Security to automatically open PRs for security fixes

5. Understand the OWASP & CWE Context

This vulnerability maps to well-known security weakness classifications:

Understanding these classifications helps you recognize the same class of vulnerability in other contexts โ€” whether it's in session token generation, password reset links, API keys, or, as in this case, multipart boundaries.

6. Code Review Checklist for Randomness

When reviewing code that generates any kind of security-sensitive random value, ask:

  • [ ] Is this using Math.random(), Date.now(), or process PID as a source of randomness? ๐Ÿšจ
  • [ ] Is this value used in a security context (tokens, IDs, boundaries, nonces)?
  • [ ] Is the entropy sufficient (at least 128 bits for most security applications)?
  • [ ] Is the output space large enough to resist brute-force guessing?

Conclusion

CVE-2025-7783 is a textbook example of a subtle but critical security mistake: using the wrong tool for the job. Math.random() is a perfectly fine function for non-security use cases, but its use in generating multipart boundaries โ€” values that need to be unpredictable to maintain the integrity of HTTP requests โ€” introduced a critical vulnerability into one of npm's most downloaded packages.

The fix is straightforward (swap Math.random() for crypto.randomBytes()), but the lesson is broad: cryptographic security requires cryptographic tools. Non-cryptographic PRNGs have no place in security-sensitive code paths, and it's the responsibility of both library authors and application developers to know the difference.

Key Takeaways:

  1. โœ… Upgrade immediately: Update form-data to 2.5.4, 3.0.4, or 4.0.4 depending on your version
  2. โœ… Audit transitive dependencies: You are responsible for the security of every package in your node_modules, not just the ones you directly import
  3. โœ… Use CSPRNGs for security: crypto.randomBytes() and crypto.randomUUID() are your friends
  4. โœ… Automate vulnerability detection: Make dependency scanning a standard part of your CI/CD pipeline
  5. โœ… Learn the patterns: CWE-338 and similar weaknesses appear across all languages and frameworks โ€” learn to recognize them

Security is a team sport. Keep your dependencies updated, keep scanning, and keep learning.


This fix was automatically detected and remediated by OrbisAI Security. Automated security tooling identified the vulnerable form-data version via Trivy scanning and opened a pull request with the appropriate patches โ€” demonstrating how automation can dramatically reduce the window of exposure for known CVEs.


References:
- CVE-2025-7783 โ€“ NVD
- form-data on npm
- CWE-338: Use of Cryptographically Weak PRNG
- Node.js Crypto Documentation
- OWASP Top 10: A02 Cryptographic Failures
- RFC 7578: Returning Values from Forms: multipart/form-data

View the Security Fix

Check out the pull request that fixed this vulnerability

View PR #12958

Related Articles

medium

Mass Assignment Vulnerability: Why Your Rails Models Need attr_accessible

A medium-severity mass assignment vulnerability was identified in a Ruby on Rails model that lacked proper attribute whitelisting via `attr_accessible` or strong parameters. Without this protection, attackers can manipulate any model attribute through crafted HTTP requests, potentially escalating privileges or corrupting data. The fix enforces explicit attribute allowlisting, closing the door on unauthorized mass assignment exploitation.

critical

Shell Injection via os.system(): How a Single Line of Code Can Compromise Your System

A critical OS command injection vulnerability (CWE-78) was discovered and patched in `voice.py`, where user-controlled input was interpolated directly into a shell command string passed to `os.system()`. An attacker who could influence the `device` variable โ€” through a config file, environment variable, or any external input โ€” could execute arbitrary system commands with the full privileges of the running process. The fix replaces the dangerous `os.system()` calls with Python's `subprocess.run()

critical

Command Injection via os.system() in DeepSpeed's Data Analyzer: A Critical Fix

A critical command injection vulnerability was discovered in DeepSpeed's `data_analyzer.py`, where an `os.system()` call directly interpolated an unsanitized file path variable into a shell command string. An attacker who could influence dataset configuration or file paths could execute arbitrary shell commands on the host machine. The fix replaces the dangerous shell invocation with safe, Python-native file operations that never touch a shell interpreter.

high

CVE-2026-40073: How a BODY_SIZE_LIMIT Bypass in @sveltejs/adapter-node Put Your App at Risk

CVE-2026-40073 is a high-severity vulnerability in `@sveltejs/adapter-node` that allows attackers to bypass the `BODY_SIZE_LIMIT` configuration, potentially enabling denial-of-service attacks and resource exhaustion against SvelteKit applications. The vulnerability was silently present in versions prior to `@sveltejs/kit` 2.57.1, and has now been patched by upgrading the dependency across all affected project examples. If your application relies on body size limits to protect against oversized p

medium

From eval() to ast.literal_eval(): Closing a Code Injection Door in Slack Data Processing

A medium-severity vulnerability was discovered in a Slack data processing component where the use of Python's built-in `eval()` function to parse error message dictionaries could allow an attacker to inject and execute arbitrary code. The fix replaces `eval()` with the safer `ast.literal_eval()`, which safely evaluates only Python literals without executing arbitrary expressions. This change eliminates a critical attack surface that could have been exploited through crafted error messages return

critical

Critical Buffer Overflow in ELF Parser: How a Missing Bounds Check Almost Became a Heap Exploit

A critical out-of-bounds memory vulnerability was discovered and patched in `utils/symbol-rawelf.c`, where two separate `memcpy` calls lacked proper bounds validation when processing ELF binary files. Without these checks, a maliciously crafted ELF file could trigger an out-of-bounds read or heap overflow, potentially leading to remote code execution or memory corruption. This post breaks down how the vulnerability works, how it was fixed, and what every C developer should know about safe memory