---
title: Migrate from Service Workers to ES Modules
description: Write your Worker code in ES modules syntax for an optimized experience.
image: https://developers.cloudflare.com/dev-products-preview.png
---

> Documentation Index  
> Fetch the complete documentation index at: https://developers.cloudflare.com/workers/llms.txt  
> Use this file to discover all available pages before exploring further.

[Skip to content](#%5Ftop) 

# Migrate from Service Workers to ES Modules

This guide will show you how to migrate your Workers from the [Service Worker ↗](https://developer.mozilla.org/en-US/docs/Web/API/Service%5FWorker%5FAPI) format to the [ES modules ↗](https://blog.cloudflare.com/workers-javascript-modules/) format.

## Advantages of migrating

There are several reasons to migrate your Workers to the ES modules format:

1. Your Worker will run faster. With service workers, bindings are exposed as globals. This means that for every request, the Workers runtime must create a new JavaScript execution context, which adds overhead and time. Workers written using ES modules can reuse the same execution context across multiple requests.
2. Implementing [Durable Objects](https://developers.cloudflare.com/durable-objects/) requires Workers that use ES modules.
3. Bindings for [D1](https://developers.cloudflare.com/d1/), [Workers AI](https://developers.cloudflare.com/workers-ai/), [Vectorize](https://developers.cloudflare.com/vectorize/), [Workflows](https://developers.cloudflare.com/workflows/), and [Images](https://developers.cloudflare.com/images/optimization/transformations/bindings/) can only be used from Workers that use ES modules.
4. You can [gradually deploy changes to your Worker](https://developers.cloudflare.com/workers/configuration/versions-and-deployments/gradual-deployments/) when you use the ES modules format.
5. You can easily publish Workers using ES modules to `npm`, allowing you to import and reuse Workers within your codebase.

## Migrate a Worker

The following example demonstrates a Worker that redirects all incoming requests to a URL with a `301` status code.

Service Workers are deprecated

Service Workers are deprecated, but still supported. We recommend using [Module Workers](https://developers.cloudflare.com/workers/reference/migrate-to-module-workers/) instead. New features may not be supported for Service Workers.

With the Service Worker syntax, the example Worker looks like:

JavaScript

```

async function handler(request) {

  const base = 'https://example.com';

  const statusCode = 301;


  const destination = new URL(request.url, base);

  return Response.redirect(destination.toString(), statusCode);

}


// Initialize Worker

addEventListener('fetch', event => {

  event.respondWith(handler(event.request));

});


```

Workers using ES modules format replace the `addEventListener` syntax with an object definition, which must be the file's default export (via `export default`). The previous example code becomes:

JavaScript

```

export default {

  fetch(request) {

    const base = "https://example.com";

    const statusCode = 301;


    const source = new URL(request.url);

    const destination = new URL(source.pathname, base);

    return Response.redirect(destination.toString(), statusCode);

  },

};


```

## Bindings

[Bindings](https://developers.cloudflare.com/workers/runtime-apis/bindings/) allow your Workers to interact with resources on the Cloudflare developer platform.

Workers using ES modules format do not rely on any global bindings. However, Service Worker syntax accesses bindings on the global scope.

To understand bindings, refer the following `TODO` KV namespace binding example. To create a `TODO` KV namespace binding, you will:

1. Create a KV namespace named `My Tasks` and receive an ID that you will use in your binding.
2. Create a Worker.
3. Find your Worker's [Wrangler configuration file](https://developers.cloudflare.com/workers/wrangler/configuration/) and add a KV namespace binding:

* [  wrangler.jsonc ](#tab-panel-9521)
* [  wrangler.toml ](#tab-panel-9522)

JSONC

```

{

  "kv_namespaces": [

    {

      "binding": "TODO",

      "id": "<ID>"

    }

  ]

}


```

TOML

```

[[kv_namespaces]]

binding = "TODO"

id = "<ID>"


```

In the following sections, you will use your binding in Service Worker and ES modules format.

Reference KV from Durable Objects and Workers

To learn more about how to reference KV from Workers, refer to the [KV bindings documentation](https://developers.cloudflare.com/kv/concepts/kv-bindings/).

### Bindings in Service Worker format

In Service Worker syntax, your `TODO` KV namespace binding is defined in the global scope of your Worker. Your `TODO` KV namespace binding is available to use anywhere in your Worker application's code.

JavaScript

```

addEventListener("fetch", async (event) => {

  return await getTodos()

});


async function getTodos() {

  // Get the value for the "to-do:123" key

  // NOTE: Relies on the TODO KV binding that maps to the "My Tasks" namespace.

  let value = await TODO.get("to-do:123");


  // Return the value, as is, for the Response

  event.respondWith(new Response(value));

}


```

### Bindings in ES modules format

In ES modules format, bindings are only available inside the `env` parameter that is provided at the entry point to your Worker.

To access the `TODO` KV namespace binding in your Worker code, the `env` parameter must be passed from the `fetch` handler in your Worker to the `getTodos` function.

JavaScript

```

import { getTodos } from './todos'


export default {

  async fetch(request, env, ctx) {

    // Passing the env parameter so other functions

    // can reference the bindings available in the Workers application

    return await getTodos(env)

  },

};


```

The following code represents a `getTodos` function that calls the `get` function on the `TODO` KV binding.

JavaScript

```

async function getTodos(env) {

  // NOTE: Relies on the TODO KV binding which has been provided inside of

  // the env parameter of the `getTodos` function

  let value = await env.TODO.get("to-do:123");

  return new Response(value);

}


export { getTodos }


```

## Environment variables

[Environment variables](https://developers.cloudflare.com/workers/configuration/environment-variables/) are accessed differently in code written in ES modules format versus Service Worker format.

Review the following example environment variable configuration in the [Wrangler configuration file](https://developers.cloudflare.com/workers/wrangler/configuration/):

* [  wrangler.jsonc ](#tab-panel-9523)
* [  wrangler.toml ](#tab-panel-9524)

JSONC

```

{

  "$schema": "./node_modules/wrangler/config-schema.json",

  "name": "my-worker-dev",

  // Define top-level environment variables

  // using the {"vars": "key": "value"} format

  "vars": {

    "API_ACCOUNT_ID": "<EXAMPLE-ACCOUNT-ID>"

  }

}


```

TOML

```

"$schema" = "./node_modules/wrangler/config-schema.json"

name = "my-worker-dev"


[vars]

API_ACCOUNT_ID = "<EXAMPLE-ACCOUNT-ID>"


```

### Environment variables in Service Worker format

In Service Worker format, the `API_ACCOUNT_ID` is defined in the global scope of your Worker application. Your `API_ACCOUNT_ID` environment variable is available to use anywhere in your Worker application's code.

JavaScript

```

addEventListener("fetch", async (event) => {

  console.log(API_ACCOUNT_ID) // Logs "<EXAMPLE-ACCOUNT-ID>"

  return new Response("Hello, world!")

})


```

### Environment variables in ES modules format

In ES modules format, environment variables are available through the `env` parameter provided at the entrypoint to your Worker application:

JavaScript

```

export default {

  async fetch(request, env, ctx) {

    console.log(env.API_ACCOUNT_ID) // Logs "<EXAMPLE-ACCOUNT-ID>"

    return new Response("Hello, world!")

  },

};


```

You can also import `env` from `cloudflare:workers` to access environment variables from anywhere in your code, including the top-level scope:

* [  JavaScript ](#tab-panel-9525)
* [  TypeScript ](#tab-panel-9526)

JavaScript

```

import { env } from "cloudflare:workers";


// Access environment variables at the top level

const accountId = env.API_ACCOUNT_ID;


export default {

  async fetch(request) {

    console.log(accountId); // Logs "<EXAMPLE-ACCOUNT-ID>"

    return new Response("Hello, world!");

  },

};


```

TypeScript

```

import { env } from "cloudflare:workers";


// Access environment variables at the top level

const accountId = env.API_ACCOUNT_ID;


export default {

  async fetch(request: Request): Promise<Response> {

    console.log(accountId) // Logs "<EXAMPLE-ACCOUNT-ID>"

    return new Response("Hello, world!")

  },

};


```

This approach is useful for initializing configuration or accessing environment variables from deeply nested functions without passing `env` through every function call. For more details, refer to [Importing env as a global](https://developers.cloudflare.com/workers/runtime-apis/bindings/#importing-env-as-a-global).

## Cron Triggers

To handle a [Cron Trigger](https://developers.cloudflare.com/workers/configuration/cron-triggers/) event in a Worker written with ES modules syntax, implement a [scheduled() event handler](https://developers.cloudflare.com/workers/runtime-apis/handlers/scheduled/#syntax), which is the equivalent of listening for a `scheduled` event in Service Worker syntax.

This example code:

JavaScript

```

addEventListener("scheduled", (event) => {

  // ...

});


```

Then becomes:

JavaScript

```

export default {

  async scheduled(event, env, ctx) {

    // ...

  },

};


```

## Access `event` or `context` data

Workers often need access to data not in the `request` object. For example, sometimes Workers use [waitUntil](https://developers.cloudflare.com/workers/runtime-apis/context/#waituntil) to delay execution. Workers using ES modules format can access `waitUntil` via the `context` parameter. Refer to [ES modules parameters](https://developers.cloudflare.com/workers/runtime-apis/handlers/fetch/#parameters) for more information.

This example code:

JavaScript

```

async function triggerEvent(event) {

  // Fetch some data

  console.log('cron processed', event.scheduledTime);

}


// Initialize Worker

addEventListener('scheduled', event => {

  event.waitUntil(triggerEvent(event));

});


```

Then becomes:

JavaScript

```

async function triggerEvent(event) {

  // Fetch some data

  console.log('cron processed', event.scheduledTime);

}


export default {

  async scheduled(event, env, ctx) {

    ctx.waitUntil(triggerEvent(event));

  },

};


```

## Service Worker syntax

A Worker written in Service Worker syntax consists of two parts:

1. An event listener that listens for `FetchEvents`.
2. An event handler that returns a [Response](https://developers.cloudflare.com/workers/runtime-apis/response/) object which is passed to the event’s `.respondWith()` method.

When a request is received on one of Cloudflare’s global network servers for a URL matching a Worker, Cloudflare's server passes the request to the Workers runtime. This dispatches a `FetchEvent` in the [isolate](https://developers.cloudflare.com/workers/reference/how-workers-works/#isolates) where the Worker is running.

JavaScript

```

addEventListener('fetch', event => {

  event.respondWith(handleRequest(event.request));

});


async function handleRequest(request) {

  return new Response('Hello worker!', {

    headers: { 'content-type': 'text/plain' },

  });

}


```

Below is an example of the request response workflow:

1. An event listener for the `FetchEvent` tells the script to listen for any request coming to your Worker. The event handler is passed the `event` object, which includes `event.request`, a [Request](https://developers.cloudflare.com/workers/runtime-apis/request/) object which is a representation of the HTTP request that triggered the `FetchEvent`.
2. The call to `.respondWith()` lets the Workers runtime intercept the request in order to send back a custom response (in this example, the plain text `'Hello worker!'`).  
   * The `FetchEvent` handler typically culminates in a call to the method `.respondWith()` with either a [Response](https://developers.cloudflare.com/workers/runtime-apis/response/) or `Promise<Response>` that determines the response.  
   * The `FetchEvent` object also provides [two other methods](https://developers.cloudflare.com/workers/runtime-apis/handlers/fetch/) to handle unexpected exceptions and operations that may complete after a response is returned.

Learn more about [the lifecycle methods of the fetch() handler](https://developers.cloudflare.com/workers/runtime-apis/rpc/lifecycle/).

### Supported `FetchEvent` properties

* `event.type` string  
   * The type of event. This will always return `"fetch"`.
* `event.request` Request  
   * The incoming HTTP request.
* `event.respondWith(responseResponse|Promise)` : void  
   * Refer to [respondWith](#respondwith).
* `event.waitUntil(promisePromise)` : void  
   * Refer to [waitUntil](#waituntil).
* `event.passThroughOnException()` : void  
   * Refer to [passThroughOnException](#passthroughonexception).

### `respondWith`

Intercepts the request and allows the Worker to send a custom response.

If a `fetch` event handler does not call `respondWith`, the runtime delivers the event to the next registered `fetch` event handler. In other words, while not recommended, this means it is possible to add multiple `fetch` event handlers within a Worker.

If no `fetch` event handler calls `respondWith`, then the runtime forwards the request to the origin as if the Worker did not. However, if there is no origin – or the Worker itself is your origin server, which is always true for `*.workers.dev` domains – then you must call `respondWith` for a valid response.

JavaScript

```

// Format: Service Worker

addEventListener('fetch', event => {

  let { pathname } = new URL(event.request.url);


  // Allow "/ignore/*" URLs to hit origin

  if (pathname.startsWith('/ignore/')) return;


  // Otherwise, respond with something

  event.respondWith(handler(event));

});


```

### `waitUntil`

The `waitUntil` command extends the lifetime of the `"fetch"` event. It accepts a `Promise`\-based task which the Workers runtime will execute before the handler terminates but without blocking the response. For example, this is ideal for [caching responses](https://developers.cloudflare.com/workers/runtime-apis/cache/#put) or handling logging.

With the Service Worker format, `waitUntil` is available within the `event` because it is a native `FetchEvent` property.

With the ES modules format, `waitUntil` is moved and available on the `context` parameter object.

JavaScript

```

// Format: Service Worker

addEventListener('fetch', event => {

  event.respondWith(handler(event));

});


async function handler(event) {

  // Forward / Proxy original request

  let res = await fetch(event.request);


  // Add custom header(s)

  res = new Response(res.body, res);

  res.headers.set('x-foo', 'bar');


  // Cache the response

  // NOTE: Does NOT block / wait

  event.waitUntil(caches.default.put(event.request, res.clone()));


  // Done

  return res;

}


```

### `passThroughOnException`

The `passThroughOnException` method prevents a runtime error response when the Worker throws an unhandled exception. Instead, the script will [fail open ↗](https://community.microfocus.com/cyberres/b/sws-22/posts/security-fundamentals-part-1-fail-open-vs-fail-closed), which will proxy the request to the origin server as though the Worker was never invoked.

To prevent JavaScript errors from causing entire requests to fail on uncaught exceptions, `passThroughOnException()` causes the Workers runtime to yield control to the origin server.

With the Service Worker format, `passThroughOnException` is added to the `FetchEvent` interface, making it available within the `event`.

With the ES modules format, `passThroughOnException` is available on the `context` parameter object.

JavaScript

```

// Format: Service Worker

addEventListener('fetch', event => {

  // Proxy to origin on unhandled/uncaught exceptions

  event.passThroughOnException();

  throw new Error('Oops');

});


```

```json
{"@context":"https://schema.org","@type":"BreadcrumbList","itemListElement":[{"@type":"ListItem","position":1,"item":{"@id":"/directory/","name":"Directory"}},{"@type":"ListItem","position":2,"item":{"@id":"/workers/","name":"Workers"}},{"@type":"ListItem","position":3,"item":{"@id":"/workers/reference/","name":"Reference"}},{"@type":"ListItem","position":4,"item":{"@id":"/workers/reference/migrate-to-module-workers/","name":"Migrate from Service Workers to ES Modules"}}]}
```
