-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy pathrouter.js
111 lines (96 loc) · 3.3 KB
/
router.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
const fastify = require('fastify')({ logger: true })
const { bech32 } = require('bech32')
const crypto = require('crypto')
const { getLnClient } = require('./lnClient')
const { getNostrPubKey, verifyZapRequest, storePendingZapRequest, handleInvoiceUpdate } = require('./nostr')
const _username = process.env.LIGESS_USERNAME
const _domain = process.env.LIGESS_DOMAIN
const _identifier = `${_username}@${_domain}`
const _lnurlpUrl = `https://${_domain}/.well-known/lnurlp/${_username}`
const _metadata = [['text/identifier', _identifier], ['text/plain', `Satoshis to ${_identifier}`]]
const _nostrPubKey = getNostrPubKey()
const unaWrapper = getLnClient()
fastify.get('/', async (request, reply) => {
// TODO Render html instead of JSON
fastify.log.warn('Unexpected request to root. When using a proxy, make sure the URL path is forwarded.')
const words = bech32.toWords(Buffer.from(_lnurlpUrl, 'utf8'))
return {
lnurlp: bech32.encode('lnurl', words, 1023),
decodedUrl: _lnurlpUrl,
info: {
title: 'Ligess: Lightning address personal server',
source: 'https://github.com/dolu89/ligess',
},
}
})
fastify.get('/.well-known/lnurlp/:username', async (request, reply) => {
try {
if (_username !== request.params.username) {
const result = { status: 'ERROR', reason: 'Username not found' }
reply.log.warn(result)
reply.code(404).send(result)
return
}
if (!request.query.amount) {
const result = {
status: 'OK',
callback: _lnurlpUrl,
tag: 'payRequest',
maxSendable: 100000000,
minSendable: 1000,
metadata: JSON.stringify(_metadata),
commentAllowed: 280,
}
if (_nostrPubKey) {
result.allowsNostr = true
result.nostrPubkey = _nostrPubKey
}
return result
} else {
const msat = request.query.amount
if (isNaN(msat)) {
const result = { status: 'ERROR', reason: 'Invalid amount specified' }
request.log.warn(result)
reply.code(400).send(result)
return
}
const zapRequest = await verifyZapRequest(request.query.nostr, msat)
const metadata = JSON.stringify(zapRequest ? zapRequest : _metadata)
const invoice = await unaWrapper.createInvoice({
amountMsats: msat,
descriptionHash: crypto
.createHash('sha256')
.update(metadata)
.digest('hex')
})
if (zapRequest) storePendingZapRequest(invoice.paymentHash, zapRequest, request.query.comment, request.log)
return {
status: 'OK',
successAction: { tag: 'message', message: `Thank you for the ${zapRequest ? 'zap' : 'payment'}! --${_username}`},
routes: [],
pr: invoice.bolt11,
disposable: false,
}
}
} catch (error) {
const result = { status: 'ERROR', reason: `An error occured while getting invoice: ${error.message}` }
request.log.warn(result)
reply.code(400).send(result)
return
}
})
if (_nostrPubKey) {
unaWrapper.watchInvoices().on('invoice-updated', (invoice) => handleInvoiceUpdate(invoice))
}
const start = async () => {
try {
await fastify.listen(
process.env.PORT || 3000,
process.env.HOST || '127.0.0.1'
)
} catch (err) {
fastify.log.error(err)
process.exit(1)
}
}
module.exports = { start }