Skip to main content

▶ Ver el flujo 402 primero

Demo de terminal interactivo — ve solicitud → 402 → pago Lightning → 200 OK, en vivo en tu navegador.

Opción A — Crear un servidor completo en un comando (más rápido)

npx create-l402-app my-api
Esto crea un proyecto completo de Express + l402-kit: server.ts, .env.example, tsconfig.json, y un endpoint /premium listo para aceptar pagos Lightning.
my-api/
  src/server.ts      ← tu API con middleware l402
  .env.example       ← plantilla de credenciales Blink/OpenNode
  package.json       ← npm install l402-kit + tsx
  tsconfig.json
  README.md
Luego:
cd my-api
cp .env.example .env   # agrega tu clave API de Blink
npm install
npm run dev
# ⚡ l402-kit server running on http://localhost:3000
# curl http://localhost:3000/premium  →  402 Payment Required

Opción B — Agregar a un proyecto existente

1. Elige tu modo

ManagedSoberano
Tiempo de configuración~2 min~5 min
Costo mensual$0$0
Comisión por transacción0.3%0%
Lo que necesitasUna dirección LightningUna cuenta Blink / Alby / BTCPay
Procesando 10,000 satsComisión de 30 satComisión de $0
Ideal paraComenzar rápidoVolumen / producción
¿No estás seguro? Comienza con Managed — sin nodo, sin cuenta, solo una dirección Lightning. Cambia a Soberano en una sola línea de código cuando quieras comisiones del 0%. Los tokens ya pagados siguen funcionando después del cambio. Obtén una dirección Lightning (gratis, 2 min): Regístrate en dashboard.blink.sv — recibirás tunombre@blink.sv. O usa Alby, Phoenix, o Wallet of Satoshi. Configuración Soberano: Regístrate en dashboard.blink.svAPI Keys → crea una clave → copia tu BTC Wallet ID desde la página de la billetera. Establece BLINK_API_KEY y BLINK_WALLET_ID en tu .env.

2. Instalar

npm install l402-kit

3. Agregar a tu API

No se necesita nodo Lightning — solo tu dirección Lightning.
import express from "express";
import { l402, ManagedProvider } from "l402-kit";

const app = express();
const lightning = ManagedProvider.fromAddress("you@yourdomain.com");

app.get("/premium", l402({ priceSats: 100, lightning }), (_req, res) => {
  res.json({ data: "You paid 100 sats. Here is your data." });
});

app.listen(3000);
// 0.3% fee · no node setup · works immediately

4. Probarlo

curl http://localhost:3000/premium
Respuesta:
{
  "error": "Payment Required",
  "price_sats": 100,
  "invoice": "lnbc1u1p...",
  "macaroon": "eyJoYXNo..."
}
Paga la factura con cualquier billetera Lightning, luego:
curl http://localhost:3000/premium \
  -H "Authorization: L402 <macaroon>:<preimage>"
Respuesta:
{ "data": "You paid 100 sats. Here is your data." }
Tu API ahora acepta pagos en Bitcoin.

Probar sin sats reales

No necesitas una billetera Lightning para probar tu integración. Usa un proveedor simulado — genera pares de tokens criptográficos válidos localmente, sin llamadas de red:
import { createHash, randomBytes } from "crypto";
import { l402 } from "l402-kit";
import type { LightningProvider, Invoice } from "l402-kit";

// Drop-in mock — generates real SHA256 hash/preimage pairs
function makeMockProvider(): LightningProvider & { preimage: string } {
  const preimage = randomBytes(32).toString("hex");
  const paymentHash = createHash("sha256").update(Buffer.from(preimage, "hex")).digest("hex");
  return {
    preimage, // use this in your test Authorization header
    async createInvoice(amountSats: number): Promise<Invoice> {
      const macaroon = Buffer.from(
        JSON.stringify({ hash: paymentHash, exp: Date.now() + 3_600_000 })
      ).toString("base64");
      return { paymentRequest: "lnbc_mock", paymentHash, macaroon, amountSats };
    },
    async checkPayment(): Promise<boolean> { return true; },
  };
}

// Usage in tests:
const mock = makeMockProvider();
app.get("/premium", l402({ priceSats: 10, lightning: mock }), handler);

// Step 1 — unauthenticated → 402
const res402 = await request(app).get("/premium");
// res402.body.macaroon  ← use this

// Step 2 — pay with mock preimage → 200
const res200 = await request(app)
  .get("/premium")
  .set("Authorization", `L402 ${res402.body.macaroon}:${mock.preimage}`);
// res200.status === 200 ✓
Para pruebas con dinero real en el entorno de pruebas, usa el testMode de OpenNode:
const lightning = new OpenNodeProvider(process.env.OPENNODE_KEY!, true); // testMode: no real sats
Guía completa de pruebas → Testing