Merge branch 'master' into tordev

This commit is contained in:
Riccardo Balbo 2024-11-03 00:36:42 +01:00 committed by GitHub
commit 9e39e0bd01
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
12 changed files with 262 additions and 65 deletions

View File

@ -33,6 +33,7 @@ import EyeClose from '@/svgs/eye-close-line.svg'
import Info from './info' import Info from './info'
import { useMe } from './me' import { useMe } from './me'
import classNames from 'classnames' import classNames from 'classnames'
import { useIsClient } from './use-client'
export class SessionRequiredError extends Error { export class SessionRequiredError extends Error {
constructor () { constructor () {
@ -455,6 +456,7 @@ function InputInner ({
const [field, meta, helpers] = noForm ? [{}, {}, {}] : useField(props) const [field, meta, helpers] = noForm ? [{}, {}, {}] : useField(props)
const formik = noForm ? null : useFormikContext() const formik = noForm ? null : useFormikContext()
const storageKeyPrefix = useContext(StorageKeyPrefixContext) const storageKeyPrefix = useContext(StorageKeyPrefixContext)
const isClient = useIsClient()
const storageKey = storageKeyPrefix ? storageKeyPrefix + '-' + props.name : undefined const storageKey = storageKeyPrefix ? storageKeyPrefix + '-' + props.name : undefined
@ -538,7 +540,7 @@ function InputInner ({
isInvalid={invalid} isInvalid={invalid}
isValid={showValid && meta.initialValue !== meta.value && meta.touched && !meta.error} isValid={showValid && meta.initialValue !== meta.value && meta.touched && !meta.error}
/> />
{(clear && field.value) && {(isClient && clear && field.value && !props.readOnly) &&
<Button <Button
variant={null} variant={null}
onClick={(e) => { onClick={(e) => {

View File

@ -29,6 +29,18 @@ const ITEM_PAID_ACTION_FIELDS = gql`
} }
}` }`
const ITEM_PAID_ACTION_FIELDS_NO_CHILD_COMMENTS = gql`
${COMMENTS}
fragment ItemPaidActionFieldsNoChildComments on ItemPaidAction {
result {
id
deleteScheduledAt
reminderScheduledAt
...CommentFields
}
}
`
const ITEM_ACT_PAID_ACTION_FIELDS = gql` const ITEM_ACT_PAID_ACTION_FIELDS = gql`
fragment ItemActPaidActionFields on ItemActPaidAction { fragment ItemActPaidActionFields on ItemActPaidAction {
result { result {
@ -226,11 +238,11 @@ export const CREATE_COMMENT = gql`
}` }`
export const UPDATE_COMMENT = gql` export const UPDATE_COMMENT = gql`
${ITEM_PAID_ACTION_FIELDS} ${ITEM_PAID_ACTION_FIELDS_NO_CHILD_COMMENTS}
${PAID_ACTION} ${PAID_ACTION}
mutation upsertComment($id: ID!, $text: String!, $boost: Int, ${HASH_HMAC_INPUT_1}) { mutation upsertComment($id: ID!, $text: String!, $boost: Int, ${HASH_HMAC_INPUT_1}) {
upsertComment(id: $id, text: $text, boost: $boost, ${HASH_HMAC_INPUT_2}) { upsertComment(id: $id, text: $text, boost: $boost, ${HASH_HMAC_INPUT_2}) {
...ItemPaidActionFields ...ItemPaidActionFieldsNoChildComments
...PaidActionFields ...PaidActionFields
} }
}` }`

View File

@ -141,6 +141,10 @@ export const WALLET = gql`
url url
secondaryPassword secondaryPassword
} }
... on WalletBlink {
apiKeyRecv
currencyRecv
}
} }
} }
} }
@ -181,6 +185,10 @@ export const WALLET_BY_TYPE = gql`
url url
secondaryPassword secondaryPassword
} }
... on WalletBlink {
apiKeyRecv
currencyRecv
}
} }
} }
} }

View File

@ -751,14 +751,33 @@ export const nwcSchema = object().shape({
...autowithdrawSchemaMembers ...autowithdrawSchemaMembers
}, ['nwcUrl', 'nwcUrlRecv']) }, ['nwcUrl', 'nwcUrlRecv'])
export const blinkSchema = object({ export const blinkSchema = object().shape({
apiKey: string() apiKey: string()
.required('required') .matches(/^blink_[A-Za-z0-9]+$/, { message: 'must match pattern blink_A-Za-z0-9' })
.matches(/^blink_[A-Za-z0-9]+$/, { message: 'must match pattern blink_A-Za-z0-9' }), .when(['apiKeyRecv'], ([apiKeyRecv], schema) => {
if (!apiKeyRecv) return schema.required('required if api key for receiving not set')
return schema.test({
test: apiKey => apiKey !== apiKeyRecv,
message: 'api key for sending cannot be the same as for receiving'
})
}),
apiKeyRecv: string()
.matches(/^blink_[A-Za-z0-9]+$/, { message: 'must match pattern blink_A-Za-z0-9' })
.when(['apiKey'], ([apiKey], schema) => {
if (!apiKey) return schema.required('required if api key for sending not set')
return schema.test({
test: apiKeyRecv => apiKeyRecv !== apiKey,
message: 'api key for receiving cannot be the same as for sending'
})
}),
currency: string() currency: string()
.transform(value => value ? value.toUpperCase() : 'BTC') .transform(value => value ? value.toUpperCase() : 'BTC')
.oneOf(['USD', 'BTC'], 'must be BTC or USD') .oneOf(['USD', 'BTC'], 'must be BTC or USD'),
}) currencyRecv: string()
.transform(value => value ? value.toUpperCase() : 'BTC')
.oneOf(['BTC'], 'must be BTC'),
...autowithdrawSchemaMembers
}, ['apiKey', 'apiKeyRecv'])
export const lncSchema = object({ export const lncSchema = object({
pairingPhrase: string() pairingPhrase: string()

View File

@ -110,7 +110,7 @@ function WalletFields ({ wallet: { config, fields, isConfigured } }) {
...props, ...props,
name, name,
initialValue: config?.[name], initialValue: config?.[name],
readOnly: isClient && isConfigured && editable === false && !!config?.[name], readOnly: isClient && editable === false && ((isConfigured && !!config?.[name]) || !!props.value),
groupClassName: props.hidden ? 'd-none' : undefined, groupClassName: props.hidden ? 'd-none' : undefined,
label: label label: label
? ( ? (

View File

@ -0,0 +1,26 @@
-- AlterEnum
ALTER TYPE "WalletType" ADD VALUE 'BLINK';
-- CreateTable
CREATE TABLE "WalletBlink" (
"id" SERIAL NOT NULL,
"walletId" INTEGER NOT NULL,
"created_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"updated_at" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
"apiKeyRecv" TEXT NOT NULL,
"currencyRecv" TEXT,
CONSTRAINT "WalletBlink_pkey" PRIMARY KEY ("id")
);
-- CreateIndex
CREATE UNIQUE INDEX "WalletBlink_walletId_key" ON "WalletBlink"("walletId");
-- AddForeignKey
ALTER TABLE "WalletBlink" ADD CONSTRAINT "WalletBlink_walletId_fkey" FOREIGN KEY ("walletId") REFERENCES "Wallet"("id") ON DELETE CASCADE ON UPDATE CASCADE;
-- Update wallet json
CREATE TRIGGER wallet_blink_as_jsonb
AFTER INSERT OR UPDATE ON "WalletBlink"
FOR EACH ROW EXECUTE PROCEDURE wallet_wallet_type_as_jsonb();

View File

@ -179,6 +179,7 @@ enum WalletType {
LNBITS LNBITS
NWC NWC
PHOENIXD PHOENIXD
BLINK
} }
model Wallet { model Wallet {
@ -205,6 +206,7 @@ model Wallet {
walletLNbits WalletLNbits? walletLNbits WalletLNbits?
walletNWC WalletNWC? walletNWC WalletNWC?
walletPhoenixd WalletPhoenixd? walletPhoenixd WalletPhoenixd?
walletBlink WalletBlink?
withdrawals Withdrawl[] withdrawals Withdrawl[]
InvoiceForward InvoiceForward[] InvoiceForward InvoiceForward[]
@ -273,6 +275,16 @@ model WalletNWC {
nwcUrlRecv String nwcUrlRecv String
} }
model WalletBlink {
id Int @id @default(autoincrement())
walletId Int @unique
wallet Wallet @relation(fields: [walletId], references: [id], onDelete: Cascade)
createdAt DateTime @default(now()) @map("created_at")
updatedAt DateTime @default(now()) @updatedAt @map("updated_at")
apiKeyRecv String
currencyRecv String?
}
model WalletPhoenixd { model WalletPhoenixd {
id Int @id @default(autoincrement()) id Int @id @default(autoincrement())
walletId Int @unique walletId Int @unique

View File

@ -1,10 +1,19 @@
import { galoyBlinkUrl } from 'wallets/blink' import { getScopes, SCOPE_READ, SCOPE_WRITE, getWallet, request } from 'wallets/blink/common'
export * from 'wallets/blink' export * from 'wallets/blink'
export async function testSendPayment ({ apiKey, currency }, { logger }) { export async function testSendPayment ({ apiKey, currency }, { logger }) {
currency = currency ? currency.toUpperCase() : 'BTC'
logger.info('trying to fetch ' + currency + ' wallet') logger.info('trying to fetch ' + currency + ' wallet')
const scopes = await getScopes(apiKey)
if (!scopes.includes(SCOPE_READ)) {
throw new Error('missing READ scope')
}
if (!scopes.includes(SCOPE_WRITE)) {
throw new Error('missing WRITE scope')
}
currency = currency ? currency.toUpperCase() : 'BTC'
await getWallet(apiKey, currency) await getWallet(apiKey, currency)
logger.ok(currency + ' wallet found') logger.ok(currency + ' wallet found')
} }
@ -143,45 +152,3 @@ async function getTxInfo (authToken, wallet, invoice) {
error: '' error: ''
} }
} }
async function getWallet (authToken, currency) {
const out = await request(authToken, `
query me {
me {
defaultAccount {
wallets {
id
walletCurrency
}
}
}
}
`, {})
const wallets = out.data.me.defaultAccount.wallets
for (const wallet of wallets) {
if (wallet.walletCurrency === currency) {
return wallet
}
}
throw new Error(`wallet ${currency} not found`)
}
async function request (authToken, query, variables = {}) {
const options = {
method: 'POST',
headers: {
'Content-Type': 'application/json',
'X-API-KEY': authToken
},
body: JSON.stringify({ query, variables })
}
const res = await fetch(galoyBlinkUrl, options)
if (res.status >= 400 && res.status <= 599) {
if (res.status === 401) {
throw new Error('unauthorized')
} else {
throw new Error('API responded with HTTP ' + res.status)
}
}
return res.json()
}

62
wallets/blink/common.js Normal file
View File

@ -0,0 +1,62 @@
export const galoyBlinkUrl = 'https://api.blink.sv/graphql'
export const galoyBlinkDashboardUrl = 'https://dashboard.blink.sv/'
export const SCOPE_READ = 'READ'
export const SCOPE_WRITE = 'WRITE'
export const SCOPE_RECEIVE = 'RECEIVE'
export async function getWallet (authToken, currency) {
const out = await request(authToken, `
query me {
me {
defaultAccount {
wallets {
id
walletCurrency
}
}
}
}
`, {})
const wallets = out.data.me.defaultAccount.wallets
for (const wallet of wallets) {
if (wallet.walletCurrency === currency) {
return wallet
}
}
throw new Error(`wallet ${currency} not found`)
}
export async function request (authToken, query, variables = {}) {
const options = {
method: 'POST',
headers: {
'Content-Type': 'application/json',
'X-API-KEY': authToken
},
body: JSON.stringify({ query, variables })
}
const res = await fetch(galoyBlinkUrl, options)
if (res.status >= 400 && res.status <= 599) {
// consume res
res.text().catch(() => {})
if (res.status === 401) {
throw new Error('unauthorized')
} else {
throw new Error('API responded with HTTP ' + res.status)
}
}
return res.json()
}
export async function getScopes (authToken) {
const out = await request(authToken, `
query scopes {
authorization {
scopes
}
}
`, {})
const scopes = out?.data?.authorization?.scopes
return scopes || []
}

View File

@ -1,34 +1,61 @@
import { blinkSchema } from '@/lib/validate' import { blinkSchema } from '@/lib/validate'
import { galoyBlinkDashboardUrl } from 'wallets/blink/common'
export const galoyBlinkUrl = 'https://api.blink.sv/graphql'
export const galoyBlinkDashboardUrl = 'https://dashboard.blink.sv/'
export const name = 'blink' export const name = 'blink'
export const walletType = 'BLINK'
export const walletField = 'walletBlink'
export const fieldValidation = blinkSchema
export const fields = [ export const fields = [
{ {
name: 'apiKey', name: 'apiKey',
label: 'api key', label: 'api key',
type: 'password', type: 'password',
help: `you can get an API key from [Blink Dashboard](${galoyBlinkDashboardUrl})`, help: `you can get an API key from [Blink Dashboard](${galoyBlinkDashboardUrl}).\nPlease make sure to select ONLY the 'Read' and 'Write' scopes when generating this API key.`,
placeholder: 'blink_...' placeholder: 'blink_...',
optional: 'for sending',
clientOnly: true,
editable: false
}, },
{ {
name: 'currency', name: 'currency',
label: 'wallet type', label: 'wallet type',
type: 'text', type: 'text',
help: 'the blink wallet to use (BTC or USD for stablesats)', help: 'the blink wallet to use for sending (BTC or USD for stablesats)',
placeholder: 'BTC', placeholder: 'BTC',
optional: true,
clear: true, clear: true,
autoComplete: 'off' autoComplete: 'off',
optional: 'for sending',
clientOnly: true,
editable: false
},
{
name: 'apiKeyRecv',
label: 'api key',
type: 'password',
help: `you can get an API key from [Blink Dashboard](${galoyBlinkDashboardUrl}).\nPlease make sure to select ONLY the 'Read' and 'Receive' scopes when generating this API key.`,
placeholder: 'blink_...',
optional: 'for receiving',
serverOnly: true,
editable: false
},
{
name: 'currencyRecv',
label: 'wallet type',
type: 'text',
help: 'the blink wallet to use for receiving (only BTC available)',
value: 'BTC',
clear: true,
autoComplete: 'off',
optional: 'for receiving',
serverOnly: true,
editable: false
} }
] ]
export const card = { export const card = {
title: 'Blink', title: 'Blink',
subtitle: 'use [Blink](https://blink.sv/) for payments', subtitle: 'use [Blink](https://blink.sv/) for payments',
badges: ['send'] badges: ['send & receive']
} }
export const fieldValidation = blinkSchema

61
wallets/blink/server.js Normal file
View File

@ -0,0 +1,61 @@
import { withTimeout } from '@/lib/time'
import { getScopes, SCOPE_READ, SCOPE_RECEIVE, SCOPE_WRITE, getWallet, request } from 'wallets/blink/common'
import { msatsToSats } from '@/lib/format'
export * from 'wallets/blink'
export async function testCreateInvoice ({ apiKeyRecv, currencyRecv }) {
const scopes = await getScopes(apiKeyRecv)
if (!scopes.includes(SCOPE_READ)) {
throw new Error('missing READ scope')
}
if (scopes.includes(SCOPE_WRITE)) {
throw new Error('WRITE scope must not be present')
}
if (!scopes.includes(SCOPE_RECEIVE)) {
throw new Error('missing RECEIVE scope')
}
const timeout = 15_000
currencyRecv = currencyRecv ? currencyRecv.toUpperCase() : 'BTC'
return await withTimeout(createInvoice({ msats: 1000, expiry: 1 }, { apiKeyRecv, currencyRecv }), timeout)
}
export async function createInvoice (
{ msats, description, expiry },
{ apiKeyRecv, currencyRecv }) {
currencyRecv = currencyRecv ? currencyRecv.toUpperCase() : 'BTC'
const wallet = await getWallet(apiKeyRecv, currencyRecv)
if (currencyRecv !== 'BTC') {
throw new Error('unsupported currency ' + currencyRecv)
}
const mutation = `
mutation LnInvoiceCreate($input: LnInvoiceCreateInput!) {
lnInvoiceCreate(input: $input) {
invoice {
paymentRequest
}
errors {
message
}
}
}
`
const out = await request(apiKeyRecv, mutation, {
input: {
amount: msatsToSats(msats),
expiresIn: Math.floor(expiry / 60) || 1,
memo: description,
walletId: wallet.id
}
})
const res = out.data.lnInvoiceCreate
const errors = res.errors
if (errors && errors.length > 0) {
throw new Error('failed to create invoice ' + errors.map(e => e.code + ' ' + e.message).join(', '))
}
const invoice = res.invoice.paymentRequest
return invoice
}

View File

@ -4,13 +4,14 @@ import * as lnAddr from 'wallets/lightning-address/server'
import * as lnbits from 'wallets/lnbits/server' import * as lnbits from 'wallets/lnbits/server'
import * as nwc from 'wallets/nwc/server' import * as nwc from 'wallets/nwc/server'
import * as phoenixd from 'wallets/phoenixd/server' import * as phoenixd from 'wallets/phoenixd/server'
import * as blink from 'wallets/blink/server'
import { addWalletLog } from '@/api/resolvers/wallet' import { addWalletLog } from '@/api/resolvers/wallet'
import walletDefs from 'wallets/server' import walletDefs from 'wallets/server'
import { parsePaymentRequest } from 'ln-service' import { parsePaymentRequest } from 'ln-service'
import { toPositiveNumber } from '@/lib/validate' import { toPositiveNumber } from '@/lib/validate'
import { PAID_ACTION_TERMINAL_STATES } from '@/lib/constants' import { PAID_ACTION_TERMINAL_STATES } from '@/lib/constants'
import { withTimeout } from '@/lib/time' import { withTimeout } from '@/lib/time'
export default [lnd, cln, lnAddr, lnbits, nwc, phoenixd] export default [lnd, cln, lnAddr, lnbits, nwc, phoenixd, blink]
const MAX_PENDING_INVOICES_PER_WALLET = 25 const MAX_PENDING_INVOICES_PER_WALLET = 25