~ 2 min read

Vue CLI Security Fix to Mitigate NPM Binary Planting

share on
NPM binary planting is a way to cause dependency confusion within installed executable packages with npx. Haoqun Jiang from the Vue.js and Vite core teams have patched the Vue.js CLI to mitigate this security risk.

In a prior edition of the Node.js Security newsletter from February I’ve dropped the story on npm binary planting and how this can be abused to trigger a sort of dependency confusion type of attack for executable package, such as those that are running via npx which JavaScript developers are so commonly.

About npx and binary planting

A normal workflow usage of npx is like this:

Terminal window
$ npx nodemon
Need to install the following packages:
nodemon@2.0.22
Ok to proceed? (y)

By doing so, npx expects to either find the one executable pointed to by nodemon and access it by that index, or, and this is where things get weird, it may allow said package to expose several executables that are not necessarily the same as the package name. This is where the binary planting comes into play.

Let’s take a realistic example from the Vue.js ecosystem, per Alex Birsan’s experiment:

  • There’s a bin script named vue-cli-service that is part of the @vue/cli-service package.
  • Weekly download counts for @vue/cli-service is around 600,000.

But now, you can register the package name vue-cli-service if it is freely available on the npm registry for the taking, and by doing so, if you run npx vue-cli-service, it will run your code instead of the one from the @vue/cli-service package, which so many developers would expect to be the one that is actually running.

This is a form of dependency confusion, and it can be exploited to run arbitrary code on the developer’s machine.

Vue CLI Security Fix

Haoqun Jiang, the core team member of the Vue.js and Vite core teams, has updated the documentation for how to use the Vue CLI to mitigate this security risk.

The fix is to use the --no flag when running npx commands. This flag prevents npx from installing any missing packages, which means that it won’t be able to run any potentially malicious code from a package that has been registered with the same name as an existing package.

Here’s an example:

Terminal window
$ npx --no vue-cli-service

Haoqun Jiang appreciates the Node.js Security Newsletter for revealing the news


Node.js Security Newsletter

Subscribe to get everything in and around the Node.js security ecosystem, direct to your inbox.

    JavaScript & web security insights, latest security vulnerabilities, hands-on secure code insights, npm ecosystem incidents, Node.js runtime feature updates, Bun and Deno runtime updates, secure coding best practices, malware, malicious packages, and more.