<!--
Sitemap:
- [Page Not Found](/404)
- [Brand](/brand): MPP brand assets and guidelines
- [Frequently asked questions](/faq): Common questions about the Machine Payments Protocol
- [Machine Payments Protocol](/overview): The open protocol for machine-to-machine payments
- [Payment methods](/payment-methods/): Available methods and how to choose one
- [Protocol overview](/protocol/): Standardizing HTTP 402 for machine-to-machine payments
- [Quickstart](/quickstart/): Get started with MPP in minutes
- [SDKs](/sdk/): Official implementations in multiple languages
- [Build with an LLM](/guides/building-with-an-llm): Use llms-full.txt to give your agent complete MPP context.
- [Accept multiple payment methods](/guides/multiple-payment-methods): Stablecoins, cards, and Bitcoin on a single endpoint
- [Accept one-time payments](/guides/one-time-payments): Charge per request with a payment-gated API
- [Accept pay-as-you-go payments](/guides/pay-as-you-go): Session-based billing with payment channels
- [Accept streamed payments](/guides/streamed-payments): Per-token billing over Server-Sent Events
- [Charge](/intents/charge): Immediate one-time payments
- [Card](/payment-methods/card/): Card payments via encrypted network tokens
- [Custom](/payment-methods/custom): Build your own payment method
- [Lightning](/payment-methods/lightning/): Bitcoin payments over the Lightning Network
- [Stripe](/payment-methods/stripe/): Cards, wallets, and other Stripe supported payment methods
- [Tempo](/payment-methods/tempo/): Stablecoin payments on the Tempo blockchain
- [Challenges](/protocol/challenges): Server-issued payment requirements
- [Credentials](/protocol/credentials): Client-submitted payment proofs
- [HTTP 402 payment required](/protocol/http-402): The status code that signals payment is required
- [Receipts](/protocol/receipts): Server acknowledgment of successful payment
- [Transports](/protocol/transports/): HTTP and MCP bindings for payment flows
- [Use with agents](/quickstart/agent): Connect your agent to MPP-enabled services
- [Use with your app](/quickstart/client): Handle payment-gated resources automatically
- [Add payments to your API](/quickstart/server): Charge for access to protected resources
- [Python SDK](/sdk/python/): The pympp Python library
- [Rust SDK](/sdk/rust/): The mpp Rust library
- [Getting started](/sdk/typescript/): The mppx TypeScript library
- [Card charge](/payment-methods/card/charge): One-time payments using encrypted network tokens
- [Lightning charge](/payment-methods/lightning/charge): One-time payments using BOLT11 invoices
- [Lightning session](/payment-methods/lightning/session): Pay-as-you-go payments over Lightning
- [Stripe charge](/payment-methods/stripe/charge): One-time payments using Shared Payment Tokens
- [Tempo charge](/payment-methods/tempo/charge): One-time TIP-20 token transfers
- [Session](/payment-methods/tempo/session): Low-cost high-throughput payments
- [HTTP transport](/protocol/transports/http): Payment flows using standard HTTP headers
- [MCP and JSON-RPC transport](/protocol/transports/mcp): Payment flows for AI tool calls
- [Client](/sdk/python/client): Handle 402 responses automatically
- [Core Types](/sdk/python/core): Challenge, Credential, and Receipt primitives
- [Server](/sdk/python/server): Protect endpoints with payment requirements
- [Client](/sdk/rust/client): Handle 402 responses automatically
- [Core types](/sdk/rust/core): Challenge, Credential, and Receipt primitives
- [Server](/sdk/rust/server): Protect endpoints with payment requirements
- [CLI Reference](/sdk/typescript/cli): Built-in command-line tool for paid HTTP requests
- [Method.from](/sdk/typescript/Method.from): Create a payment method from a definition
- [Proxy](/sdk/typescript/proxy): Paid API proxy
- [McpClient.wrap](/sdk/typescript/client/McpClient.wrap): Payment-aware MCP client
- [stripe](/sdk/typescript/client/Method.stripe): Register all Stripe intents
- [Method.stripe.charge](/sdk/typescript/client/Method.stripe.charge): One-time payments via Shared Payment Tokens
- [tempo](/sdk/typescript/client/Method.tempo): Register all Tempo intents
- [Method.tempo.charge](/sdk/typescript/client/Method.tempo.charge): One-time payments
- [Method.tempo.session](/sdk/typescript/client/Method.tempo.session): Low-cost high-throughput payments
- [tempo.session](/sdk/typescript/client/Method.tempo.session-manager): Standalone session manager
- [Mppx.create](/sdk/typescript/client/Mppx.create): Create a payment-aware fetch client
- [Mppx.restore](/sdk/typescript/client/Mppx.restore): Restore the original global fetch
- [Transport.from](/sdk/typescript/client/Transport.from): Create a custom transport
- [Transport.http](/sdk/typescript/client/Transport.http): HTTP transport for payments
- [Transport.mcp](/sdk/typescript/client/Transport.mcp): MCP transport for payments
- [BodyDigest.compute](/sdk/typescript/core/BodyDigest.compute): Compute a body digest hash
- [BodyDigest.verify](/sdk/typescript/core/BodyDigest.verify): Verify a body digest hash
- [Challenge.deserialize](/sdk/typescript/core/Challenge.deserialize): Deserialize a Challenge from a header
- [Challenge.from](/sdk/typescript/core/Challenge.from): Create a new Challenge
- [Challenge.fromHeaders](/sdk/typescript/core/Challenge.fromHeaders): Extract a Challenge from Headers
- [Challenge.fromMethod](/sdk/typescript/core/Challenge.fromMethod): Create a Challenge from a method
- [Challenge.fromResponse](/sdk/typescript/core/Challenge.fromResponse): Extract a Challenge from a Response
- [Challenge.meta](/sdk/typescript/core/Challenge.meta): Extract correlation data from a Challenge
- [Challenge.serialize](/sdk/typescript/core/Challenge.serialize): Serialize a Challenge to a header
- [Challenge.verify](/sdk/typescript/core/Challenge.verify): Verify a Challenge HMAC
- [Credential.deserialize](/sdk/typescript/core/Credential.deserialize): Deserialize a Credential from a header
- [Credential.from](/sdk/typescript/core/Credential.from): Create a new Credential
- [Credential.fromRequest](/sdk/typescript/core/Credential.fromRequest): Extract a Credential from a Request
- [Credential.serialize](/sdk/typescript/core/Credential.serialize): Serialize a Credential to a header
- [Expires](/sdk/typescript/core/Expires): Generate relative expiration timestamps
- [Method.from](/sdk/typescript/core/Method.from): Create a payment method definition
- [Method.toClient](/sdk/typescript/core/Method.toClient): Extend a method with client logic
- [Method.toServer](/sdk/typescript/core/Method.toServer): Extend a method with server verification
- [PaymentRequest.deserialize](/sdk/typescript/core/PaymentRequest.deserialize): Deserialize a payment request
- [PaymentRequest.from](/sdk/typescript/core/PaymentRequest.from): Create a payment request
- [PaymentRequest.serialize](/sdk/typescript/core/PaymentRequest.serialize): Serialize a payment request to a string
- [Receipt.deserialize](/sdk/typescript/core/Receipt.deserialize): Deserialize a Receipt from a header
- [Receipt.from](/sdk/typescript/core/Receipt.from): Create a new Receipt
- [Receipt.fromResponse](/sdk/typescript/core/Receipt.fromResponse): Extract a Receipt from a Response
- [Receipt.serialize](/sdk/typescript/core/Receipt.serialize): Serialize a Receipt to a string
- [Elysia](/sdk/typescript/middlewares/elysia): Payment middleware for Elysia
- [Express](/sdk/typescript/middlewares/express): Payment middleware for Express
- [Hono](/sdk/typescript/middlewares/hono): Payment middleware for Hono
- [Next.js](/sdk/typescript/middlewares/nextjs): Payment middleware for Next.js
- [stripe](/sdk/typescript/server/Method.stripe): Register all Stripe intents
- [Method.stripe.charge](/sdk/typescript/server/Method.stripe.charge): One-time payments via Shared Payment Tokens
- [tempo](/sdk/typescript/server/Method.tempo): Register all Tempo intents
- [Method.tempo.charge](/sdk/typescript/server/Method.tempo.charge): One-time stablecoin payments
- [Method.tempo.session](/sdk/typescript/server/Method.tempo.session): Low-cost high-throughput payments
- [Mppx.compose](/sdk/typescript/server/Mppx.compose): Present multiple payment options
- [Mppx.create](/sdk/typescript/server/Mppx.create): Create a server-side payment handler
- [Mppx.toNodeListener](/sdk/typescript/server/Mppx.toNodeListener): Adapt payments for Node.js HTTP
- [Request.toNodeListener](/sdk/typescript/server/Request.toNodeListener): Convert Fetch handlers to Node.js
- [Response.requirePayment](/sdk/typescript/server/Response.requirePayment): Create a 402 response
- [Transport.from](/sdk/typescript/server/Transport.from): Create a custom transport
- [Transport.http](/sdk/typescript/server/Transport.http): HTTP server-side transport
- [Transport.mcp](/sdk/typescript/server/Transport.mcp): Raw JSON-RPC MCP transport
- [Transport.mcpSdk](/sdk/typescript/server/Transport.mcpSdk): MCP SDK server-side transport
-->

# Accept streamed payments \[Per-token billing over Server-Sent Events]

Build a payment-gated poetry API that streams poems word-by-word and charges $0.001 per word using `mppx` sessions with Server-Sent Events (SSE).

:::info
Streamed payments extend [pay-as-you-go sessions](/guides/pay-as-you-go) with SSE. The server charges per token as content streams—if the channel balance runs out mid-stream, the client automatically sends a new voucher and the stream resumes.
:::

## Demo

Try the payment-gated poetry API. Click **Run demo** to create a wallet, fund it, and stream a paid poem.

<div style={{ height: 480 }}>
  <TerminalPoem />
</div>

## Prompt mode

Paste this into your coding agent to build the entire guide in one prompt:

<PromptBlock>
  {`Use https://mpp.dev/guides/streamed-payments.md as reference.
    Add mppx to my app with a payment-gated SSE endpoint
    that streams text word-by-word and charges $0.001 per word using the
    Tempo session payment method with PathUSD and sse: true.`}
</PromptBlock>

## Manual mode

Select your framework to follow a step-by-step guide. If your framework isn't listed, choose **Other** for a generic [Fetch API](https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API) approach compatible with most TypeScript server frameworks.

<Tabs>
  <Tab title="Next.js">
    ::::steps

    #### Install `mppx`

    :::code-group

    ```bash [npm]
    npm install mppx viem
    ```

    ```bash [pnpm]
    pnpm add mppx viem
    ```

    ```bash [bun]
    bun add mppx viem
    ```

    :::

    #### Set up `Mppx` instance with streaming

    Set up an `Mppx` instance with `sse: true` to enable SSE support on the session method.

    ```ts [app/api/sessions/poem/route.ts]
    import { Mppx, tempo } from "mppx/nextjs";

    export const mppx = Mppx.create({
      methods: [
        tempo({
          currency: "0x20c0000000000000000000000000000000000000",
          recipient: "0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266",
          sse: true,
        }),
      ],
    });
    ```

    #### Create the `/api/sessions/poem` route

    Create the poem route. The `withReceipt` method accepts an async generator—each yielded value is one SSE event and one charged word.

    ```ts [app/api/sessions/poem/route.ts]
    import { Mppx, tempo } from "mppx/nextjs";

    export const mppx = Mppx.create({
      methods: [
        tempo({
          currency: "0x20c0000000000000000000000000000000000000",
          recipient: "0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266",
          sse: true,
        }),
      ],
    });

    // [!code focus:start]
    const poem = {
      title: "The Road Not Taken",
      author: "Robert Frost",
      lines: [
        "Two roads diverged in a yellow wood,",
        "And sorry I could not travel both",
        "And be one traveler, long I stood",
        "And looked down one as far as I could",
        "To where it bent in the undergrowth;",
      ],
    };

    export const GET = mppx.session({ amount: "0.001", unitType: "word" })(
      async () => {
        const words = poem.lines.flatMap((line) => [...line.split(" "), "\\n"]);
        return async function* (stream) {
          yield JSON.stringify({ title: poem.title, author: poem.author });
          for (const word of words) {
            await stream.charge();
            yield word;
          }
        };
      },
    );
    // [!code focus:end]
    ```

    #### Test via the `mppx` CLI

    ```bash [terminal]
    # Create account funded with testnet tokens
    $ npx mppx account create

    # Stream a paid poem
    $ npx mppx http://localhost:3000/api/sessions/poem
    ```

    ::::
  </Tab>

  <Tab title="Hono">
    ::::steps

    #### Install `mppx` and `hono`

    :::code-group

    ```bash [npm]
    npm install mppx hono viem
    ```

    ```bash [pnpm]
    pnpm add mppx hono viem
    ```

    ```bash [bun]
    bun add mppx hono viem
    ```

    :::

    #### Set up `Mppx` instance with streaming

    Set up an `Mppx` instance with `sse: true` to enable SSE support on the session method.

    ```ts [server.ts]
    import { Hono } from "hono";
    import { Mppx, tempo } from "mppx/hono";

    const app = new Hono();

    const mppx = Mppx.create({
      methods: [
        tempo({
          currency: "0x20c0000000000000000000000000000000000000",
          recipient: "0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266",
          sse: true,
        }),
      ],
    });
    ```

    #### Create the `/api/sessions/poem` route

    Create the poem route with the session middleware. The handler returns an async generator—each yielded value is one SSE event and one charged word.

    ```ts [server.ts]
    import { Hono } from "hono";
    import { Mppx, tempo } from "mppx/hono";

    const app = new Hono();

    const mppx = Mppx.create({
      methods: [
        tempo({
          currency: "0x20c0000000000000000000000000000000000000",
          recipient: "0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266",
          sse: true,
        }),
      ],
    });

    // [!code focus:start]
    const poem = {
      title: "The Road Not Taken",
      author: "Robert Frost",
      lines: [
        "Two roads diverged in a yellow wood,",
        "And sorry I could not travel both",
        "And be one traveler, long I stood",
        "And looked down one as far as I could",
        "To where it bent in the undergrowth;",
      ],
    };

    app.get(
      "/api/sessions/poem",
      mppx.session({ amount: "0.001", unitType: "word" }),
      async (c) => {
        const words = poem.lines.flatMap((line) => [...line.split(" "), "\\n"]);
        return async function* (stream) {
          yield JSON.stringify({ title: poem.title, author: poem.author });
          for (const word of words) {
            await stream.charge();
            yield word;
          }
        };
      },
    );
    // [!code focus:end]
    ```

    #### Test via the `mppx` CLI

    ```bash [terminal]
    # Create account funded with testnet tokens
    $ npx mppx account create

    # Stream a paid poem
    $ npx mppx http://localhost:3000/api/sessions/poem
    ```

    ::::
  </Tab>

  <Tab title="Workers">
    ::::steps

    #### Install `mppx`

    :::code-group

    ```bash [npm]
    npm install mppx viem
    ```

    ```bash [pnpm]
    pnpm add mppx viem
    ```

    ```bash [bun]
    bun add mppx viem
    ```

    :::

    #### Set up `Mppx` instance with streaming

    Set up an `Mppx` instance with `sse: true` to enable SSE support on the session method.

    ```ts [src/index.ts]
    import { Mppx, tempo } from "mppx/server";

    const mppx = Mppx.create({
      methods: [
        tempo({
          currency: "0x20c0000000000000000000000000000000000000",
          recipient: "0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266",
          sse: true,
        }),
      ],
    });
    ```

    #### Create the `/api/sessions/poem` route

    Create the poem route. The `withReceipt` method accepts an async generator—each yielded value is one SSE event and one charged word.

    ```ts [src/index.ts]
    import { Mppx, tempo } from "mppx/server";

    const mppx = Mppx.create({
      methods: [
        tempo({
          currency: "0x20c0000000000000000000000000000000000000",
          recipient: "0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266",
          sse: true,
        }),
      ],
    });

    // [!code focus:start]
    const poem = {
      title: "The Road Not Taken",
      author: "Robert Frost",
      lines: [
        "Two roads diverged in a yellow wood,",
        "And sorry I could not travel both",
        "And be one traveler, long I stood",
        "And looked down one as far as I could",
        "To where it bent in the undergrowth;",
      ],
    };

    export default {
      async fetch(request: Request) {
        const result = await mppx.session({
          amount: "0.001",
          unitType: "word",
        })(request);

        if (result.status === 402) return result.challenge;

        const words = poem.lines.flatMap((line) => [...line.split(" "), "\\n"]);
        return result.withReceipt(async function* (stream) {
          yield JSON.stringify({ title: poem.title, author: poem.author });
          for (const word of words) {
            await stream.charge();
            yield word;
          }
        });
      },
    };
    // [!code focus:end]
    ```

    #### Test via the `mppx` CLI

    ```bash [terminal]
    # Create account funded with testnet tokens
    $ npx mppx account create

    # Stream a paid poem
    $ npx mppx http://localhost:8787
    ```

    ::::
  </Tab>

  <Tab title="Express">
    ::::steps

    #### Install `mppx` and `express`

    :::code-group

    ```bash [npm]
    npm install mppx express viem
    ```

    ```bash [pnpm]
    pnpm add mppx express viem
    ```

    ```bash [bun]
    bun add mppx express viem
    ```

    :::

    #### Set up `Mppx` instance with streaming

    Set up an `Mppx` instance with `sse: true` to enable SSE support on the session method.

    ```ts [server.ts]
    import express from "express";
    import { Mppx, tempo } from "mppx/express";

    const app = express();

    const mppx = Mppx.create({
      methods: [
        tempo({
          currency: "0x20c0000000000000000000000000000000000000",
          recipient: "0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266",
          sse: true,
        }),
      ],
    });
    ```

    #### Create the `/api/sessions/poem` route

    Create the poem route with the session middleware. The handler returns an async generator—each yielded value is one SSE event and one charged word.

    ```ts [server.ts]
    import express from "express";
    import { Mppx, tempo } from "mppx/express";

    const app = express();

    const mppx = Mppx.create({
      methods: [
        tempo({
          currency: "0x20c0000000000000000000000000000000000000",
          recipient: "0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266",
          sse: true,
        }),
      ],
    });

    // [!code focus:start]
    const poem = {
      title: "The Road Not Taken",
      author: "Robert Frost",
      lines: [
        "Two roads diverged in a yellow wood,",
        "And sorry I could not travel both",
        "And be one traveler, long I stood",
        "And looked down one as far as I could",
        "To where it bent in the undergrowth;",
      ],
    };

    app.get(
      "/api/sessions/poem",
      mppx.session({ amount: "0.001", unitType: "word" }),
      async (req, res) => {
        const words = poem.lines.flatMap((line) => [...line.split(" "), "\\n"]);
        return async function* (stream) {
          yield JSON.stringify({ title: poem.title, author: poem.author });
          for (const word of words) {
            await stream.charge();
            yield word;
          }
        };
      },
    );
    // [!code focus:end]
    ```

    #### Test via the `mppx` CLI

    ```bash [terminal]
    # Create account funded with testnet tokens
    $ npx mppx account create

    # Stream a paid poem
    $ npx mppx http://localhost:3000/api/sessions/poem
    ```

    ::::
  </Tab>

  <Tab title="Other">
    This guide walks through using `mppx/server` directly with any [Fetch API](https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API)-compatible framework: [Bun](https://bun.sh), [Deno](https://deno.com), [Cloudflare Workers](https://workers.dev), and others.

    <div className="h-6" />

    ::::steps

    #### Install `mppx`

    :::code-group

    ```bash [npm]
    npm install mppx viem
    ```

    ```bash [pnpm]
    pnpm add mppx viem
    ```

    ```bash [bun]
    bun add mppx viem
    ```

    :::

    #### Set up `Mppx` instance with streaming

    Set up an `Mppx` instance with `sse: true` to enable SSE support on the session method.

    ```ts [server.ts]
    import { Mppx, tempo } from "mppx/server";

    const mppx = Mppx.create({
      methods: [
        tempo({
          currency: "0x20c0000000000000000000000000000000000000",
          recipient: "0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266",
          sse: true,
        }),
      ],
    });
    ```

    #### Create the streaming poem route

    Create the route handler. `withReceipt` accepts an async generator—each yielded value becomes one SSE `event: message` and is charged one tick (`$0.001`). If the channel balance runs out mid-stream, the server emits `event: payment-need-voucher` and pauses until the client sends a new voucher.

    ```ts [server.ts]
    import { Mppx, tempo } from "mppx/server";

    const mppx = Mppx.create({
      methods: [
        tempo({
          currency: "0x20c0000000000000000000000000000000000000",
          recipient: "0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266",
          sse: true,
        }),
      ],
    });

    // [!code focus:start]
    const poem = {
      title: "The Road Not Taken",
      author: "Robert Frost",
      lines: [
        "Two roads diverged in a yellow wood,",
        "And sorry I could not travel both",
        "And be one traveler, long I stood",
        "And looked down one as far as I could",
        "To where it bent in the undergrowth;",
      ],
    };

    Bun.serve({
      async fetch(request) {
        const result = await mppx.session({
          amount: "0.001",
          unitType: "word",
        })(request);

        if (result.status === 402) return result.challenge;

        const words = poem.lines.flatMap((line) => [...line.split(" "), "\\n"]);
        return result.withReceipt(async function* (stream) {
          yield JSON.stringify({ title: poem.title, author: poem.author });
          for (const word of words) {
            await stream.charge();
            yield word;
          }
        });
      },
    });
    // [!code focus:end]
    ```

    #### Test via the `mppx` CLI

    ```bash [terminal]
    # Create account funded with testnet tokens
    $ npx mppx account create

    # Stream a paid poem
    $ npx mppx http://localhost:3000
    ```

    ::::
  </Tab>
</Tabs>

## Client setup

Use `tempo.session()` from `mppx/client` to create a session manager. The `.sse()` method connects to the SSE endpoint and handles voucher renewal automatically—if the server requests a new voucher mid-stream, the client signs and sends one without interrupting the stream.

```ts [client.ts]
import { tempo } from "mppx/client";
import { privateKeyToAccount } from "viem/accounts";

const session = tempo.session({
  account: privateKeyToAccount("0x..."),
  maxDeposit: "1", // Lock up to 1 pathUSD per channel
});

// .sse() returns an async iterable of SSE data payloads
const stream = await session.sse("http://localhost:3000/api/sessions/poem");

for await (const word of stream) {
  process.stdout.write(word + " ");
}
```

* **`tempo.session()`** — Creates a session manager that handles the full channel lifecycle: open, voucher signing, and close.
* **`.sse()`** — Connects to an SSE endpoint. Automatically sends new vouchers when the server emits `payment-need-voucher` events.
* **`maxDeposit: '1'`** — Locks up to 1 pathUSD. At $0.001/word, this covers ~1,000 words before the channel needs a top-up.

### Closing the channel

After streaming completes, close the channel to settle and reclaim unspent deposit:

```ts twoslash [client.ts]
import { tempo } from 'mppx/client'
import { privateKeyToAccount } from 'viem/accounts'

const session = tempo.session({
  account: privateKeyToAccount('0x...'),
  maxDeposit: '1',
})

const stream = await session.sse('http://localhost:3000/api/sessions/poem')

for await (const word of stream) {
  process.stdout.write(word + ' ')
}

// Settle on-chain and reclaim unspent deposit
const receipt = await session.close()
```

## Next steps
