A TypeScript/JavaScript SDK for the CLINK protocol — Nostr-native static Lightning payment offers and debits.
@shocknet/clink-sdk provides a simple, robust interface for working with CLINK (noffer1... and ndebit1...) on Nostr. It enables applications and wallets to create, encode, decode, send, and receive CLINK payment requests and authorizations using Nostr relays, with full support for NIP-44 encryption and the CLINK protocol.
- No web server or Onion Messaging required: Unlike LNURL and Bolt12, all communication is over Nostr relays.
- Compatible with Lightning.Pub, ShockWallet, and other CLINK-enabled apps.
- Implements the official CLINK protocol.
- Encode/decode CLINK Static Offers (
noffer1...) and Debits (ndebit1...) per the spec. - Send and receive CLINK payment requests (kind 21001) and debit requests (kind 21002) over Nostr.
- NIP-44 encrypted payloads for privacy and security.
- TypeScript types for all protocol payloads and responses.
- Simple, high-level API for wallet and service integration.
npm install @shocknet/clink-sdk
# or
yarn add @shocknet/clink-sdkimport { nofferEncode, ndebitEncode, decodeBech32, OfferPriceType } from '@shocknet/clink-sdk';
// Encode a CLINK Offer
const noffer = nofferEncode({
pubkey: '<receiver_pubkey_hex>',
relay: 'wss://relay.example.com',
offer: 'my_offer_id',
priceType: OfferPriceType.Fixed,
price: 1000 // sats
});
// Decode a CLINK Offer
const decoded = decodeBech32(noffer);
console.log(decoded);
// { type: 'noffer', data: { pubkey, relay, offer, priceType, price } }import { ClinkSDK, NofferData } from '@shocknet/clink-sdk';
import { generatePrivateKey } from 'nostr-tools';
const sdk = new ClinkSDK({
privateKey: generatePrivateKey(), // Uint8Array
relays: ['wss://relay.example.com'],
toPubKey: '<receiver_pubkey_hex>',
});
const request: NofferData = {
offer: 'my_offer_id',
amount_sats: 1000, // sats
description: 'coffee for bob', // optional, max 100 chars
expires_in_seconds: 3600, // optional
payer_data: { name: 'Alice' },
};
const receiptCallback = (recepit) => {
console.log("got receipt", recepit)
}
sdk.Noffer(request, receiptCallback).then(response => {
if ('bolt11' in response) {
console.log('Invoice:', response.bolt11);
} else {
console.error('Error:', response.error);
}
});import { ClinkSDK, NdebitData } from '@shocknet/clink-sdk';
import { generatePrivateKey } from 'nostr-tools';
const sdk = new ClinkSDK({
privateKey: generatePrivateKey(),
relays: ['wss://relay.example.com'],
toPubKey: '<wallet_service_pubkey_hex>',
});
// Request the service to pay an invoice
const simplePaymentRequest = newNdebitPaymentRequest('<BOLT11_invoice_string>', 5000, 'my_pointer_id')
sdk.Ndebit(simplePaymentRequest).then(response => {
if (response.res === 'ok') {
if ('preimage' in response) {
console.log('Payment preimage:', response.preimage);
} else {
console.log('Payment settled internally.');
}
} else if (response.res === 'GFY') {
console.error('Debit error:', response.error);
}
});
// Request whitelisting for future payment requests
const fullAccessRequest = newNdebitFullAccessRequest('my_pointer_id')
sdk.Ndebit(fullAccessRequest).then(response => {
if (response.res === 'ok') {
console.log('Full access aproved:');
} else if (response.res === 'GFY') {
console.error('Full access request failed:', response.error);
}
});
// Request a budget
const budgetRequest = newNdebitBudgetRequest({ number: 1, unit: 'week' }, 1000, 'my_pointer_id')
sdk.Ndebit(budgetRequest).then(response => {
if (response.res === 'ok') {
console.log('Budget aproved:');
} else if (response.res === 'GFY') {
console.error('Budget request failed:', response.error);
}
});new ClinkSDK(settings: ClinkSettings, pool?: AbstractSimplePool)settings:{ privateKey: Uint8Array, relays: string[], toPubKey: string, defaultTimeoutSeconds?: number }pool: Optional, pass a custom Nostr pool (defaults toSimplePoolfrom nostr-tools).
Noffer(data: NofferData, onReceipt?: (receipt: NofferReceipt) => void, timeoutSeconds?: number)- Sends a
kind: 21001offer request. - Returns a
Promise<NofferResponse>that resolves with the invoice or an error. - The optional
onReceiptcallback is triggered if the service sends a payment receipt after the invoice is paid.
- Sends a
Ndebit(data: NdebitData, timeoutSeconds?: number)- Sends a
kind: 21002debit request. - Returns a
Promise<NdebitResponse>that resolves with the payment/budget confirmation or an error.
- Sends a
Nmanage(data: NmanageRequest, timeoutSeconds?: number)- Sends a
kind: 21003management request. - Returns a
Promise<NmanageResponse>that resolves with the result of the management action.
- Sends a
nofferEncode(offer: OfferPointer): stringndebitEncode(debit: DebitPointer): stringdecodeBech32(nip19: string): DecodeResult
NofferData:{ offer: string, amount_sats?: number, description?: string, expires_in_seconds?: number, zap?: string, payer_data?: any }NofferResponse:{ bolt11: string } | { code: number, error: string, range?: { min: number, max: number } }NofferReceipt:{ preimage?: string }NdebitData:{ pointer?: string, amount_sats?: number, bolt11?: string, frequency?: BudgetFrequency }NdebitResponse:{ res: 'ok', preimage?: string } | { res: 'GFY', error: string, code: number }OfferPointer:{ pubkey: string, relay: string, offer: string, priceType: OfferPriceType, price?: number }DebitPointer:{ pubkey: string, relay: string, pointer?: string }OfferPriceType:enum { Fixed = 0, Variable = 1, Spontaneous = 2 }BudgetFrequency:{ number: number, unit: 'day' | 'week' | 'month' }
MIT