Skip to main content
On this page

ECMAScript Modules in Deno

Deno by default standardizes the way modules are imported in both JavaScript and TypeScript using the ECMAScript module standard. In line with this standard, file names must be specified in full. You should not omit the file extension and there is no special handling of index.js.

// Always include the file extension
import { add } from "./calc.ts";

Importing modules Jump to heading

In this example the add function is imported from a local calc.ts module.

calc.ts
export function add(a: number, b: number): number {
  return a + b;
}
main.ts
import { add } from "./calc.ts";

console.log(add(1, 2)); // 3

You can run this example by calling deno run main.ts in the directory that contains main.ts and calc.ts.

Adding third party modules and libraries to deno.json Jump to heading

Using remote modules (downloaded from a registry) in Deno uses the same import syntax as your local code.

The modules that you add to your project are tracked as imports in deno.json - we call this the import map.

deno.json
{
  "imports": {
    "@scopename/mypackage": "jsr:@scopename/mypackage@^16.1.0"
  }
}

You can add modules using the deno add subcommand:

# Add the latest version of the module to deno.json
$ deno add @luca/cases
Add @luca/cases - jsr:@luca/cases@^1.0.0
deno.json
{
  "imports": {
    "@luca/cases": "jsr:@luca/cases@^1.0.0"
  }
}

The deno add command will automatically add the latest version of the module you requested to your project imports, unless you specify an exact version:

# Passing an exact version
$ deno add @luca/cases@1.0.0
Add @luca/cases - jsr:@luca/cases@^1.0.0

Using installed modules Jump to heading

Once a package is added to the import map in deno.json, you can import the module by its name, and Deno will automatically resolve the module. For example:

main.ts
import { camelCase } from "@luca/cases";

camelCase("hello world"); // "helloWorld"

Package Registries Jump to heading

Deno recommends that you use JSR, the JavaScript registry, to publish and manage your modules for the best publishing experience. JSR offers automatic documentation generation, semver resolution and improved performance for TypeScript code. Deno also supports other platforms for publishing modules, such as npm, and JavaScript CDNs like deno.land/x, esm.h and unpkg.com

By default when you use the CLI, the package will be resolved from JSR.

deno add @scopename/mypackage

The resulting import map contains the jsr: import specifier in the resulting deno.json file.

{
  "imports": {
    "@scopename/mypackage": "jsr:@scopename/mypackage@^16.1.0"
  }
}

Importing from other package registries Jump to heading

Using existing packages from registries like npm is equivalent to the experience in node. To add a package from npm you need to prefix the package name with the npm: specifier:

$ deno add npm:uuid
Add uuid - npm:uuid@^10.0.0

The resulting deno.json will look like this:

{
  "imports": {
    "uuid": "npm:uuid@^10.0.0"
  }
}

You will then be able to use the npm module by importing it with the name from the import map.

mod.ts
import * as uuid from "uuid";

console.log(uuid.v4());

Supported package registries and specifiers Jump to heading

We support the following registries:

Registry Specifier
JSR jsr:
npm npm:

Understanding Package Versions Jump to heading

When using Deno add, you can specify a version range for the package you are adding. This is done using the @ symbol followed by a version range specifier, and follows the semver versioning scheme.

deno add @scopename/mypackage           # latest version
deno add @scopename/mypackage@16.1.0    # exact version
deno add @scopename/mypackage@^16.1.0   # latest patch version 16.x
deno add @scopename/mypackage@~16.1.0   # latest version that doesn't increment the first non-zero portion

Importing by URL Jump to heading

Deno also supports import statements that reference URLs, either directly:

import { Application } from "https://deno.land/x/oak/mod.ts";

or part of your deno.json import map:

{
  "imports": {
    "oak": "https://deno.land/x/oak/mod.ts"
  }
}

Supporting URL imports enables us to support the following JavaScript CDNs, as they provide URL access to JavaScript modules:

URL imports are useful if you have a small, often single file, Deno project that doesn't require any other configuration because you can avoid having a Deno.json file at all. It's not advised to use this style of import in larger applications as you may end up with version conflicts (where different files use different version specifiers).

From a security perspective, the contents of files at URLs can change, so we do not generally recommend this approach for third-party components.

URL imports remain supported, but we recommend using a package registry for the best experience.

Overriding URL imports Jump to heading

The other situation where import maps can be very useful is to override URL imports in specific modules.

Let's say you want to override a https://deno.land/x/my-library@1.0.0/mod.ts specifier that is used inside files coming from https://deno.land/x/example/ to a local patched version. You can do this by using a scope in the import map that looks something like this:

{
  "imports": {
    "example/": "https://deno.land/x/example/"
  },
  "scopes": {
    "https://deno.land/x/example/": {
      "https://deno.land/x/my-library@1.0.0/mod.ts": "./patched/mod.ts"
    }
  }
}

It is important to note that URL imports have no notion of packages. Only the import map at the root of your project is used. Import maps used inside URL dependencies are ignored.

Proxies Jump to heading

Deno supports proxies for module downloads and the Web standard fetch API.

Proxy configuration is read from environmental variables: HTTP_PROXY, HTTPS_PROXY and NO_PROXY.

In case of Windows, if environment variables are not found Deno falls back to reading proxies from registry.