---
title: WebSocket connections
description: Connect to WebSocket servers running in sandboxes.
image: https://developers.cloudflare.com/dev-products-preview.png
---

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

[Skip to content](#%5Ftop) 

# WebSocket connections

This guide shows you how to work with WebSocket servers running in your sandboxes.

## Choose your approach

**Expose via preview URL** \- Get a public URL for external clients to connect to. Best for public chat rooms, multiplayer games, or real-time dashboards.

**Connect with wsConnect()** \- Your Worker establishes the WebSocket connection. Best for custom routing logic, authentication gates, or when your Worker needs real-time data from sandbox services.

## Connect to WebSocket echo server

**Create the echo server:**

echo-server.ts

```

Bun.serve({

  port: 8080,

  hostname: "0.0.0.0",

  fetch(req, server) {

    if (server.upgrade(req)) {

      return;

    }

    return new Response("WebSocket echo server");

  },

  websocket: {

    message(ws, message) {

      ws.send(`Echo: ${message}`);

    },

    open(ws) {

      console.log("Client connected");

    },

    close(ws) {

      console.log("Client disconnected");

    },

  },

});


console.log("WebSocket server listening on port 8080");


```

**Extend the Dockerfile:**

Dockerfile

```

FROM docker.io/cloudflare/sandbox:0.3.3


# Copy echo server into the container

COPY echo-server.ts /workspace/echo-server.ts


# Create custom startup script

COPY startup.sh /container-server/startup.sh

RUN chmod +x /container-server/startup.sh


```

**Create startup script:**

startup.sh

```

#!/bin/bash

# Start your WebSocket server in the background

bun /workspace/echo-server.ts &

# Start SDK's control plane (needed for the SDK to work)

exec bun dist/index.js


```

**Connect from your Worker:**

* [  JavaScript ](#tab-panel-8161)
* [  TypeScript ](#tab-panel-8162)

JavaScript

```

import { getSandbox } from "@cloudflare/sandbox";


export { Sandbox } from "@cloudflare/sandbox";


export default {

  async fetch(request, env) {

    if (request.headers.get("Upgrade")?.toLowerCase() === "websocket") {

      const sandbox = getSandbox(env.Sandbox, "echo-service");

      return await sandbox.wsConnect(request, 8080);

    }


    return new Response("WebSocket endpoint");

  },

};


```

TypeScript

```

import { getSandbox } from '@cloudflare/sandbox';


export { Sandbox } from "@cloudflare/sandbox";


export default {

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

    if (request.headers.get('Upgrade')?.toLowerCase() === 'websocket') {

      const sandbox = getSandbox(env.Sandbox, 'echo-service');

      return await sandbox.wsConnect(request, 8080);

    }


    return new Response('WebSocket endpoint');


}

};


```

**Client connects:**

JavaScript

```

const ws = new WebSocket('wss://your-worker.com');

ws.onmessage = (event) => console.log(event.data);

ws.send('Hello!'); // Receives: "Echo: Hello!"


```

## Expose WebSocket service via preview URL

Get a public URL for your WebSocket server:

* [  JavaScript ](#tab-panel-8163)
* [  TypeScript ](#tab-panel-8164)

JavaScript

```

import { getSandbox, proxyToSandbox } from "@cloudflare/sandbox";


export { Sandbox } from "@cloudflare/sandbox";


export default {

  async fetch(request, env) {

    // Auto-route all requests via proxyToSandbox first

    const proxyResponse = await proxyToSandbox(request, env);

    if (proxyResponse) return proxyResponse;


    // Extract hostname from request

    const { hostname } = new URL(request.url);

    const sandbox = getSandbox(env.Sandbox, "echo-service");


    // Expose the port to get preview URL

    const { url } = await sandbox.exposePort(8080, { hostname });


    // Return URL to clients

    if (request.url.includes("/ws-url")) {

      return Response.json({ url: url.replace("https", "wss") });

    }


    return new Response("Not found", { status: 404 });

  },

};


```

TypeScript

```

import { getSandbox, proxyToSandbox } from '@cloudflare/sandbox';


export { Sandbox } from '@cloudflare/sandbox';


export default {

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

    // Auto-route all requests via proxyToSandbox first

    const proxyResponse = await proxyToSandbox(request, env);

    if (proxyResponse) return proxyResponse;


    // Extract hostname from request

    const { hostname } = new URL(request.url);

    const sandbox = getSandbox(env.Sandbox, 'echo-service');


    // Expose the port to get preview URL

    const { url } = await sandbox.exposePort(8080, { hostname });


    // Return URL to clients

    if (request.url.includes('/ws-url')) {

      return Response.json({ url: url.replace('https', 'wss') });

    }


    return new Response('Not found', { status: 404 });


}

};


```

**Client connects to preview URL:**

JavaScript

```

// Get the preview URL

const response = await fetch('https://your-worker.com/ws-url');

const { url } = await response.json();


// Connect

const ws = new WebSocket(url);

ws.onmessage = (event) => console.log(event.data);

ws.send('Hello!'); // Receives: "Echo: Hello!"


```

## Connect from Worker to get real-time data

Your Worker can connect to a WebSocket service to get real-time data, even when the incoming request isn't a WebSocket:

* [  JavaScript ](#tab-panel-8165)
* [  TypeScript ](#tab-panel-8166)

JavaScript

```

import { getSandbox } from "@cloudflare/sandbox";


export { Sandbox } from "@cloudflare/sandbox";


let initialized = false;


export default {

  async fetch(request, env) {

    // Get or create a sandbox instance

    const sandbox = getSandbox(env.Sandbox, "data-processor");


    // Check for WebSocket upgrade

    const upgrade = request.headers.get("Upgrade")?.toLowerCase();


    if (upgrade === "websocket") {

      // Initialize server on first connection

      if (!initialized) {

        await sandbox.writeFile(

          "/workspace/server.js",

          `Bun.serve({

            port: 8080,

            fetch(req, server) {

              server.upgrade(req);

            },

            websocket: {

              message(ws, msg) {

                ws.send(\`Echo: \${msg}\`);

              }

            }

          });`,

        );

        await sandbox.startProcess("bun /workspace/server.js");

        initialized = true;

      }

      // Connect to WebSocket server

      return await sandbox.wsConnect(request, 8080);

    }


    return new Response("Processed real-time data");

  },

};


```

TypeScript

```

import { getSandbox } from '@cloudflare/sandbox';


export { Sandbox } from '@cloudflare/sandbox';


let initialized = false;


export default {

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


     // Get or create a sandbox instance

    const sandbox = getSandbox(env.Sandbox, 'data-processor');


    // Check for WebSocket upgrade

    const upgrade = request.headers.get('Upgrade')?.toLowerCase();


    if (upgrade === 'websocket') {

      // Initialize server on first connection

      if (!initialized) {

        await sandbox.writeFile(

          '/workspace/server.js',

          `Bun.serve({

            port: 8080,

            fetch(req, server) {

              server.upgrade(req);

            },

            websocket: {

              message(ws, msg) {

                ws.send(\`Echo: \${msg}\`);

              }

            }

          });`

        );

        await sandbox.startProcess(

          'bun /workspace/server.js'

        );

        initialized = true;

      }

      // Connect to WebSocket server

      return await sandbox.wsConnect(request, 8080);

    }


    return new Response('Processed real-time data');


}

};


```

This pattern is useful when you need streaming data from sandbox services but want to return HTTP responses to clients.

## Troubleshooting

### Upgrade failed

Verify request has WebSocket headers:

* [  JavaScript ](#tab-panel-8159)
* [  TypeScript ](#tab-panel-8160)

JavaScript

```

console.log(request.headers.get("Upgrade")); // 'websocket'

console.log(request.headers.get("Connection")); // 'Upgrade'


```

TypeScript

```

console.log(request.headers.get('Upgrade'));    // 'websocket'

console.log(request.headers.get('Connection')); // 'Upgrade'


```

### Local development

Expose ports in Dockerfile for `wrangler dev`:

Dockerfile

```

FROM docker.io/cloudflare/sandbox:0.3.3


COPY echo-server.ts /workspace/echo-server.ts

COPY startup.sh /container-server/startup.sh

RUN chmod +x /container-server/startup.sh


# Required for local development

EXPOSE 8080


```

Note

Port exposure in Dockerfile is only required for local development. In production, all ports are automatically accessible.

## Related resources

* [Ports API reference](https://developers.cloudflare.com/sandbox/api/ports/) \- Complete API documentation
* [Preview URLs concept](https://developers.cloudflare.com/sandbox/concepts/preview-urls/) \- How preview URLs work
* [Background processes guide](https://developers.cloudflare.com/sandbox/guides/background-processes/) \- Managing long-running services

```json
{"@context":"https://schema.org","@type":"BreadcrumbList","itemListElement":[{"@type":"ListItem","position":1,"item":{"@id":"/directory/","name":"Directory"}},{"@type":"ListItem","position":2,"item":{"@id":"/sandbox/","name":"Sandbox SDK"}},{"@type":"ListItem","position":3,"item":{"@id":"/sandbox/guides/","name":"How-to guides"}},{"@type":"ListItem","position":4,"item":{"@id":"/sandbox/guides/websocket-connections/","name":"WebSocket connections"}}]}
```
