@paykit-sdk/polar
v1.1.99
Published
Polar provider for PayKit
Maintainers
Readme
@paykit-sdk/polar
Polar provider for PayKit
Quick Start
import { createEndpointHandlers, PayKit } from '@paykit-sdk/core';
import { polar, createPolar } from '@paykit-sdk/polar';
// Method 1: Using environment variables
const provider = polar(); // Ensure POLAR_ACCESS_TOKEN environment variable is set
// Method 2: Direct configuration
const provider = createPolar({
accessToken: process.env.POLAR_ACCESS_TOKEN,
});
export const paykit = new PayKit(provider);
export const endpoints = createEndpointHandlers(paykit);Next.js Catch All API Route (/api/paykit/[...endpoint]/route.ts)
import { endpoints } from '@/lib/paykit';
import { EndpointPath } from '@paykit-sdk/core';
import { NextRequest, NextResponse } from 'next/server';
export async function POST(
request: NextRequest,
{ params }: { params: { endpoint: string[] } },
) {
try {
// Construct the endpoint path with full type safety
const endpoint = ('/' + params.endpoint.join('/')) as EndpointPath;
const handler = endpoints[endpoint];
if (!handler) {
return NextResponse.json({ message: 'Endpoint not found' }, { status: 404 });
}
// Parse request body
const body = await request.json();
const { args } = body;
const result = await handler(...args);
return NextResponse.json({ result });
} catch (error) {
console.error('PayKit API Error:', error);
return NextResponse.json(
{ message: error instanceof Error ? error.message : 'Internal server error' },
{ status: 500 },
);
}
}Next.js Webhooks (/api/paykit/webhooks/route.ts)
import { paykit } from '@/lib/paykit';
import { NextRequest, NextResponse } from 'next/server';
export async function POST(request: NextRequest) {
const webhookSecret = process.env.POLAR_WEBHOOK_SECRET;
if (!webhookSecret) {
return NextResponse.json({ error: 'Webhook secret not configured' }, { status: 500 });
}
const webhook = paykit.webhooks
.setup({ webhookSecret })
.on('customer.created', async event => {
console.log('Customer created:', event.data);
})
.on('subscription.created', async event => {
console.log('Subscription created:', event.data);
})
.on('payment.created', async event => {
console.log('Payment created:', event.data);
})
.on('refund.created', async event => {
console.log('Refund created:', event.data);
});
const body = await request.text();
const headers = request.headers;
const fullUrl = request.url;
await webhook.handle({ body, headers, fullUrl });
return NextResponse.json({ success: true });
}Express.js with Webhook Handler
import { endpoints, paykit } from '@/lib/paykit';
import { EndpointPath } from '@paykit-sdk/core';
import express from 'express';
const app = express();
// IMPORTANT: Webhook route must come BEFORE express.json() middleware
// This ensures we get the raw body for signature verification
app.post(
'/api/webhooks/polar',
express.raw({ type: 'application/json' }),
async (req, res) => {
try {
const webhookSecret = process.env.POLAR_WEBHOOK_SECRET;
if (!webhookSecret) {
return res.status(500).json({ error: 'Webhook secret not configured' });
}
const webhook = paykit.webhooks
.setup({ webhookSecret })
.on('customer.created', async event => {
console.log('Customer created:', event.data);
})
.on('subscription.created', async event => {
console.log('Subscription created:', event.data);
})
.on('payment.created', async event => {
console.log('Payment created:', event.data);
})
.on('refund.created', async event => {
console.log('Refund created:', event.data);
});
const body = typeof req.body === 'string' ? req.body : JSON.stringify(req.body);
const headers = new Headers(Object.entries(req.headers) as [string, string][]);
const fullUrl = `${req.protocol}://${req.get('host')}${req.originalUrl}`;
await webhook.handle({ body, headers, fullUrl });
res.json({ success: true });
} catch (error) {
console.error('Webhook error:', error);
res.status(500).json({
message: error instanceof Error ? error.message : 'Webhook processing failed',
});
}
},
);
// Regular API routes use JSON middleware
app.use(express.json());
app.post('/api/paykit/*', async (req, res) => {
try {
const endpoint = req.path.replace('/api/paykit', '') as EndpointPath;
const handler = endpoints[endpoint];
if (!handler) {
return res.status(404).json({ message: 'Endpoint not found' });
}
const { args } = req.body;
const result = await handler(...args);
res.json({ result });
} catch (error) {
res.status(500).json({
message: error instanceof Error ? error.message : 'Internal server error',
});
}
});
app.listen(3000, () => {
console.log('Server running on port 3000');
});Webhook Events
- order.paid
- order.created
- customer.created
- customer.updated
- customer.deleted
- subscription.updated
- subscription.created
- subscription.revoked
- refund.created
Environment Variables
POLAR_ACCESS_TOKEN=polar_at_...
POLAR_WEBHOOK_SECRET=polar_wh_...Support
License
ISC
