use-m: dynamically import any JavaScript module anywhere
use-m
(m
stands for module
) is a utility for dynamically importing any JavaScript module (npm package) at runtime anywhere (browser or server).
It may be useful for standalone scripts that do not require a package.json
. Also it may make your code portable across environments (for example it may require no changes in your code when executed in CommonJS, ES Modules and browser). You can ensure predictable behavior of your code over time by specifying the exact version to import directly in your script, similar to how versions are specified in package.json. You even can import multiple versions of the same library at the same time. You can use use-m
when you don't want your package.json
to be poluted with optional packages. You may keep your package.json
with as little dependencies as needed thanks to use-m
.
- Dynamic package loading: In
node.js
,use-m
loads and imports npm packages on-demand with global installation (usingnpm i -g
with separate alias for each version), making them available across projects and reusable without needing to reinstall each time. In case of a browseruse-m
loads npm packages directly from CDNs (by defaultesm.sh
is used). - Version-safe imports: Allows multiple versions of the same library to coexist without conflicts, so you can specify any version for each import (usage) without affecting other scripts or other usages (imports) in the same script.
Works in CommonJS, ES Modules and browser environments.
fetch('https://unpkg.com/use-m/use.js')
.then(async useJs => {
const { use } = eval(await useJs.text());
const _ = await use('lodash@4.17.21');
console.log(`_.add(1, 2) = ${_.add(1, 2)}`);
});
Universal execution comes at cost of eval
usage, that is considered potential security threat. In case of this library only single file is evaled, it short, unminified and has no dependencies, so you can check the contents yourself. Once you have use
function instance no more eval
function will be executed by this library. If you don't want to use eval
you can use await import()
in browser or in node.js
. In node.js
you can also just install the package from npm
as usual.
If you don't want to use eval
in the browser, you can import use-m
like this:
const { use } = await import("https://esm.sh/use-m");
const _ = await use('lodash@4.17.21');
console.log(`_.add(1, 2) = ${_.add(1, 2)}`);
Only 2 lines and now have an interactive playground for JavaScript and almost any NPM library directly in your browser's console. No more cloud based sandboxes required. Sorry VSCode, you don't have such super powers yet.
It is possible to use --experimental-network-imports
to enable the same style of imports as in browser version. See the example.
-
Create file named
example.mjs
:const { use } = await import('https://unpkg.com/use-m/use.mjs'); const _ = await use('lodash@4.17.21'); console.log(`_.add(1, 2) = ${_.add(1, 2)}`);
-
Execute the script using
--experimental-network-imports
option:node --experimental-network-imports example.mjs
If you need to use use-m
without adding it to a project locally, you can load it directly from unpkg
using fetch
. This is particularly useful in environments like zx or in other standalone scripts like execa
, when you don't want to use any package.json
, node_modules
, etc.
-
Install zx globally
npm install -g zx
-
Create a file named
example.mjs
:#!/usr/bin/env zx --verbose const { use } = eval( await fetch('https://unpkg.com/use-m/use.js').then(u => u.text()) ); const _ = await use('lodash@latest'); const { stdout } = await $`ls`.pipe`grep js`; const files = _.filter( _.split(stdout, '\n'), (item) => !_.isEmpty(item) ); console.log(files);
-
Give execution permissions
chmod +x example.mjs
-
Execute:
./example.mjs
-
Create a file named
example.mjs
:const { use } = eval( await fetch('https://unpkg.com/use-m/use.js').then(u => u.text()) ); const _ = await use('lodash'); const { $: $$ } = await use('execa'); const $ = $$({ verbose: 'full' }); const { stdout } = await $`ls`.pipe`grep js`; const files = _.filter( _.split(stdout, '\n'), (item) => !_.isEmpty(item) ); console.log(files);
Note: in ES Module environments where __filename
and require
are not defined, you may need to add meta
option into use
function constructor, as it is not possible to access import.meta
inside eval
.
-
Execute:
node example.mjs
You can still install and import use-m
in node.js
as usual. For example if you don't want to use eval
in node.js
.
Add use-m
to your project with Yarn:
yarn add use-m
Or NPM:
npm i use-m
Load use-m
to dynamically import the lodash
package from npm:
const { use } = require('use-m');
(async () => {
const _ = await use('lodash@4.17.21');
console.log(`_.add(1, 2) = ${_.add(1, 2)}`);
})();
or
import('use-m')
.then(async ({ use }) => {
const _ = await use('lodash@4.17.21');
console.log(`_.add(1, 2) = ${_.add(1, 2)}`);
});
import { use } from 'use-m';
const _ = await use('lodash@4.17.21');
console.log(`_.add(1, 2) = ${_.add(1, 2)}`);
or
const { use } = await import('use-m');
const _ = await use('lodash@4.17.21');
console.log(`_.add(1, 2) = ${_.add(1, 2)}`);
You can check out usage examples source code. You can also explore our tests to get even more examples.
If you have any questions or issues, please write us on GitHub. Together we can ensure this package will have highest quality possible. Your feedback is valuable and helps improve the project.
We welcome contributions! To contribute please open Pull Request with any suggested changes.
This project is licensed under the Unlicense (public domain). That means you are absolutely free to use this library, there is no conditions or limitations on how this library and its code can be used.