<!--
Sitemap:
- [CLI Playground](/_cli)
- [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 internet-native 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 & Tools](/sdk/): Official implementations in multiple languages
- [Building with AI](/guides/building-with-ai): Use llms-full.txt to give your coding agent complete MPP context.
- [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
- [Custom](/payment-methods/custom): Build your own payment method
- [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
- [Client quickstart](/quickstart/client): Handle payment-gated resources automatically
- [presto](/quickstart/presto): Make paid HTTP requests from the command line
- [Server quickstart](/quickstart/server): Charge for resources and verify payment credentials
- [Python SDK](/sdk/python/): The pympp Python library
- [Rust SDK](/sdk/rust/): The mpp Rust library
- [Getting started](/sdk/typescript/): The mppx TypeScript library
- [presto](/tools/presto): Command-line HTTP client for MPP
- [Stripe charge](/payment-methods/stripe/charge): One-time payments using Stripe 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 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
- [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)
- [presto examples](/tools/presto/examples): Real-world usage patterns
- [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
- [Mppx.create](/sdk/typescript/client/Mppx.create)
- [Mppx.restore](/sdk/typescript/client/Mppx.restore)
- [Transport.from](/sdk/typescript/client/Transport.from)
- [Transport.http](/sdk/typescript/client/Transport.http)
- [Transport.mcp](/sdk/typescript/client/Transport.mcp)
- [BodyDigest.compute](/sdk/typescript/core/BodyDigest.compute)
- [BodyDigest.verify](/sdk/typescript/core/BodyDigest.verify)
- [Challenge.deserialize](/sdk/typescript/core/Challenge.deserialize)
- [Challenge.from](/sdk/typescript/core/Challenge.from)
- [Challenge.fromHeaders](/sdk/typescript/core/Challenge.fromHeaders)
- [Challenge.fromMethod](/sdk/typescript/core/Challenge.fromMethod)
- [Challenge.fromResponse](/sdk/typescript/core/Challenge.fromResponse)
- [Challenge.meta](/sdk/typescript/core/Challenge.meta)
- [Challenge.serialize](/sdk/typescript/core/Challenge.serialize)
- [Challenge.verify](/sdk/typescript/core/Challenge.verify)
- [Credential.deserialize](/sdk/typescript/core/Credential.deserialize)
- [Credential.from](/sdk/typescript/core/Credential.from)
- [Credential.fromRequest](/sdk/typescript/core/Credential.fromRequest)
- [Credential.serialize](/sdk/typescript/core/Credential.serialize)
- [Expires](/sdk/typescript/core/Expires)
- [Method.from](/sdk/typescript/core/Method.from)
- [Method.toClient](/sdk/typescript/core/Method.toClient)
- [Method.toServer](/sdk/typescript/core/Method.toServer)
- [PaymentRequest.deserialize](/sdk/typescript/core/PaymentRequest.deserialize)
- [PaymentRequest.from](/sdk/typescript/core/PaymentRequest.from)
- [PaymentRequest.serialize](/sdk/typescript/core/PaymentRequest.serialize)
- [Receipt.deserialize](/sdk/typescript/core/Receipt.deserialize)
- [Receipt.from](/sdk/typescript/core/Receipt.from)
- [Receipt.fromResponse](/sdk/typescript/core/Receipt.fromResponse)
- [Receipt.serialize](/sdk/typescript/core/Receipt.serialize)
- [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
- [Method.tempo.charge](/sdk/typescript/server/Method.tempo.charge)
- [Method.tempo.session](/sdk/typescript/server/Method.tempo.session): Low-cost high-throughput payments
- [Mppx.create](/sdk/typescript/server/Mppx.create)
- [Mppx.toNodeListener](/sdk/typescript/server/Mppx.toNodeListener)
- [Transport.from](/sdk/typescript/server/Transport.from)
- [Transport.http](/sdk/typescript/server/Transport.http)
- [Transport.mcp](/sdk/typescript/server/Transport.mcp)
- [Transport.mcpSdk](/sdk/typescript/server/Transport.mcpSdk)
-->

# Accept pay-as-you-go payments \[Session-based billing with payment channels]

Build a payment-gated photo gallery API that charges $0.01 per photo using `mppx` sessions.
The server returns random photos from [Picsum](https://picsum.photos) behind a paywall,
but you could imagine generating images with an AI model instead like [OpenAI Image Generation](https://developers.openai.com/api/reference/resources/images).

:::info
Unlike [one-time payments](/guides/one-time-payments), sessions open a payment channel once and
use offchain vouchers for each subsequent request—vouchers are **not bottlenecked by blockchain throughput**, they are processed in pure CPU-bound signature checks.
:::

## Demo

Try the live payment-gated photo gallery API running in this command line interface.

<Cli.Demo token={pathUsd} restartStep={1}>
  <Cli.Startup />

  <Cli.ConnectWallet />

  <Cli.Faucet />

  <Cli.Gallery />
</Cli.Demo>

## Prompt mode

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

<PromptBlock>
  {`Use https://mpp.dev/guides/pay-as-you-go.md as reference.
    Add mppx to my app with a payment-gated gallery endpoint
    that charges $0.01 per photo using the Tempo session payment method with
    PathUSD. When payment is verified, fetch a random photo from
    https://picsum.photos/200/200 and return the URL as JSON.`}
</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

    Set up an `Mppx` instance with the `tempo` method.

    * `recipient` is the address where you receive payments.
    * `currency` is the token address for payments (in this case, `pathUSD`).

    ```ts [app/api/sessions/photo/route.ts]
    import { Mppx, tempo } from 'mppx/nextjs'

    export const mppx = Mppx.create({
      methods: [tempo({
        currency: '0x20c0000000000000000000000000000000000000',
        recipient: '0xa726a1CD723409074DF9108A2187cfA19899aCF8',
      })],
    })
    ```

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

    Create the gallery route. This route is **currently unpaid**.

    ```ts [app/api/sessions/photo/route.ts]
    import { Mppx, tempo } from 'mppx/nextjs'

    export const mppx = Mppx.create({
      methods: [tempo({
        currency: '0x20c0000000000000000000000000000000000000',
        recipient: '0xa726a1CD723409074DF9108A2187cfA19899aCF8',
      })],
    })

    // [!code focus:start]
    export const GET = async () => {
      const res = await fetch('https://picsum.photos/200/200')
      return Response.json({ url: res.url })
    }
    // [!code focus:end]
    ```

    #### Add `.session` to the route handler

    Add payment verification using `mppx.session` as route middleware.
    The handler runs only after payment is verified.

    ```ts [app/api/sessions/photo/route.ts]
    import { Mppx, tempo } from 'mppx/nextjs'

    export const mppx = Mppx.create({
      methods: [tempo({
        currency: '0x20c0000000000000000000000000000000000000',
        recipient: '0xa726a1CD723409074DF9108A2187cfA19899aCF8',
      })],
    })

    // [!code focus:start]
    export const GET =
      mppx.session({ amount: '0.01', unitType: 'photo' }) // [!code ++]
      (async () => {
        const res = await fetch('https://picsum.photos/200/200')
        return Response.json({ url: res.url })
      })
    // [!code focus:end]
    ```

    #### Test via the `mppx` CLI

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

    # Make a paid request
    $ npx mppx http://localhost:3000/api/sessions/photo 
    ```

    ::::
  </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

    Set up an `Mppx` instance with the `tempo` method.

    * `recipient` is the address where you receive payments.
    * `currency` is the token address for payments (in this case, `pathUSD`).

    ```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: '0xa726a1CD723409074DF9108A2187cfA19899aCF8',
      })],
    })
    ```

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

    Create the gallery route. This route is **currently unpaid**.

    ```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: '0xa726a1CD723409074DF9108A2187cfA19899aCF8',
      })],
    })

    // [!code focus:start]
    app.get('/api/sessions/photo', async (c) => {
      const res = await fetch('https://picsum.photos/200/200')
      return c.json({ url: res.url })
    })
    // [!code focus:end]
    ```

    #### Add `.session` to the route handler

    Add payment verification using `mppx.session` as route middleware.
    The handler runs only after payment is verified.

    ```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: '0xa726a1CD723409074DF9108A2187cfA19899aCF8',
      })],
    })

    // [!code focus:start]
    app.get(
      '/api/sessions/photo',
      mppx.session({ amount: '0.01', unitType: 'photo' }), // [!code ++]
      async (c) => {
        const res = await fetch('https://picsum.photos/200/200')
        return c.json({ url: res.url })
      },
    )
    // [!code focus:end]
    ```

    #### Test via the `mppx` CLI

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

    # Make a paid request
    $ npx mppx http://localhost:3000/api/sessions/photo 
    ```

    ::::
  </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

    Set up an `Mppx` instance with the `tempo` method.

    * `recipient` is the address where you receive payments.
    * `currency` is the token address for payments (in this case, `pathUSD`).

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

    const mppx = Mppx.create({
      methods: [tempo({
        currency: '0x20c0000000000000000000000000000000000000',
        recipient: '0xa726a1CD723409074DF9108A2187cfA19899aCF8',
      })],
    })
    ```

    #### Create the gallery route

    Create the gallery route. This route is **currently unpaid**.

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

    const mppx = Mppx.create({
      methods: [tempo({
        currency: '0x20c0000000000000000000000000000000000000',
        recipient: '0xa726a1CD723409074DF9108A2187cfA19899aCF8',
      })],
    })

    // [!code focus:start]
    export default {
      async fetch(request: Request) {
        const res = await fetch('https://picsum.photos/200/200')
        return Response.json({ url: res.url })
      },
    }
    // [!code focus:end]
    ```

    #### Add `.session` to the route handler

    Add payment verification using `mppx.session`. If the status is `402`, return the Challenge. Otherwise, fetch the photo and attach a Receipt to the response.

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

    const mppx = Mppx.create({
      methods: [tempo({
        currency: '0x20c0000000000000000000000000000000000000',
        recipient: '0xa726a1CD723409074DF9108A2187cfA19899aCF8',
      })],
    })

    // [!code focus:start]
    export default {
      async fetch(request: Request) {
        const result = await mppx.session({ // [!code ++]
          amount: '0.01', // [!code ++]
          unitType: 'photo', // [!code ++]
        })(request) // [!code ++]

        if (result.status === 402) return result.challenge // [!code ++]

        const res = await fetch('https://picsum.photos/200/200')
        return result.withReceipt(Response.json({ url: res.url })) // [!code ++]
      },
    }
    // [!code focus:end]
    ```

    #### Test via the `mppx` CLI

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

    # Make a paid request
    $ 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

    Set up an `Mppx` instance with the `tempo` method.

    * `recipient` is the address where you receive payments.
    * `currency` is the token address for payments (in this case, `pathUSD`).

    ```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: '0xa726a1CD723409074DF9108A2187cfA19899aCF8',
      })],
    })
    ```

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

    Create the gallery route. This route is **currently unpaid**.

    ```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: '0xa726a1CD723409074DF9108A2187cfA19899aCF8',
      })],
    })

    // [!code focus:start]
    app.get('/api/sessions/photo', async (req, res) => {
      const response = await fetch('https://picsum.photos/200/200')
      res.json({ url: response.url })
    })
    // [!code focus:end]
    ```

    #### Add `.session` to the route handler

    Add payment verification using `mppx.session` as route middleware.
    The handler runs only after payment is verified.

    ```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: '0xa726a1CD723409074DF9108A2187cfA19899aCF8',
      })],
    })

    // [!code focus:start]
    app.get(
      '/api/sessions/photo',
      mppx.session({ amount: '0.01', unitType: 'photo' }), // [!code ++]
      async (req, res) => {
        const response = await fetch('https://picsum.photos/200/200')
        res.json({ url: response.url })
      },
    )
    // [!code focus:end]
    ```

    #### Test via the `mppx` CLI

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

    # Make a paid request
    $ npx mppx http://localhost:3000/api/sessions/photo 
    ```

    ::::
  </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

    Set up an `Mppx` instance with the `tempo` method.

    * `recipient` is the address where you receive payments.
    * `currency` is the token address for payments (in this case, `pathUSD`).

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

    const mppx = Mppx.create({
      methods: [tempo({
        currency: '0x20c0000000000000000000000000000000000000',
        recipient: '0xa726a1CD723409074DF9108A2187cfA19899aCF8',
      })],
    })
    ```

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

    Create the gallery route. This route is **currently unpaid**.

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

    const mppx = Mppx.create({
      methods: [tempo({
        currency: '0x20c0000000000000000000000000000000000000',
        recipient: '0xa726a1CD723409074DF9108A2187cfA19899aCF8',
      })],
    })

    // [!code focus:start]
    Bun.serve({
      async fetch(request) {
        const res = await fetch('https://picsum.photos/200/200')
        return Response.json({ url: res.url })
      },
    })
    // [!code focus:end]
    ```

    #### Add `.session` to the route handler

    Add payment verification using `mppx.session`. If the status is `402`, return the Challenge. Otherwise, fetch the photo and attach a Receipt to the response.

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

    const mppx = Mppx.create({
      methods: [tempo({
        currency: '0x20c0000000000000000000000000000000000000',
        recipient: '0xa726a1CD723409074DF9108A2187cfA19899aCF8',
      })],
    })

    // [!code focus:start]
    Bun.serve({
      async fetch(request) {
        const result = await mppx.session({ // [!code ++]
          amount: '0.01', // [!code ++]
          unitType: 'photo', // [!code ++]
        })(request) // [!code ++]

        if (result.status === 402) return result.challenge // [!code ++]

        const res = await fetch('https://picsum.photos/200/200')
        return result.withReceipt(Response.json({ url: res.url })) // [!code ++]
      },
    })
    // [!code focus:end]
    ```

    #### Test via the `mppx` CLI

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

    # Make a paid request
    $ npx mppx http://localhost:3000
    ```

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

## Client setup

When using sessions from a client, set `maxDeposit` to enable automatic channel management. This is the maximum amount of tokens the client locks into the payment channel's escrow contract. Any unspent deposit is refunded when the channel closes.

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

const mppx = Mppx.create({
  methods: [tempo({
    account: privateKeyToAccount('0x...'),
    maxDeposit: '1', // Lock up to 1 pathUSD per channel
  })],
})

// Each fetch automatically manages the session lifecycle:
// 1st request: opens channel on-chain, sends initial voucher
// 2nd+ requests: sends off-chain vouchers (no on-chain tx)
const res = await fetch('http://localhost:3000/api/sessions/photo')
```

* **`maxDeposit: '1'`** — Locks up to 1 pathUSD into the payment channel. At $0.01/photo, this covers up to 100 requests before the channel runs out.
* The client handles the full session lifecycle automatically: channel open, voucher signing, and retry after `402` responses.
* If the server sets `suggestedDeposit`, the client uses `min(suggestedDeposit, maxDeposit)`.

## Next steps
