One-liner x402 pay-per-request for FastAPI endpoints
Project description
FastAPI x402
One-liner cryptocurrency payments for FastAPI endpoints using the x402 protocol.
Transform any FastAPI endpoint into a paid API with just a single decorator. Accept stablecoin payments (USDC) on Base network with automatic settlement.
from fastapi import FastAPI
from fastapi_x402 import init_x402, pay
app = FastAPI()
init_x402(app, pay_to="0x...", facilitator_url="https://x402.org/facilitator")
@app.get("/premium-data")
@pay("$0.01") # Require 1 cent payment
async def get_premium_data():
return {"data": "This costs $0.01 to access!"}
🚀 Features
- 🔒 Pay-per-request: Monetize individual API calls
- 🌐 Multi-network support: Accept payments on Base, Avalanche, and IoTeX
- 💰 Stablecoin payments: Accept USDC across all supported networks
- ⚡ Real-time settlement: Automatic blockchain settlement
- 🛡️ Replay protection: Cryptographic payment verification
- 📝 Standard protocol: Built on the x402 payment standard
- 🔧 One-line integration: Just add
@pay("$0.01")to any endpoint - 🔄 Auto-sync: Stays synchronized with Coinbase Facilitator supported networks
- 🎯 Network control: Server dictates which networks clients can use for payment
📦 Installation
pip install fastapi-x402
🎯 Quick Start
1. Environment Setup
Create a .env file in your project root:
# Required: Your merchant wallet address
PAY_TO_ADDRESS=0x1234567890123456789012345678901234567890
# For testnet development (default)
# Uses public x402.org facilitator - no additional setup needed
# For mainnet production (requires Coinbase CDP)
# CDP_API_KEY_ID=your_cdp_api_key_id
# CDP_API_KEY_SECRET=your_cdp_api_secret
2. Basic FastAPI Integration
from fastapi import FastAPI
from fastapi_x402 import init_x402, pay
app = FastAPI()
# Initialize x402 (loads from .env and adds middleware automatically)
init_x402(app, network="base-sepolia") # or "base" for mainnet
@app.get("/free")
async def free_endpoint():
return {"message": "This endpoint is free!"}
@app.get("/premium")
@pay("$0.01") # Require 1 cent payment
async def premium_endpoint():
return {"data": "This cost 1 cent to access!"}
@app.get("/expensive")
@pay("$1.00") # Require $1 payment
async def expensive_endpoint():
return {"premium_data": "This is worth $1!"}
3. Run Your Server
uvicorn main:app --host 0.0.0.0 --port 8000
4. Test the Payment Flow
Without payment (gets 402 response):
curl http://localhost:8000/paid-endpoint
With payment (success):
curl -H "X-PAYMENT: eyJ4NDAyVmVyc2lvbiI6..." http://localhost:8000/paid-endpoint
💳 Client Integration
JavaScript/TypeScript
import { X402Client } from '@x402/client';
const client = new X402Client({
network: 'base-sepolia',
privateKey: 'your-wallet-private-key'
});
const response = await client.get('http://localhost:8000/paid-endpoint');
console.log(response.data); // Automatically handles payment
Python
import httpx
from x402_client import X402Client # Coming soon
client = X402Client(network='base-sepolia', private_key='...')
response = await client.get('http://localhost:8000/paid-endpoint')
🔧 Configuration
Environment Variables (.env file)
# Required
PAY_TO_ADDRESS=0x1234567890123456789012345678901234567890
# Optional: Network configuration
X402_NETWORK=base-sepolia # Single network
# X402_NETWORK=base,avalanche,iotex # Multiple networks
# X402_NETWORK=mainnets # All mainnets
# X402_NETWORK=testnets # All testnets
# Optional: Custom facilitator (advanced)
# FACILITATOR_URL=https://your-facilitator.com
# Required for mainnet production
# CDP_API_KEY_ID=your_coinbase_cdp_api_key_id
# CDP_API_KEY_SECRET=your_coinbase_cdp_api_secret
Testnet vs Mainnet Setup
🧪 Testnet Development (Default)
# .env
PAY_TO_ADDRESS=0x...
X402_NETWORK=base-sepolia
# main.py
init_x402(app) # Uses public facilitator, no API keys needed
🚀 Mainnet Production
# .env
PAY_TO_ADDRESS=0x...
X402_NETWORK=base # or avalanche, iotex
CDP_API_KEY_ID=your_key_id
CDP_API_KEY_SECRET=your_secret
# main.py
init_x402(app) # Auto-detects CDP credentials for mainnet
Multi-Network Configuration
from fastapi_x402 import init_x402
# Support all available networks
init_x402(app, network="all") # Accepts payments on Base, Avalanche, and IoTeX
# Support specific networks
init_x402(app, network=["base", "avalanche"]) # Multiple networks
# Network shortcuts
init_x402(app, network="testnets") # Base Sepolia + Avalanche Fuji
init_x402(app, network="mainnets") # Base + Avalanche + IoTeX mainnet
Manual Configuration (No .env)
# Direct parameter passing (overrides .env)
init_x402(
app,
pay_to="0x...",
network="base-sepolia",
facilitator_url="https://x402.org/facilitator"
)
Network Information
from fastapi_x402 import get_supported_networks_list, get_config_for_network
# List all supported networks
networks = get_supported_networks_list()
print(networks) # ['base-sepolia', 'base', 'avalanche-fuji', 'avalanche', 'iotex']
# Get configuration for a specific network
config = get_config_for_network("base")
print(config["default_asset"]["address"]) # 0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913
🎯 Network Control Patterns
Your API can control which networks clients can use to pay:
1. Single Network (Server Dictates)
# Only accept Base network payments
init_x402(app, pay_to="0x...", network="base")
@pay("$0.01")
@app.get("/base-only")
def base_only():
return {"data": "Must pay with Base USDC"}
402 Response: Client MUST pay on Base network
{"accepts": [{"network": "base", "asset": "0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913"}]}
2. Multiple Options (Client Chooses)
# Accept payments from multiple networks
init_x402(app, pay_to="0x...", network=["base", "avalanche", "iotex"])
@pay("$0.01")
@app.get("/multi-choice")
def multi_choice():
return {"data": "Pay with any supported network"}
402 Response: Client can choose from multiple networks
{"accepts": [
{"network": "base", "asset": "0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913"},
{"network": "avalanche", "asset": "0xB97EF9Ef8734C71904D8002F8b6Bc66Dd9c48a6E"},
{"network": "iotex", "asset": "0xcdf79194c6c285077a58da47641d4dbe51f63542"}
]}
3. Network-Specific Endpoints
@app.get("/pay-with-avalanche")
async def avalanche_only(request: Request):
# Custom 402 response for Avalanche only
return JSONResponse(status_code=402, content={
"accepts": [{"network": "avalanche", "asset": "0xB97..."}]
})
4. Dynamic Network Selection
@app.get("/premium/{network}")
def network_specific(network: str):
if network not in ["base", "avalanche"]:
raise HTTPException(400, "Network not supported")
# Endpoint accepts payment only on specified network
📋 API Reference
init_x402(app, merchant_wallet, facilitator_url=None, config=None)
Initialize x402 middleware for your FastAPI app.
Parameters:
app(FastAPI): Your FastAPI application instancemerchant_wallet(str): Your wallet address to receive paymentsfacilitator_url(str, optional): Custom facilitator URLconfig(X402Config, optional): Advanced configuration
@pay(amount)
Decorator to require payment for an endpoint.
Parameters:
amount(str): Payment amount (e.g., "$0.01", "$1.00")
Returns:
- HTTP 402 if no valid payment provided
- HTTP 200 with settlement info if payment valid
🌐 Networks & Assets
Supported Networks
- Base Sepolia (testnet) -
base-sepolia - Base Mainnet -
base - Avalanche Fuji (testnet) -
avalanche-fuji - Avalanche Mainnet -
avalanche - IoTeX Mainnet -
iotex
Supported Assets
All networks support USDC with automatic configuration:
- Base Sepolia:
0x036CbD53842c5426634e7929541eC2318f3dCF7e - Base Mainnet:
0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913 - Avalanche Fuji:
0x5425890298aed601595a70AB815c96711a31Bc65 - Avalanche Mainnet:
0xB97EF9Ef8734C71904D8002F8b6Bc66Dd9c48a6E - IoTeX:
0xcdf79194c6c285077a58da47641d4dbe51f63542
🔑 Coinbase CDP Setup (Mainnet)
To accept payments on mainnet networks, you need Coinbase Developer Platform (CDP) credentials:
1. Create CDP Account
- Go to Coinbase Developer Platform
- Sign up for a CDP account
- Navigate to API Keys section
2. Generate API Keys
- Create a new API key pair
- Download and securely store your credentials
- Note your
API Key IDandAPI Secret
For mainnet production (requires Coinbase CDP)
CDP_API_KEY_ID=[REDACTED:api-key]
CDP_API_KEY_SECRET=[REDACTED:api-key]
Note: Based on the official Coinbase x402 Python implementation, facilitator requests use simple HTTP without authentication headers. CDP credentials may be stored for future use but are not currently required for facilitator API calls.
3. Add to Environment
# .env
PAY_TO_ADDRESS=0x...
CDP_API_KEY_ID=your_api_key_id_here
CDP_API_KEY_SECRET=your_api_secret_here
X402_NETWORK=base # or avalanche, iotex
4. Initialize for Mainnet
# Automatically detects CDP credentials and enables mainnet
init_x402(app, network="mainnets") # Supports Base, Avalanche, IoTeX
Why CDP? Mainnet payment settlement requires authenticated access to blockchain infrastructure. CDP provides reliable, scalable blockchain access with the security needed for production applications.
Note: Based on the official Coinbase x402 Python implementation, facilitator requests currently use simple HTTP without authentication headers. CDP credentials may be stored for future use but are not currently required for facilitator API calls.
🧪 Testing
Run Tests
# Install dev dependencies
pip install -e .[dev]
# Run tests
pytest
# Run with coverage
pytest --cov=fastapi_x402
Test on Base Sepolia
- Get testnet USDC from Base Sepolia faucet
- Set up test wallet with CDP SDK
- Use the test client examples in
/examples
🔐 Security
- Replay Protection: Each payment can only be used once
- Cryptographic Verification: All payments are cryptographically signed
- Timeout Protection: Payments expire after specified timeout
- Facilitator Validation: Third-party validation of payment authenticity
🤝 Contributing
- Fork the repository
- Create a feature branch (
git checkout -b feature/amazing-feature) - Commit your changes (
git commit -m 'Add some amazing feature') - Push to the branch (
git push origin feature/amazing-feature) - Open a Pull Request
Development Setup
git clone https://github.com/jordo1138/fastapi-x402.git
cd fastapi-x402
pip install -e .[dev]
pre-commit install
📚 Examples
Check out the /examples directory for:
- Basic server setup
- Client integration examples
- Advanced configuration
- Testing utilities
🆘 Support
- Issues: GitHub Issues
- Documentation: x402 protocol docs
📄 License
This project is licensed under the MIT License - see the LICENSE file for details.
🙏 Acknowledgments
- Built on the x402 payment standard
- Powered by Base and Coinbase CDP
- Inspired by FastAPI's elegant API design
Ready to monetize your APIs? Install fastapi-x402 and start earning in minutes! 🚀
Project details
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file fastapi_x402-0.1.6.tar.gz.
File metadata
- Download URL: fastapi_x402-0.1.6.tar.gz
- Upload date:
- Size: 28.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.12.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
cbbd5f0678475896223ca8e409bf173407262deaa92edf17458063959cfb6a41
|
|
| MD5 |
192edc99828907ad765c40444e5ce1ff
|
|
| BLAKE2b-256 |
e6ab9ce3fb936909b2989e17dd3ed3856f4867e15a68e63151cd727c7e19fc2d
|
Provenance
The following attestation bundles were made for fastapi_x402-0.1.6.tar.gz:
Publisher:
publish.yml on jordo1138/fastapi-x402
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
fastapi_x402-0.1.6.tar.gz -
Subject digest:
cbbd5f0678475896223ca8e409bf173407262deaa92edf17458063959cfb6a41 - Sigstore transparency entry: 240990975
- Sigstore integration time:
-
Permalink:
jordo1138/fastapi-x402@2ed53e2a3a9e71080933a338e00da36daba95c0a -
Branch / Tag:
refs/tags/v0.1.6 - Owner: https://github.com/jordo1138
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@2ed53e2a3a9e71080933a338e00da36daba95c0a -
Trigger Event:
release
-
Statement type:
File details
Details for the file fastapi_x402-0.1.6-py3-none-any.whl.
File metadata
- Download URL: fastapi_x402-0.1.6-py3-none-any.whl
- Upload date:
- Size: 21.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.12.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
840353d94a03669befbf9f11dcfb5a84c9050c56ef75baea209922abeb749119
|
|
| MD5 |
bb1d08d5ac4589fdb204902c8f2ac19f
|
|
| BLAKE2b-256 |
e39e4c26fabb0d51433f3313a07ef5408e4e174789eabf4ffe44432bcdc9786b
|
Provenance
The following attestation bundles were made for fastapi_x402-0.1.6-py3-none-any.whl:
Publisher:
publish.yml on jordo1138/fastapi-x402
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
fastapi_x402-0.1.6-py3-none-any.whl -
Subject digest:
840353d94a03669befbf9f11dcfb5a84c9050c56ef75baea209922abeb749119 - Sigstore transparency entry: 240990981
- Sigstore integration time:
-
Permalink:
jordo1138/fastapi-x402@2ed53e2a3a9e71080933a338e00da36daba95c0a -
Branch / Tag:
refs/tags/v0.1.6 - Owner: https://github.com/jordo1138
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@2ed53e2a3a9e71080933a338e00da36daba95c0a -
Trigger Event:
release
-
Statement type: