// https://github.com/getAlby/js-sdk/blob/master/src/webln/NostrWeblnProvider.ts import { createContext, useCallback, useContext, useEffect, useState } from 'react' import { Relay, finalizeEvent, nip04 } from 'nostr-tools' const NWCContext = createContext() export function NWCProvider ({ children }) { const [nwcUrl, setNwcUrl] = useState('') const [walletPubkey, setWalletPubkey] = useState() const [relayUrl, setRelayUrl] = useState() const [secret, setSecret] = useState() const [enabled, setEnabled] = useState() const [isDefault, setIsDefault] = useState() const [relay, setRelay] = useState() const name = 'NWC' const storageKey = 'webln:provider:nwc' const loadConfig = useCallback(async () => { const configStr = window.localStorage.getItem(storageKey) if (!configStr) { setEnabled(undefined) return } const config = JSON.parse(configStr) const { nwcUrl, isDefault } = config setNwcUrl(nwcUrl) setIsDefault(isDefault) const params = parseWalletConnectUrl(nwcUrl) setRelayUrl(params.relayUrl) setWalletPubkey(params.walletPubkey) setSecret(params.secret) try { const supported = await validateParams(params) setEnabled(supported.includes('pay_invoice')) } catch (err) { console.error('invalid NWC config:', err) setEnabled(false) throw err } }, []) const saveConfig = useCallback(async (config) => { // immediately store config so it's not lost even if config is invalid const { nwcUrl, isDefault } = config setNwcUrl(nwcUrl) setIsDefault(isDefault) if (!nwcUrl) { setEnabled(undefined) return } const params = parseWalletConnectUrl(nwcUrl) setRelayUrl(params.relayUrl) setWalletPubkey(params.walletPubkey) setSecret(params.secret) // XXX Even though NWC allows to configure budget, // this is definitely not ideal from a security perspective. window.localStorage.setItem(storageKey, JSON.stringify(config)) try { const supported = await validateParams(params) setEnabled(supported.includes('pay_invoice')) } catch (err) { console.error('invalid NWC config:', err) setEnabled(false) throw err } }, []) const clearConfig = useCallback(() => { window.localStorage.removeItem(storageKey) setNwcUrl('') setRelayUrl(undefined) setWalletPubkey(undefined) setSecret(undefined) setEnabled(undefined) }, []) useEffect(() => { let relay (async function () { if (relayUrl) { relay = await Relay.connect(relayUrl) setRelay(relay) } })().catch((err) => { console.error(err) setRelay(null) }) return () => { relay?.close() setRelay(null) } }, [relayUrl]) const sendPayment = useCallback((bolt11) => { return new Promise(function (resolve, reject) { (async function () { // XXX set this to mock NWC relays const MOCK_NWC_RELAY = false // timeout since NWC is async (user needs to confirm payment in wallet) // timeout is same as invoice expiry const timeout = MOCK_NWC_RELAY ? 3000 : 180_000 let timer const resetTimer = () => { clearTimeout(timer) timer = setTimeout(() => { sub?.close() if (MOCK_NWC_RELAY) { const heads = Math.random() < 0.5 if (heads) { return resolve({ preimage: null }) } return reject(new Error('mock error')) } return reject(new Error('timeout')) }, timeout) } if (MOCK_NWC_RELAY) return resetTimer() const payload = { method: 'pay_invoice', params: { invoice: bolt11 } } const content = await nip04.encrypt(secret, walletPubkey, JSON.stringify(payload)) const request = finalizeEvent({ kind: 23194, created_at: Math.floor(Date.now() / 1000), tags: [['p', walletPubkey]], content }, secret) await relay.publish(request) resetTimer() const filter = { kinds: [23195], authors: [walletPubkey], '#e': [request.id] } const sub = relay.subscribe([filter], { async onevent (response) { resetTimer() try { const content = JSON.parse(await nip04.decrypt(secret, walletPubkey, response.content)) if (content.error) return reject(new Error(content.error.message)) if (content.result) return resolve({ preimage: content.result.preimage }) } catch (err) { return reject(err) } finally { clearTimeout(timer) sub.close() } }, onclose (reason) { clearTimeout(timer) reject(new Error(reason)) } }) })().catch(reject) }) }, [relay, walletPubkey, secret]) const getInfo = useCallback(() => getInfoWithRelay(relay, walletPubkey), [relay, walletPubkey]) useEffect(() => { loadConfig().catch(console.error) }, []) const value = { name, nwcUrl, relayUrl, walletPubkey, secret, saveConfig, clearConfig, enabled, isDefault, setIsDefault, getInfo, sendPayment } return ( {children} ) } export function useNWC () { return useContext(NWCContext) } async function validateParams ({ relayUrl, walletPubkey, secret }) { let infoRelay try { // validate connection by fetching info event infoRelay = await Relay.connect(relayUrl) return await getInfoWithRelay(infoRelay, walletPubkey) } finally { infoRelay?.close() } } async function getInfoWithRelay (relay, walletPubkey) { return await new Promise((resolve, reject) => { const timeout = 5000 const timer = setTimeout(() => { reject(new Error('timeout waiting for response')) sub?.close() }, timeout) const sub = relay.subscribe([ { kinds: [13194], authors: [walletPubkey] } ], { onevent (event) { clearTimeout(timer) const supported = event.content.split() resolve(supported) sub.close() }, onclose (reason) { clearTimeout(timer) reject(new Error(reason)) }, oneose () { clearTimeout(timer) reject(new Error('info event not found')) } }) }) } function parseWalletConnectUrl (walletConnectUrl) { walletConnectUrl = walletConnectUrl .replace('nostrwalletconnect://', 'http://') .replace('nostr+walletconnect://', 'http://') // makes it possible to parse with URL in the different environments (browser/node/...) const url = new URL(walletConnectUrl) const params = {} params.walletPubkey = url.host const secret = url.searchParams.get('secret') const relayUrl = url.searchParams.get('relay') if (secret) { params.secret = secret } if (relayUrl) { params.relayUrl = relayUrl } return params }