Compare commits
No commits in common. "main" and "v1.1.0" have entirely different histories.
22 changed files with 3911 additions and 2898 deletions
4
.github/workflows/release.yml
vendored
4
.github/workflows/release.yml
vendored
|
|
@ -7,7 +7,7 @@ jobs:
|
|||
release:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
- uses: actions/checkout@v3
|
||||
- name: Create github release
|
||||
env:
|
||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||
|
|
@ -19,7 +19,7 @@ jobs:
|
|||
needs: [release]
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
- uses: actions/checkout@v3
|
||||
with:
|
||||
token: ${{ secrets.EXT_GITHUB }}
|
||||
repository: lnbits/lnbits-extensions
|
||||
|
|
|
|||
24
Makefile
24
Makefile
|
|
@ -5,27 +5,27 @@ format: prettier black ruff
|
|||
check: mypy pyright checkblack checkruff checkprettier
|
||||
|
||||
prettier:
|
||||
uv run ./node_modules/.bin/prettier --write .
|
||||
poetry run ./node_modules/.bin/prettier --write .
|
||||
pyright:
|
||||
uv run ./node_modules/.bin/pyright
|
||||
poetry run ./node_modules/.bin/pyright
|
||||
|
||||
mypy:
|
||||
uv run mypy .
|
||||
poetry run mypy .
|
||||
|
||||
black:
|
||||
uv run black .
|
||||
poetry run black .
|
||||
|
||||
ruff:
|
||||
uv run ruff check . --fix
|
||||
poetry run ruff check . --fix
|
||||
|
||||
checkruff:
|
||||
uv run ruff check .
|
||||
poetry run ruff check .
|
||||
|
||||
checkprettier:
|
||||
uv run ./node_modules/.bin/prettier --check .
|
||||
poetry run ./node_modules/.bin/prettier --check .
|
||||
|
||||
checkblack:
|
||||
uv run black --check .
|
||||
poetry run black --check .
|
||||
|
||||
checkeditorconfig:
|
||||
editorconfig-checker
|
||||
|
|
@ -33,14 +33,14 @@ checkeditorconfig:
|
|||
test:
|
||||
PYTHONUNBUFFERED=1 \
|
||||
DEBUG=true \
|
||||
uv run pytest
|
||||
poetry run pytest
|
||||
install-pre-commit-hook:
|
||||
@echo "Installing pre-commit hook to git"
|
||||
@echo "Uninstall the hook with uv run pre-commit uninstall"
|
||||
uv run pre-commit install
|
||||
@echo "Uninstall the hook with poetry run pre-commit uninstall"
|
||||
poetry run pre-commit install
|
||||
|
||||
pre-commit:
|
||||
uv run pre-commit run --all-files
|
||||
poetry run pre-commit run --all-files
|
||||
|
||||
|
||||
checkbundle:
|
||||
|
|
|
|||
49
__init__.py
49
__init__.py
|
|
@ -17,51 +17,4 @@ withdraw_ext.include_router(withdraw_ext_generic)
|
|||
withdraw_ext.include_router(withdraw_ext_api)
|
||||
withdraw_ext.include_router(withdraw_ext_lnurl)
|
||||
|
||||
|
||||
def withdraw_start() -> None:
|
||||
"""
|
||||
Register this extension's RPCs with the LNbits nostr transport so an
|
||||
HTTP-allergic client (e.g. lamassu-next ATM) can manage LNURL-withdraw
|
||||
links without touching the HTTP API. Also wires the link-owner
|
||||
resolver so subscribe_payments({tag:"withdraw", link_id:...}) can
|
||||
verify ownership.
|
||||
|
||||
No-op if the core transport module isn't present in the LNbits build.
|
||||
No runtime `if nostr_transport_enabled` guard is needed — when
|
||||
disabled, the relay pool never publishes, so registered RPCs are
|
||||
simply unreachable.
|
||||
"""
|
||||
try:
|
||||
from lnbits.core.services.nostr_transport.dispatcher import (
|
||||
AUTH_ACCOUNT,
|
||||
AUTH_WALLET,
|
||||
register_rpc,
|
||||
)
|
||||
from lnbits.core.services.nostr_transport.subscriptions import (
|
||||
register_link_owner_resolver,
|
||||
)
|
||||
except ImportError:
|
||||
return
|
||||
|
||||
from .transport_rpcs import (
|
||||
handle_lnurlw_create_link,
|
||||
handle_lnurlw_delete_link,
|
||||
handle_lnurlw_get_link,
|
||||
handle_lnurlw_list_links,
|
||||
handle_lnurlw_unique_hashes,
|
||||
handle_lnurlw_update_link,
|
||||
resolve_withdraw_owner,
|
||||
)
|
||||
|
||||
register_rpc("lnurlw_create_link", handle_lnurlw_create_link, AUTH_WALLET)
|
||||
register_rpc("lnurlw_get_link", handle_lnurlw_get_link, AUTH_WALLET)
|
||||
register_rpc("lnurlw_list_links", handle_lnurlw_list_links, AUTH_ACCOUNT)
|
||||
register_rpc("lnurlw_unique_hashes", handle_lnurlw_unique_hashes, AUTH_WALLET)
|
||||
register_rpc("lnurlw_update_link", handle_lnurlw_update_link, AUTH_WALLET)
|
||||
register_rpc("lnurlw_delete_link", handle_lnurlw_delete_link, AUTH_WALLET)
|
||||
register_link_owner_resolver(
|
||||
"withdraw", resolve_withdraw_owner, link_extra_key="withdrawal_link_id"
|
||||
)
|
||||
|
||||
|
||||
__all__ = ["db", "withdraw_ext", "withdraw_start", "withdraw_static_files"]
|
||||
__all__ = ["withdraw_ext", "withdraw_static_files", "db"]
|
||||
|
|
|
|||
|
|
@ -2,7 +2,7 @@
|
|||
"name": "Withdraw Links",
|
||||
"short_description": "Make LNURL withdraw links",
|
||||
"tile": "/withdraw/static/image/lnurl-withdraw.png",
|
||||
"version": "1.2.2-aio.2",
|
||||
"version": "1.1.0",
|
||||
"min_lnbits_version": "1.3.0",
|
||||
"contributors": [
|
||||
{
|
||||
|
|
|
|||
17
crud.py
17
crud.py
|
|
@ -1,10 +1,11 @@
|
|||
from datetime import datetime
|
||||
from typing import Optional
|
||||
|
||||
import shortuuid
|
||||
from lnbits.db import Database
|
||||
from lnbits.helpers import urlsafe_short_hash
|
||||
|
||||
from .models import CreateWithdrawData, HashCheck, PaginatedWithdraws, WithdrawLink
|
||||
from .models import CreateWithdrawData, HashCheck, WithdrawLink
|
||||
|
||||
db = Database("ext_withdraw")
|
||||
|
||||
|
|
@ -32,14 +33,13 @@ async def create_withdraw_link(
|
|||
webhook_headers=data.webhook_headers,
|
||||
webhook_body=data.webhook_body,
|
||||
custom_url=data.custom_url,
|
||||
extra=data.extra,
|
||||
number=0,
|
||||
)
|
||||
await db.insert("withdraw.withdraw_link", withdraw_link)
|
||||
return withdraw_link
|
||||
|
||||
|
||||
async def get_withdraw_link(link_id: str, num=0) -> WithdrawLink | None:
|
||||
async def get_withdraw_link(link_id: str, num=0) -> Optional[WithdrawLink]:
|
||||
link = await db.fetchone(
|
||||
"SELECT * FROM withdraw.withdraw_link WHERE id = :id",
|
||||
{"id": link_id},
|
||||
|
|
@ -52,7 +52,7 @@ async def get_withdraw_link(link_id: str, num=0) -> WithdrawLink | None:
|
|||
return link
|
||||
|
||||
|
||||
async def get_withdraw_link_by_hash(unique_hash: str, num=0) -> WithdrawLink | None:
|
||||
async def get_withdraw_link_by_hash(unique_hash: str, num=0) -> Optional[WithdrawLink]:
|
||||
link = await db.fetchone(
|
||||
"SELECT * FROM withdraw.withdraw_link WHERE unique_hash = :hash",
|
||||
{"hash": unique_hash},
|
||||
|
|
@ -67,7 +67,7 @@ async def get_withdraw_link_by_hash(unique_hash: str, num=0) -> WithdrawLink | N
|
|||
|
||||
async def get_withdraw_links(
|
||||
wallet_ids: list[str], limit: int, offset: int
|
||||
) -> PaginatedWithdraws:
|
||||
) -> tuple[list[WithdrawLink], int]:
|
||||
q = ",".join([f"'{w}'" for w in wallet_ids])
|
||||
|
||||
query_str = f"""
|
||||
|
|
@ -86,15 +86,16 @@ async def get_withdraw_links(
|
|||
query_params,
|
||||
WithdrawLink,
|
||||
)
|
||||
|
||||
result = await db.execute(
|
||||
f"""
|
||||
SELECT COUNT(*) as total FROM withdraw.withdraw_link
|
||||
WHERE wallet IN ({q})
|
||||
"""
|
||||
)
|
||||
result2 = result.mappings().first()
|
||||
total = result.mappings().first()
|
||||
|
||||
return PaginatedWithdraws(data=links, total=int(result2.total))
|
||||
return links, total.total
|
||||
|
||||
|
||||
async def remove_unique_withdraw_link(link: WithdrawLink, unique_hash: str) -> None:
|
||||
|
|
@ -109,7 +110,7 @@ async def remove_unique_withdraw_link(link: WithdrawLink, unique_hash: str) -> N
|
|||
|
||||
async def increment_withdraw_link(link: WithdrawLink) -> None:
|
||||
link.used = link.used + 1
|
||||
link.open_time = int(datetime.now().timestamp())
|
||||
link.open_time = int(datetime.now().timestamp()) + link.wait_time
|
||||
await update_withdraw_link(link)
|
||||
|
||||
|
||||
|
|
|
|||
26
helpers.py
26
helpers.py
|
|
@ -1,5 +1,4 @@
|
|||
from fastapi import Request
|
||||
from lnbits.settings import settings
|
||||
from lnurl import Lnurl
|
||||
from lnurl import encode as lnurl_encode
|
||||
from shortuuid import uuid
|
||||
|
|
@ -27,28 +26,3 @@ def create_lnurl(link: WithdrawLink, req: Request) -> Lnurl:
|
|||
f"Error creating LNURL with url: `{url!s}`, "
|
||||
"check your webserver proxy configuration."
|
||||
) from e
|
||||
|
||||
|
||||
def create_lnurl_from_baseurl(link: WithdrawLink) -> Lnurl:
|
||||
"""
|
||||
Same shape as `create_lnurl`, but composes the callback URL from
|
||||
`settings.lnbits_baseurl` instead of a FastAPI `Request`. Used by
|
||||
the nostr-transport RPC handlers, which have no HTTP request to
|
||||
derive a base URL from.
|
||||
"""
|
||||
base = settings.lnbits_baseurl.rstrip("/")
|
||||
if link.is_unique:
|
||||
usescssv = link.usescsv.split(",")
|
||||
tohash = link.id + link.unique_hash + usescssv[link.number]
|
||||
multihash = uuid(name=tohash)
|
||||
url = f"{base}/withdraw/api/v1/lnurl/{link.unique_hash}/{multihash}"
|
||||
else:
|
||||
url = f"{base}/withdraw/api/v1/lnurl/{link.unique_hash}"
|
||||
|
||||
try:
|
||||
return lnurl_encode(url)
|
||||
except Exception as e:
|
||||
raise ValueError(
|
||||
f"Error creating LNURL with url: `{url!s}`, "
|
||||
"check your `LNBITS_BASEURL` configuration."
|
||||
) from e
|
||||
|
|
|
|||
|
|
@ -139,9 +139,3 @@ async def m007_add_created_at_timestamp(db):
|
|||
"ALTER TABLE withdraw.withdraw_link "
|
||||
f"ADD COLUMN created_at TIMESTAMP DEFAULT {db.timestamp_column_default}"
|
||||
)
|
||||
|
||||
|
||||
async def m008_add_enabled_column(db):
|
||||
await db.execute(
|
||||
"ALTER TABLE withdraw.withdraw_link ADD COLUMN enabled BOOLEAN DEFAULT true;"
|
||||
)
|
||||
|
|
|
|||
|
|
@ -1,44 +0,0 @@
|
|||
"""
|
||||
Fork-specific database migrations for the aiolabs withdraw extension.
|
||||
|
||||
These migrations are tracked separately under `withdraw_fork` in the
|
||||
`dbversions` table (loaded by `lnbits/core/helpers.py:migrate_extension_database`),
|
||||
so they do not collide with upstream's `m{NNN}_*` numbering in
|
||||
`migrations.py`. Keeping the upstream-tracked file untouched means
|
||||
`git pull upstream` stays rebase-clean for schema changes.
|
||||
|
||||
Conventions:
|
||||
- Sequential numbering starting from m001.
|
||||
- Each migration is `async def m{NNN}_<description>(db)`.
|
||||
- DDL must be idempotent: a fresh install runs every migration; an
|
||||
install that already carries the column must not crash. Use
|
||||
`_alter_add_column_safe` so re-runs are no-ops.
|
||||
"""
|
||||
|
||||
|
||||
async def _alter_add_column_safe(db, sql: str) -> None:
|
||||
"""ALTER TABLE ADD COLUMN that swallows duplicate-column errors, so a
|
||||
re-run on a DB that already has the column is a silent no-op."""
|
||||
try:
|
||||
await db.execute(sql)
|
||||
except Exception as exc:
|
||||
msg = str(exc).lower()
|
||||
if "duplicate column" in msg or "already exists" in msg:
|
||||
return
|
||||
raise
|
||||
|
||||
|
||||
async def m001_aio_withdraw_schema(db):
|
||||
"""
|
||||
Apply every aiolabs schema delta on top of upstream withdraw.
|
||||
|
||||
`withdraw_link.extra` — arbitrary JSON merged into the payout payment's
|
||||
`extra` when the link is claimed (see views_lnurl). Lets a caller tag the
|
||||
resulting payment with settlement/attribution metadata an external listener
|
||||
can key on — e.g. bitSpire stamps {source, type, principal_sats, fee_sats,
|
||||
...} so the spirekeeper cash-in settlement fires off an LNURL-withdraw
|
||||
payout. Stored as TEXT; (de)serialized to a dict by the WithdrawLink model.
|
||||
"""
|
||||
await _alter_add_column_safe(
|
||||
db, "ALTER TABLE withdraw.withdraw_link ADD COLUMN extra TEXT"
|
||||
)
|
||||
32
models.py
32
models.py
|
|
@ -15,13 +15,6 @@ class CreateWithdrawData(BaseModel):
|
|||
webhook_headers: str = Query(None)
|
||||
webhook_body: str = Query(None)
|
||||
custom_url: str = Query(None)
|
||||
enabled: bool = Query(True)
|
||||
# Arbitrary JSON merged into the payout payment's `extra` when this link is
|
||||
# claimed (see views_lnurl). Lets a caller tag the resulting payment with
|
||||
# settlement/attribution metadata an external listener can key on — e.g.
|
||||
# bitSpire stamps {source, type, principal_sats, fee_sats, ...} so the
|
||||
# spirekeeper cash-in settlement fires off an LNURL-withdraw payout.
|
||||
extra: dict | None = None
|
||||
|
||||
|
||||
class WithdrawLink(BaseModel):
|
||||
|
|
@ -43,27 +36,7 @@ class WithdrawLink(BaseModel):
|
|||
webhook_headers: str = Query(None)
|
||||
webhook_body: str = Query(None)
|
||||
custom_url: str = Query(None)
|
||||
# Persisted as TEXT (JSON); merged into the payout payment's `extra` on
|
||||
# claim. LNbits' db layer (de)serializes dict-typed columns to/from JSON
|
||||
# natively (same as Payment.extra) — no per-field validator needed.
|
||||
extra: dict | None = None
|
||||
created_at: datetime
|
||||
enabled: bool = Query(True)
|
||||
lnurl: str | None = Field(
|
||||
default=None,
|
||||
no_database=True,
|
||||
deprecated=True,
|
||||
description=(
|
||||
"Deprecated: Instead of using this bech32 encoded string, dynamically "
|
||||
"generate your own static link (lud17/bech32) on the client side. "
|
||||
"Example: lnurlw://${window.location.hostname}/lnurlw/${id}"
|
||||
),
|
||||
)
|
||||
lnurl_url: str | None = Field(
|
||||
default=None,
|
||||
no_database=True,
|
||||
description="The raw LNURL callback URL (use for QR code generation)",
|
||||
)
|
||||
|
||||
@property
|
||||
def is_spent(self) -> bool:
|
||||
|
|
@ -73,8 +46,3 @@ class WithdrawLink(BaseModel):
|
|||
class HashCheck(BaseModel):
|
||||
hash: bool
|
||||
lnurl: bool
|
||||
|
||||
|
||||
class PaginatedWithdraws(BaseModel):
|
||||
data: list[WithdrawLink]
|
||||
total: int
|
||||
|
|
|
|||
3673
poetry.lock
generated
Normal file
3673
poetry.lock
generated
Normal file
File diff suppressed because it is too large
Load diff
|
|
@ -1,34 +1,38 @@
|
|||
[project]
|
||||
[tool.poetry]
|
||||
name = "lnbits-withdraw"
|
||||
version = "0.0.0"
|
||||
requires-python = ">=3.10,<3.13"
|
||||
description = "LNbits, free and open-source Lightning wallet and accounts system."
|
||||
authors = [{ name = "Alan Bits", email = "alan@lnbits.com" }]
|
||||
urls = { Homepage = "https://lnbits.com", Repository = "https://github.com/lnbits/bitcoinswitch_extension" }
|
||||
dependencies = [ "lnbits>1" ]
|
||||
|
||||
[tool.poetry]
|
||||
authors = ["Alan Bits <alan@lnbits.com>"]
|
||||
package-mode = false
|
||||
|
||||
[tool.uv]
|
||||
dev-dependencies = [
|
||||
"black",
|
||||
"pytest-asyncio",
|
||||
"pytest",
|
||||
"mypy",
|
||||
"pre-commit",
|
||||
"ruff",
|
||||
"pytest-md",
|
||||
[tool.poetry.dependencies]
|
||||
python = "~3.12 | ~3.11 | ~3.10"
|
||||
lnbits = {version = "*", allow-prereleases = true}
|
||||
|
||||
[tool.poetry.group.dev.dependencies]
|
||||
black = "^24.3.0"
|
||||
pytest-asyncio = "^0.21.0"
|
||||
pytest = "^7.3.2"
|
||||
mypy = "^1.5.1"
|
||||
pre-commit = "^3.2.2"
|
||||
ruff = "^0.3.2"
|
||||
|
||||
[build-system]
|
||||
requires = ["poetry-core>=1.0.0"]
|
||||
build-backend = "poetry.core.masonry.api"
|
||||
|
||||
[[tool.mypy.overrides]]
|
||||
module = [
|
||||
"lnbits.*",
|
||||
"lnurl.*",
|
||||
"loguru.*",
|
||||
"fastapi.*",
|
||||
"pydantic.*",
|
||||
"pyqrcode.*",
|
||||
"shortuuid.*",
|
||||
"httpx.*",
|
||||
]
|
||||
|
||||
[tool.mypy]
|
||||
plugins = ["pydantic.mypy"]
|
||||
|
||||
[tool.pydantic-mypy]
|
||||
init_forbid_extra = true
|
||||
init_typed = true
|
||||
warn_required_dynamic_aliases = true
|
||||
warn_untyped_fields = true
|
||||
ignore_missing_imports = "True"
|
||||
|
||||
[tool.pytest.ini_options]
|
||||
log_cli = false
|
||||
|
|
|
|||
|
|
@ -1,6 +1,17 @@
|
|||
const locationPath = [
|
||||
window.location.protocol,
|
||||
'//',
|
||||
window.location.host,
|
||||
window.location.pathname
|
||||
].join('')
|
||||
|
||||
const mapWithdrawLink = function (obj) {
|
||||
obj._data = _.clone(obj)
|
||||
obj.min_fsat = new Intl.NumberFormat(LOCALE).format(obj.min_withdrawable)
|
||||
obj.max_fsat = new Intl.NumberFormat(LOCALE).format(obj.max_withdrawable)
|
||||
obj.uses_left = obj.uses - obj.used
|
||||
obj.print_url = [locationPath, 'print/', obj.id].join('')
|
||||
obj.withdraw_url = [locationPath, obj.id].join('')
|
||||
obj._data.use_custom = Boolean(obj.custom_url)
|
||||
return obj
|
||||
}
|
||||
|
|
@ -14,7 +25,6 @@ window.app = Vue.createApp({
|
|||
return {
|
||||
checker: null,
|
||||
withdrawLinks: [],
|
||||
lnurl: '',
|
||||
withdrawLinksTable: {
|
||||
columns: [
|
||||
{name: 'title', align: 'left', label: 'Title', field: 'title'},
|
||||
|
|
@ -24,7 +34,7 @@ window.app = Vue.createApp({
|
|||
label: 'Created At',
|
||||
field: 'created_at',
|
||||
sortable: true,
|
||||
format: function (val) {
|
||||
format: function (val, row) {
|
||||
return new Date(val).toLocaleString()
|
||||
}
|
||||
},
|
||||
|
|
@ -37,7 +47,7 @@ window.app = Vue.createApp({
|
|||
{
|
||||
name: 'uses',
|
||||
align: 'right',
|
||||
label: 'Uses',
|
||||
label: 'Created',
|
||||
field: 'uses'
|
||||
},
|
||||
{
|
||||
|
|
@ -46,13 +56,8 @@ window.app = Vue.createApp({
|
|||
label: 'Uses left',
|
||||
field: 'uses_left'
|
||||
},
|
||||
{
|
||||
name: 'max_withdrawable',
|
||||
align: 'right',
|
||||
label: 'Max (sat)',
|
||||
field: 'max_withdrawable',
|
||||
format: LNbits.utils.formatSat
|
||||
}
|
||||
{name: 'min', align: 'right', label: 'Min (sat)', field: 'min_fsat'},
|
||||
{name: 'max', align: 'right', label: 'Max (sat)', field: 'max_fsat'}
|
||||
],
|
||||
pagination: {
|
||||
page: 1,
|
||||
|
|
@ -68,8 +73,7 @@ window.app = Vue.createApp({
|
|||
data: {
|
||||
is_unique: false,
|
||||
use_custom: false,
|
||||
has_webhook: false,
|
||||
enabled: true
|
||||
has_webhook: false
|
||||
}
|
||||
},
|
||||
simpleformDialog: {
|
||||
|
|
@ -79,8 +83,7 @@ window.app = Vue.createApp({
|
|||
use_custom: false,
|
||||
title: 'Vouchers',
|
||||
min_withdrawable: 0,
|
||||
wait_time: 1,
|
||||
enabled: true
|
||||
wait_time: 1
|
||||
}
|
||||
},
|
||||
qrCodeDialog: {
|
||||
|
|
@ -127,22 +130,22 @@ window.app = Vue.createApp({
|
|||
this.formDialog.data = {
|
||||
is_unique: false,
|
||||
use_custom: false,
|
||||
has_webhook: false,
|
||||
enabled: true
|
||||
has_webhook: false
|
||||
}
|
||||
},
|
||||
simplecloseFormDialog() {
|
||||
this.simpleformDialog.data = {
|
||||
is_unique: false,
|
||||
use_custom: false,
|
||||
enabled: true
|
||||
use_custom: false
|
||||
}
|
||||
},
|
||||
openQrCodeDialog(linkId) {
|
||||
const link = _.findWhere(this.withdrawLinks, {id: linkId})
|
||||
|
||||
this.qrCodeDialog.data = _.clone(link)
|
||||
this.qrCodeDialog.data.url =
|
||||
window.location.protocol + '//' + window.location.host
|
||||
this.qrCodeDialog.show = true
|
||||
this.activeUrl = link.lnurl_url
|
||||
},
|
||||
openUpdateDialog(linkId) {
|
||||
let link = _.findWhere(this.withdrawLinks, {id: linkId})
|
||||
|
|
@ -255,7 +258,7 @@ window.app = Vue.createApp({
|
|||
'/withdraw/api/v1/links/' + linkId,
|
||||
_.findWhere(this.g.user.wallets, {id: link.wallet}).adminkey
|
||||
)
|
||||
.then(() => {
|
||||
.then(response => {
|
||||
this.withdrawLinks = _.reject(this.withdrawLinks, function (obj) {
|
||||
return obj.id === linkId
|
||||
})
|
||||
|
|
|
|||
12
templates/withdraw/csv.html
Normal file
12
templates/withdraw/csv.html
Normal file
|
|
@ -0,0 +1,12 @@
|
|||
{% extends "print.html" %} {% block page %} {% for page in link %} {% for threes
|
||||
in page %} {% for one in threes %} {{one}}, {% endfor %} {% endfor %} {% endfor
|
||||
%} {% endblock %} {% block scripts %}
|
||||
<script>
|
||||
window.app = Vue.createApp({
|
||||
el: '#vue',
|
||||
data: function () {
|
||||
return {}
|
||||
}
|
||||
})
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
|
@ -4,32 +4,24 @@
|
|||
<q-card class="q-pa-lg">
|
||||
<q-card-section class="q-pa-none">
|
||||
<div class="text-center">
|
||||
<q-badge v-if="spent" color="red" class="q-mb-md"
|
||||
>Withdraw is spent.</q-badge
|
||||
>
|
||||
<q-badge v-if="spent" color="red" class="q-mb-md"
|
||||
>Withdraw is spent.</q-badge
|
||||
>
|
||||
<q-badge v-else-if="!enabled" color="grey" class="q-mb-md"
|
||||
>Withdraw is disabled.</q-badge
|
||||
>
|
||||
<a v-else class="text-secondary" :href="link">
|
||||
<lnbits-qrcode-lnurl
|
||||
prefix="lnurlw"
|
||||
:url="url"
|
||||
@update:lnurl="v => lnurl = v"
|
||||
></lnbits-qrcode-lnurl>
|
||||
{% if link.is_spent %}
|
||||
<q-badge color="red" class="q-mb-md">Withdraw is spent.</q-badge>
|
||||
{% endif %}
|
||||
<a class="text-secondary" href="lightning:{{ lnurl }}">
|
||||
<lnbits-qrcode
|
||||
:value="this.here + '/?lightning={{lnurl }}'"
|
||||
></lnbits-qrcode>
|
||||
</a>
|
||||
</div>
|
||||
<div class="row q-mt-lg q-gutter-sm">
|
||||
<q-btn outline color="grey" @click="copyText(lnurl)"
|
||||
<q-btn outline color="grey" @click="copyText('{{ lnurl }}')"
|
||||
>Copy LNURL</q-btn
|
||||
>
|
||||
<q-btn
|
||||
outline
|
||||
color="grey"
|
||||
icon="nfc"
|
||||
@click="writeNfcTag(lnurl)"
|
||||
@click="writeNfcTag(' {{ lnurl }} ')"
|
||||
:disable="nfcTagWriting"
|
||||
></q-btn>
|
||||
</div>
|
||||
|
|
@ -60,11 +52,8 @@
|
|||
mixins: [window.windowMixin],
|
||||
data() {
|
||||
return {
|
||||
spent: {{ 'true' if spent else 'false' }},
|
||||
url: '{{ lnurl_url }}',
|
||||
lnurl: '',
|
||||
nfcTagWriting: false,
|
||||
enabled: {{ 'true' if enabled else 'false' }}
|
||||
here: location.protocol + '//' + location.host,
|
||||
nfcTagWriting: false
|
||||
}
|
||||
}
|
||||
})
|
||||
|
|
|
|||
|
|
@ -38,7 +38,6 @@
|
|||
>
|
||||
<template v-slot:header="props">
|
||||
<q-tr :props="props">
|
||||
<q-th auto-width></q-th>
|
||||
<q-th auto-width></q-th>
|
||||
<q-th auto-width></q-th>
|
||||
<q-th
|
||||
|
|
@ -52,48 +51,49 @@
|
|||
</template>
|
||||
<template v-slot:body="props">
|
||||
<q-tr :props="props">
|
||||
<q-td auto-width>
|
||||
<q-icon
|
||||
name="power_settings_new"
|
||||
:color="props.row.enabled ? 'green' : 'red'"
|
||||
size="xs"
|
||||
>
|
||||
<q-tooltip>
|
||||
<span
|
||||
v-text="props.row.enabled ? 'Withdraw link is enabled' : 'Withdraw link is disabled'"
|
||||
></span>
|
||||
</q-tooltip>
|
||||
</q-icon>
|
||||
</q-td>
|
||||
<q-td auto-width>
|
||||
<q-btn
|
||||
unelevated
|
||||
dense
|
||||
size="xs"
|
||||
icon="launch"
|
||||
:color="($q.dark.isActive) ? 'grey-7' : 'grey-5'"
|
||||
type="a"
|
||||
:href="'/withdraw/' + props.row.id"
|
||||
:href="props.row.withdraw_url"
|
||||
target="_blank"
|
||||
>
|
||||
<q-tooltip>Shareable link</q-tooltip></q-btn
|
||||
<q-tooltip> shareable link </q-tooltip></q-btn
|
||||
>
|
||||
<q-btn
|
||||
unelevated
|
||||
dense
|
||||
size="xs"
|
||||
icon="web_asset"
|
||||
:color="($q.dark.isActive) ? 'grey-7' : 'grey-5'"
|
||||
type="a"
|
||||
:href="'/withdraw/img/' + props.row.id"
|
||||
target="_blank"
|
||||
><q-tooltip> embeddable image </q-tooltip></q-btn
|
||||
>
|
||||
<q-btn
|
||||
unelevated
|
||||
dense
|
||||
size="xs"
|
||||
icon="reorder"
|
||||
:color="($q.dark.isActive) ? 'grey-7' : 'grey-5'"
|
||||
type="a"
|
||||
:href="'/withdraw/csv/' + props.row.id"
|
||||
target="_blank"
|
||||
><q-tooltip>CSV download</q-tooltip></q-btn
|
||||
><q-tooltip> csv list </q-tooltip></q-btn
|
||||
>
|
||||
<q-btn
|
||||
unelevated
|
||||
dense
|
||||
size="xs"
|
||||
icon="visibility"
|
||||
:color="($q.dark.isActive) ? 'grey-7' : 'grey-5'"
|
||||
@click="openQrCodeDialog(props.row.id)"
|
||||
><q-tooltip>view LNURL</q-tooltip></q-btn
|
||||
><q-tooltip> view LNURL </q-tooltip></q-btn
|
||||
>
|
||||
</q-td>
|
||||
<q-td auto-width>
|
||||
|
|
@ -139,7 +139,7 @@
|
|||
<q-card>
|
||||
<q-card-section>
|
||||
<h6 class="text-subtitle1 q-my-none">
|
||||
LNbits LNURL withdraw extension
|
||||
{{SITE_TITLE}} LNURL-withdraw extension
|
||||
</h6>
|
||||
</q-card-section>
|
||||
<q-card-section class="q-pa-none">
|
||||
|
|
@ -252,20 +252,6 @@
|
|||
hint="Custom data as JSON string, will get posted along with webhook 'body' field."
|
||||
></q-input>
|
||||
<q-list>
|
||||
<q-item tag="label" class="rounded-borders">
|
||||
<q-item-section avatar>
|
||||
<q-checkbox
|
||||
v-model="formDialog.data.enabled"
|
||||
color="primary"
|
||||
></q-checkbox>
|
||||
</q-item-section>
|
||||
<q-item-section>
|
||||
<q-item-label>Enable / Disable </q-item-label>
|
||||
<q-item-label caption
|
||||
>You can enable or disable these vouchers</q-item-label
|
||||
>
|
||||
</q-item-section>
|
||||
</q-item>
|
||||
<q-item tag="label" class="rounded-borders">
|
||||
<q-item-section avatar>
|
||||
<q-checkbox
|
||||
|
|
@ -378,20 +364,6 @@
|
|||
label="Number of vouchers"
|
||||
></q-input>
|
||||
<q-list>
|
||||
<q-item tag="label" class="rounded-borders">
|
||||
<q-item-section avatar>
|
||||
<q-checkbox
|
||||
v-model="simpleformDialog.data.enabled"
|
||||
color="primary"
|
||||
></q-checkbox>
|
||||
</q-item-section>
|
||||
<q-item-section>
|
||||
<q-item-label>Enable / Disable </q-item-label>
|
||||
<q-item-label caption
|
||||
>You can enable or disable these vouchers</q-item-label
|
||||
>
|
||||
</q-item-section>
|
||||
</q-item>
|
||||
<q-item tag="label" class="rounded-borders">
|
||||
<q-item-section avatar>
|
||||
<q-checkbox
|
||||
|
|
@ -441,11 +413,9 @@
|
|||
|
||||
<q-dialog v-model="qrCodeDialog.show" position="top">
|
||||
<q-card v-if="qrCodeDialog.data" class="q-pa-lg lnbits__dialog-card">
|
||||
<lnbits-qrcode-lnurl
|
||||
:url="activeUrl"
|
||||
@update:lnurl="v => lnurl = v"
|
||||
prefix="lnurlw"
|
||||
></lnbits-qrcode-lnurl>
|
||||
<lnbits-qrcode
|
||||
:value="qrCodeDialog.data.url + '/?lightning=' + qrCodeDialog.data.lnurl"
|
||||
></lnbits-qrcode>
|
||||
<p style="word-break: break-all">
|
||||
<strong>ID:</strong> <span v-text="qrCodeDialog.data.id"></span><br />
|
||||
<strong>Unique:</strong>
|
||||
|
|
@ -470,32 +440,31 @@
|
|||
<q-btn
|
||||
outline
|
||||
color="grey"
|
||||
@click="copyText(lnurl, 'LNURL copied to clipboard!')"
|
||||
@click="copyText(qrCodeDialog.data.lnurl, 'LNURL copied to clipboard!')"
|
||||
class="q-ml-sm"
|
||||
>Copy LNURL</q-btn
|
||||
>
|
||||
<q-btn
|
||||
outline
|
||||
color="grey"
|
||||
icon="link"
|
||||
@click="copyText(qrCodeDialog.data.withdraw_url, 'Link copied to clipboard!')"
|
||||
><q-tooltip>Copy sharable link</q-tooltip>
|
||||
</q-btn>
|
||||
<q-btn
|
||||
outline
|
||||
color="grey"
|
||||
icon="nfc"
|
||||
@click="writeNfcTag(lnurl)"
|
||||
@click="writeNfcTag(qrCodeDialog.data.lnurl)"
|
||||
:disable="nfcTagWriting"
|
||||
><q-tooltip>Write to NFC</q-tooltip></q-btn
|
||||
>
|
||||
<q-btn
|
||||
outline
|
||||
color="grey"
|
||||
icon="link"
|
||||
:href="'/withdraw/' + qrCodeDialog.data.id"
|
||||
target="_blank"
|
||||
><q-tooltip>Open sharable link</q-tooltip>
|
||||
</q-btn>
|
||||
<q-btn
|
||||
outline
|
||||
color="grey"
|
||||
icon="print"
|
||||
type="a"
|
||||
:href="'/withdraw/print/' + qrCodeDialog.data.id"
|
||||
:href="qrCodeDialog.data.print_url"
|
||||
target="_blank"
|
||||
><q-tooltip>Print</q-tooltip></q-btn
|
||||
>
|
||||
|
|
|
|||
|
|
@ -4,21 +4,23 @@
|
|||
<div class="" id="vue">
|
||||
{% for page in link %}
|
||||
<page size="A4" id="pdfprint">
|
||||
<div class="full-height content-center">
|
||||
{% for row in page %}
|
||||
<div class="row" style="max-height: 54mm">
|
||||
{% for one in row %}
|
||||
<div class="col-6">
|
||||
<table style="width: 100%">
|
||||
{% for threes in page %}
|
||||
<tr style="height: 59.4mm">
|
||||
{% for one in threes %}
|
||||
<td style="width: 105mm">
|
||||
<center>
|
||||
<lnbits-qrcode
|
||||
style="width: 50mm"
|
||||
style="width: fit-content"
|
||||
:value="theurl + '/?lightning={{one}}'"
|
||||
:show-buttons="false"
|
||||
:options="{width: 150}"
|
||||
></lnbits-qrcode>
|
||||
</div>
|
||||
</center>
|
||||
</td>
|
||||
{% endfor %}
|
||||
</div>
|
||||
</tr>
|
||||
{% endfor %}
|
||||
</div>
|
||||
</table>
|
||||
</page>
|
||||
{% endfor %}
|
||||
</div>
|
||||
|
|
|
|||
|
|
@ -11,8 +11,7 @@
|
|||
<div class="lnurlw">
|
||||
<lnbits-qrcode
|
||||
:value="theurl + '/?lightning={{one}}'"
|
||||
:show-buttons="false"
|
||||
:options="{width: 150}"
|
||||
:options="{width: 98, margin: 2, logo: false}"
|
||||
></lnbits-qrcode>
|
||||
</div>
|
||||
</div>
|
||||
|
|
@ -62,10 +61,9 @@
|
|||
.wrapper .lnurlw {
|
||||
display: block;
|
||||
position: absolute;
|
||||
top: calc(3mm + 1rem);
|
||||
left: calc(6mm + 1rem);
|
||||
top: calc(7.3mm + 1rem);
|
||||
left: calc(7.5mm + 1rem);
|
||||
transform: rotate(45deg);
|
||||
width: 27mm;
|
||||
}
|
||||
|
||||
@media print {
|
||||
|
|
@ -85,8 +83,8 @@
|
|||
.wrapper .lnurlw {
|
||||
display: block;
|
||||
position: absolute;
|
||||
top: 3mm;
|
||||
left: 6mm;
|
||||
top: 7.3mm;
|
||||
left: 7.5mm;
|
||||
transform: rotate(45deg);
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,225 +0,0 @@
|
|||
"""
|
||||
Nostr-transport RPC handlers for the withdraw (LNURL-withdraw) extension.
|
||||
|
||||
Names mirror the Lightning.Pub `withdraw.*` contract that the lamassu-next
|
||||
ATM consumes (see ~/dev/shocknet/lamassu-next/packages/lightning/src/client.ts
|
||||
lines ~301–351). That keeps the lamassu-next-side adapter a pure name
|
||||
translation — no semantic reshaping.
|
||||
|
||||
Auth model (set in `__init__.py:withdraw_start`):
|
||||
- create / get / update / delete → AUTH_WALLET; the calling pubkey must
|
||||
own the wallet the link is scoped to. *_get / *_update / *_delete also
|
||||
verify the link's stored `wallet` matches the caller's wallet id.
|
||||
|
||||
`resolve_withdraw_owner` is registered with the core subscription module
|
||||
under tag `"withdraw"` and extras-key `"withdrawal_link_id"` (matching
|
||||
where the extension stamps the link id on settlement — see
|
||||
`views_lnurl.py:144`). That lets `subscribe_payments({tag:"withdraw",
|
||||
link_id:...})` enforce ownership without core importing this module.
|
||||
"""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
from lnbits.core.crud.wallets import get_wallets
|
||||
from lnbits.core.models import Account
|
||||
from lnbits.core.models.wallets import WalletTypeInfo
|
||||
from lnbits.core.services.nostr_transport.models import NostrRpcRequest
|
||||
from shortuuid import uuid
|
||||
|
||||
from .crud import (
|
||||
create_withdraw_link,
|
||||
delete_withdraw_link,
|
||||
get_withdraw_link,
|
||||
get_withdraw_links,
|
||||
update_withdraw_link,
|
||||
)
|
||||
from .helpers import create_lnurl_from_baseurl
|
||||
from .models import CreateWithdrawData, WithdrawLink
|
||||
|
||||
|
||||
async def handle_lnurlw_create_link(
|
||||
auth: WalletTypeInfo, request: NostrRpcRequest
|
||||
) -> dict:
|
||||
body = request.body or {}
|
||||
data = CreateWithdrawData(**body)
|
||||
link = await create_withdraw_link(data, auth.wallet.id)
|
||||
return _to_dict(_populate_lnurl(link))
|
||||
|
||||
|
||||
async def handle_lnurlw_get_link(
|
||||
auth: WalletTypeInfo, request: NostrRpcRequest
|
||||
) -> dict:
|
||||
link_id = _require_id(request)
|
||||
link = await _require_owned_link(link_id, auth.wallet.id)
|
||||
return _to_dict(_populate_lnurl(link))
|
||||
|
||||
|
||||
async def handle_lnurlw_update_link(
|
||||
auth: WalletTypeInfo, request: NostrRpcRequest
|
||||
) -> dict:
|
||||
link_id = _require_id(request)
|
||||
link = await _require_owned_link(link_id, auth.wallet.id)
|
||||
body = request.body or {}
|
||||
_MUTABLE = {
|
||||
"title",
|
||||
"min_withdrawable",
|
||||
"max_withdrawable",
|
||||
"uses",
|
||||
"wait_time",
|
||||
"is_unique",
|
||||
"webhook_url",
|
||||
"webhook_headers",
|
||||
"webhook_body",
|
||||
"custom_url",
|
||||
"enabled",
|
||||
}
|
||||
for k, v in body.items():
|
||||
if k in _MUTABLE:
|
||||
setattr(link, k, v)
|
||||
updated = await update_withdraw_link(link)
|
||||
return _to_dict(_populate_lnurl(updated))
|
||||
|
||||
|
||||
async def handle_lnurlw_delete_link(
|
||||
auth: WalletTypeInfo, request: NostrRpcRequest
|
||||
) -> dict:
|
||||
link_id = _require_id(request)
|
||||
await _require_owned_link(link_id, auth.wallet.id)
|
||||
await delete_withdraw_link(link_id)
|
||||
return {"ok": True}
|
||||
|
||||
|
||||
async def handle_lnurlw_list_links(auth: Account, request: NostrRpcRequest) -> dict:
|
||||
"""List withdraw links across all wallets owned by the calling account.
|
||||
Useful for ATMs to re-discover their links after a reconnect.
|
||||
|
||||
Body fields:
|
||||
- limit: int (0 means no limit; default 0)
|
||||
- offset: int (default 0)
|
||||
If `request.wallet_id` is set and is one of the caller's wallets,
|
||||
narrow to just that wallet.
|
||||
"""
|
||||
body = request.body or {}
|
||||
limit = int(body.get("limit") or 0)
|
||||
offset = int(body.get("offset") or 0)
|
||||
|
||||
wallets = await get_wallets(auth.id)
|
||||
wallet_ids = [w.id for w in wallets]
|
||||
if not wallet_ids:
|
||||
return {"data": [], "total": 0}
|
||||
if request.wallet_id and request.wallet_id in wallet_ids:
|
||||
wallet_ids = [request.wallet_id]
|
||||
|
||||
page = await get_withdraw_links(wallet_ids, limit, offset)
|
||||
return {
|
||||
"data": [_to_dict(_populate_lnurl(link)) for link in page.data],
|
||||
"total": page.total,
|
||||
}
|
||||
|
||||
|
||||
async def handle_lnurlw_unique_hashes(
|
||||
auth: WalletTypeInfo, request: NostrRpcRequest
|
||||
) -> dict:
|
||||
"""
|
||||
For a `is_unique=True` link, return the per-use `id_unique_hash`
|
||||
values that the ATM uses to generate distinct QR codes — one per
|
||||
unredeemed slot. Mirrors the formula in `helpers.py:create_lnurl`
|
||||
exactly so an ATM never has to re-implement the derivation:
|
||||
|
||||
id_unique_hash = shortuuid.uuid(name=link.id + link.unique_hash + index)
|
||||
|
||||
`link.usescsv` is the canonical list of *unredeemed* slot indexes;
|
||||
after a customer claims a slot it gets removed there (see
|
||||
`crud.remove_unique_withdraw_link`). The hashes returned here are
|
||||
therefore exactly the ones still claimable.
|
||||
|
||||
Response:
|
||||
{
|
||||
"link_id": str,
|
||||
"unique_hash": str, # base hash
|
||||
"is_unique": bool,
|
||||
"unredeemed_hashes": [ # one entry per remaining slot
|
||||
{"index": str, "id_unique_hash": str}, ...
|
||||
]
|
||||
}
|
||||
|
||||
For `is_unique=False` links the list is empty and `unique_hash`
|
||||
alone identifies the callback path
|
||||
(`/withdraw/api/v1/lnurl/<unique_hash>`). For `is_unique=True`
|
||||
each callback path is
|
||||
`/withdraw/api/v1/lnurl/<unique_hash>/<id_unique_hash>`.
|
||||
"""
|
||||
link_id = _require_id(request)
|
||||
link = await _require_owned_link(link_id, auth.wallet.id)
|
||||
|
||||
unredeemed = []
|
||||
if link.is_unique:
|
||||
# usescsv is comma-separated; split and skip empties (after the
|
||||
# last slot is consumed it becomes the empty string).
|
||||
for index_str in [s for s in link.usescsv.split(",") if s.strip()]:
|
||||
tohash = link.id + link.unique_hash + index_str
|
||||
unredeemed.append(
|
||||
{
|
||||
"index": index_str.strip(),
|
||||
"id_unique_hash": uuid(name=tohash),
|
||||
}
|
||||
)
|
||||
|
||||
return {
|
||||
"link_id": link.id,
|
||||
"unique_hash": link.unique_hash,
|
||||
"is_unique": link.is_unique,
|
||||
"unredeemed_hashes": unredeemed,
|
||||
}
|
||||
|
||||
|
||||
async def resolve_withdraw_owner(link_id: str) -> str | None:
|
||||
"""For the core subscription module: link_id -> wallet_id (or None)."""
|
||||
link = await get_withdraw_link(link_id)
|
||||
return link.wallet if link else None
|
||||
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
# helpers
|
||||
# ---------------------------------------------------------------------------
|
||||
|
||||
|
||||
def _require_id(request: NostrRpcRequest) -> str:
|
||||
body = request.body or {}
|
||||
link_id = body.get("id")
|
||||
if not link_id:
|
||||
raise ValueError("withdraw: body.id is required")
|
||||
return str(link_id)
|
||||
|
||||
|
||||
async def _require_owned_link(link_id: str, wallet_id: str):
|
||||
link = await get_withdraw_link(link_id)
|
||||
if link is None:
|
||||
raise ValueError(f"withdraw: link not found: {link_id}")
|
||||
if link.wallet != wallet_id:
|
||||
raise PermissionError("withdraw: link does not belong to caller's wallet")
|
||||
return link
|
||||
|
||||
|
||||
def _populate_lnurl(link: WithdrawLink) -> WithdrawLink:
|
||||
"""
|
||||
Compose `lnurl` / `lnurl_url` from `settings.lnbits_baseurl` so
|
||||
nostr-transport responses match the HTTP `views_api` shape, where
|
||||
these fields are populated from `request.url_for(...)`. Without
|
||||
this, consumers (ATMs, etc.) would have to re-derive the callback
|
||||
URL themselves from a separately-provisioned LNbits HTTPS URL —
|
||||
duplicating state LNbits already knows. See aiolabs/withdraw#1.
|
||||
"""
|
||||
try:
|
||||
encoded = create_lnurl_from_baseurl(link)
|
||||
link.lnurl = str(encoded.bech32)
|
||||
link.lnurl_url = str(encoded.url)
|
||||
except ValueError:
|
||||
pass
|
||||
return link
|
||||
|
||||
|
||||
def _to_dict(link) -> dict:
|
||||
import json
|
||||
|
||||
return json.loads(link.json())
|
||||
77
views.py
77
views.py
|
|
@ -1,6 +1,7 @@
|
|||
import io
|
||||
from http import HTTPStatus
|
||||
from io import BytesIO
|
||||
|
||||
import pyqrcode
|
||||
from fastapi import APIRouter, Depends, HTTPException, Request
|
||||
from fastapi.responses import HTMLResponse, StreamingResponse
|
||||
from lnbits.core.models import User
|
||||
|
|
@ -45,9 +46,43 @@ async def display(request: Request, link_id):
|
|||
"withdraw/display.html",
|
||||
{
|
||||
"request": request,
|
||||
"spent": link.is_spent,
|
||||
"lnurl_url": str(lnurl.url),
|
||||
"enabled": link.enabled,
|
||||
"link": link.json(),
|
||||
"lnurl": lnurl,
|
||||
"unique": True,
|
||||
},
|
||||
)
|
||||
|
||||
|
||||
@withdraw_ext_generic.get("/img/{link_id}", response_class=StreamingResponse)
|
||||
async def img(request: Request, link_id):
|
||||
link = await get_withdraw_link(link_id, 0)
|
||||
if not link:
|
||||
raise HTTPException(
|
||||
status_code=HTTPStatus.NOT_FOUND, detail="Withdraw link does not exist."
|
||||
)
|
||||
try:
|
||||
lnurl = create_lnurl(link, request)
|
||||
except ValueError as exc:
|
||||
raise HTTPException(
|
||||
status_code=HTTPStatus.INTERNAL_SERVER_ERROR,
|
||||
detail=str(exc),
|
||||
) from exc
|
||||
|
||||
qr = pyqrcode.create(lnurl)
|
||||
stream = BytesIO()
|
||||
qr.svg(stream, scale=3)
|
||||
stream.seek(0)
|
||||
|
||||
async def _generator(stream: BytesIO):
|
||||
yield stream.getvalue()
|
||||
|
||||
return StreamingResponse(
|
||||
_generator(stream),
|
||||
headers={
|
||||
"Content-Type": "image/svg+xml",
|
||||
"Cache-Control": "no-cache, no-store, must-revalidate",
|
||||
"Pragma": "no-cache",
|
||||
"Expires": "0",
|
||||
},
|
||||
)
|
||||
|
||||
|
|
@ -59,8 +94,11 @@ async def print_qr(request: Request, link_id):
|
|||
raise HTTPException(
|
||||
status_code=HTTPStatus.NOT_FOUND, detail="Withdraw link does not exist."
|
||||
)
|
||||
# response.status_code = HTTPStatus.NOT_FOUND
|
||||
# return "Withdraw link does not exist."
|
||||
|
||||
if link.uses == 0:
|
||||
|
||||
return withdraw_renderer().TemplateResponse(
|
||||
"withdraw/print_qr.html",
|
||||
{"request": request, "link": link.json(), "unique": False},
|
||||
|
|
@ -81,7 +119,7 @@ async def print_qr(request: Request, link_id):
|
|||
status_code=HTTPStatus.INTERNAL_SERVER_ERROR,
|
||||
detail=str(exc),
|
||||
) from exc
|
||||
links.append(str(lnurl.bech32))
|
||||
links.append(str(lnurl))
|
||||
count = count + 1
|
||||
page_link = list(chunks(links, 2))
|
||||
linked = list(chunks(page_link, 5))
|
||||
|
|
@ -110,14 +148,18 @@ async def csv(request: Request, link_id):
|
|||
raise HTTPException(
|
||||
status_code=HTTPStatus.NOT_FOUND, detail="Withdraw link does not exist."
|
||||
)
|
||||
# response.status_code = HTTPStatus.NOT_FOUND
|
||||
# return "Withdraw link does not exist."
|
||||
|
||||
if link.uses == 0:
|
||||
raise HTTPException(
|
||||
status_code=HTTPStatus.BAD_REQUEST, detail="Withdraw is spent."
|
||||
)
|
||||
|
||||
buffer = io.StringIO()
|
||||
return withdraw_renderer().TemplateResponse(
|
||||
"withdraw/csv.html",
|
||||
{"request": request, "link": link.json(), "unique": False},
|
||||
)
|
||||
links = []
|
||||
count = 0
|
||||
|
||||
for _ in link.usescsv.split(","):
|
||||
linkk = await get_withdraw_link(link_id, count)
|
||||
if not linkk:
|
||||
|
|
@ -131,16 +173,11 @@ async def csv(request: Request, link_id):
|
|||
status_code=HTTPStatus.INTERNAL_SERVER_ERROR,
|
||||
detail=str(exc),
|
||||
) from exc
|
||||
buffer.write(f"{lnurl.bech32!s}\n")
|
||||
count += 1
|
||||
links.append(str(lnurl))
|
||||
count = count + 1
|
||||
page_link = list(chunks(links, 2))
|
||||
linked = list(chunks(page_link, 5))
|
||||
|
||||
# Move buffer cursor to the beginning
|
||||
buffer.seek(0)
|
||||
|
||||
return StreamingResponse(
|
||||
buffer,
|
||||
media_type="text/csv",
|
||||
headers={
|
||||
"Content-Disposition": f"attachment; filename=withdraw-links-{link_id}.csv"
|
||||
},
|
||||
return withdraw_renderer().TemplateResponse(
|
||||
"withdraw/csv.html", {"request": request, "link": linked, "unique": True}
|
||||
)
|
||||
|
|
|
|||
49
views_api.py
49
views_api.py
|
|
@ -1,9 +1,10 @@
|
|||
import json
|
||||
from http import HTTPStatus
|
||||
from typing import Optional
|
||||
|
||||
from fastapi import APIRouter, Depends, HTTPException, Query, Request
|
||||
from lnbits.core.crud import get_user
|
||||
from lnbits.core.models import SimpleStatus, WalletTypeInfo
|
||||
from lnbits.core.models import WalletTypeInfo
|
||||
from lnbits.decorators import require_admin_key, require_invoice_key
|
||||
|
||||
from .crud import (
|
||||
|
|
@ -15,7 +16,7 @@ from .crud import (
|
|||
update_withdraw_link,
|
||||
)
|
||||
from .helpers import create_lnurl
|
||||
from .models import CreateWithdrawData, HashCheck, PaginatedWithdraws, WithdrawLink
|
||||
from .models import CreateWithdrawData, HashCheck
|
||||
|
||||
withdraw_ext_api = APIRouter(prefix="/api/v1")
|
||||
|
||||
|
|
@ -27,35 +28,38 @@ async def api_links(
|
|||
all_wallets: bool = Query(False),
|
||||
offset: int = Query(0),
|
||||
limit: int = Query(0),
|
||||
) -> PaginatedWithdraws:
|
||||
):
|
||||
wallet_ids = [key_info.wallet.id]
|
||||
|
||||
if all_wallets:
|
||||
user = await get_user(key_info.wallet.user)
|
||||
wallet_ids = user.wallet_ids if user else []
|
||||
|
||||
links = await get_withdraw_links(wallet_ids, limit, offset)
|
||||
links, total = await get_withdraw_links(wallet_ids, limit, offset)
|
||||
|
||||
for linkk in links.data:
|
||||
data = []
|
||||
for link in links:
|
||||
try:
|
||||
lnurl = create_lnurl(linkk, request)
|
||||
lnurl = create_lnurl(link, request)
|
||||
except ValueError as exc:
|
||||
raise HTTPException(
|
||||
status_code=HTTPStatus.INTERNAL_SERVER_ERROR,
|
||||
detail=str(exc),
|
||||
) from exc
|
||||
linkk.lnurl = str(lnurl.bech32)
|
||||
linkk.lnurl_url = str(lnurl.url)
|
||||
data.append({**link.dict(), **{"lnurl": lnurl}})
|
||||
|
||||
return links
|
||||
return {
|
||||
"data": data,
|
||||
"total": total,
|
||||
}
|
||||
|
||||
|
||||
@withdraw_ext_api.get("/links/{link_id}", status_code=HTTPStatus.OK)
|
||||
async def api_link_retrieve(
|
||||
request: Request,
|
||||
link_id: str,
|
||||
request: Request,
|
||||
key_info: WalletTypeInfo = Depends(require_invoice_key),
|
||||
) -> WithdrawLink:
|
||||
):
|
||||
link = await get_withdraw_link(link_id, 0)
|
||||
|
||||
if not link:
|
||||
|
|
@ -67,7 +71,6 @@ async def api_link_retrieve(
|
|||
raise HTTPException(
|
||||
detail="Not your withdraw link.", status_code=HTTPStatus.FORBIDDEN
|
||||
)
|
||||
|
||||
try:
|
||||
lnurl = create_lnurl(link, request)
|
||||
except ValueError as exc:
|
||||
|
|
@ -75,19 +78,17 @@ async def api_link_retrieve(
|
|||
status_code=HTTPStatus.INTERNAL_SERVER_ERROR,
|
||||
detail=str(exc),
|
||||
) from exc
|
||||
link.lnurl = str(lnurl.bech32)
|
||||
link.lnurl_url = str(lnurl.url)
|
||||
return link
|
||||
return {**link.dict(), **{"lnurl": lnurl}}
|
||||
|
||||
|
||||
@withdraw_ext_api.post("/links", status_code=HTTPStatus.CREATED)
|
||||
@withdraw_ext_api.put("/links/{link_id}")
|
||||
@withdraw_ext_api.put("/links/{link_id}", status_code=HTTPStatus.OK)
|
||||
async def api_link_create_or_update(
|
||||
request: Request,
|
||||
data: CreateWithdrawData,
|
||||
link_id: str | None = None,
|
||||
link_id: Optional[str] = None,
|
||||
key_info: WalletTypeInfo = Depends(require_admin_key),
|
||||
) -> WithdrawLink:
|
||||
):
|
||||
if data.uses > 250:
|
||||
raise HTTPException(detail="250 uses max.", status_code=HTTPStatus.BAD_REQUEST)
|
||||
|
||||
|
|
@ -159,6 +160,7 @@ async def api_link_create_or_update(
|
|||
link = await update_withdraw_link(link)
|
||||
else:
|
||||
link = await create_withdraw_link(wallet_id=key_info.wallet.id, data=data)
|
||||
|
||||
try:
|
||||
lnurl = create_lnurl(link, request)
|
||||
except ValueError as exc:
|
||||
|
|
@ -167,16 +169,13 @@ async def api_link_create_or_update(
|
|||
detail=str(exc),
|
||||
) from exc
|
||||
|
||||
link.lnurl = str(lnurl.bech32)
|
||||
link.lnurl_url = str(lnurl.url)
|
||||
|
||||
return link
|
||||
return {**link.dict(), **{"lnurl": lnurl}}
|
||||
|
||||
|
||||
@withdraw_ext_api.delete("/links/{link_id}")
|
||||
@withdraw_ext_api.delete("/links/{link_id}", status_code=HTTPStatus.OK)
|
||||
async def api_link_delete(
|
||||
link_id: str, key_info: WalletTypeInfo = Depends(require_admin_key)
|
||||
) -> SimpleStatus:
|
||||
):
|
||||
link = await get_withdraw_link(link_id)
|
||||
|
||||
if not link:
|
||||
|
|
@ -190,7 +189,7 @@ async def api_link_delete(
|
|||
)
|
||||
|
||||
await delete_withdraw_link(link_id)
|
||||
return SimpleStatus(success=True, message="Withdraw link deleted.")
|
||||
return {"success": True}
|
||||
|
||||
|
||||
@withdraw_ext_api.get(
|
||||
|
|
|
|||
|
|
@ -1,9 +1,9 @@
|
|||
import json
|
||||
from datetime import datetime
|
||||
from typing import Optional
|
||||
|
||||
import httpx
|
||||
import shortuuid
|
||||
from bolt11 import decode as decode_bolt11
|
||||
from fastapi import APIRouter, Request
|
||||
from fastapi.responses import JSONResponse
|
||||
from lnbits.core.crud import update_payment
|
||||
|
|
@ -44,9 +44,6 @@ async def api_lnurl_response(
|
|||
if not link:
|
||||
return LnurlErrorResponse(reason="Withdraw link does not exist.")
|
||||
|
||||
if not link.enabled:
|
||||
return LnurlErrorResponse(reason="Withdraw link is disabled.")
|
||||
|
||||
if link.is_spent:
|
||||
return LnurlErrorResponse(reason="Withdraw is spent.")
|
||||
|
||||
|
|
@ -88,25 +85,13 @@ async def api_lnurl_callback(
|
|||
unique_hash: str,
|
||||
k1: str,
|
||||
pr: str,
|
||||
id_unique_hash: str | None = None,
|
||||
id_unique_hash: Optional[str] = None,
|
||||
) -> LnurlErrorResponse | LnurlSuccessResponse:
|
||||
|
||||
link = await get_withdraw_link_by_hash(unique_hash)
|
||||
if not link:
|
||||
return LnurlErrorResponse(reason="withdraw link not found.")
|
||||
|
||||
if not link.enabled:
|
||||
return LnurlErrorResponse(reason="Withdraw link is disabled.")
|
||||
|
||||
bolt11 = decode_bolt11(pr)
|
||||
if not bolt11.amount_msat:
|
||||
return LnurlErrorResponse(reason="0 amount invoices are not supported.")
|
||||
|
||||
if (
|
||||
link.min_withdrawable * 1000 > bolt11.amount_msat
|
||||
or bolt11.amount_msat > link.max_withdrawable * 1000
|
||||
):
|
||||
return LnurlErrorResponse(reason="Amount not within limits.")
|
||||
|
||||
if link.is_spent:
|
||||
return LnurlErrorResponse(reason="withdraw is spent.")
|
||||
|
||||
|
|
@ -115,9 +100,9 @@ async def api_lnurl_callback(
|
|||
|
||||
now = int(datetime.now().timestamp())
|
||||
|
||||
if now < link.open_time + link.wait_time:
|
||||
if now < link.open_time:
|
||||
return LnurlErrorResponse(
|
||||
reason=f"Wait {link.open_time + link.wait_time - now} seconds."
|
||||
reason=f"wait link open_time {link.open_time - now} seconds."
|
||||
)
|
||||
|
||||
if not id_unique_hash and link.is_unique:
|
||||
|
|
@ -141,16 +126,7 @@ async def api_lnurl_callback(
|
|||
wallet_id=link.wallet,
|
||||
payment_request=pr,
|
||||
max_sat=link.max_withdrawable,
|
||||
# Merge the link's caller-supplied `extra` onto the payout so an
|
||||
# external listener can key on it (e.g. bitSpire cash-in
|
||||
# settlements via spirekeeper). The withdraw extension's own
|
||||
# `tag`/`withdrawal_link_id` are written last so a caller cannot
|
||||
# clobber them.
|
||||
extra={
|
||||
**(link.extra or {}),
|
||||
"tag": "withdraw",
|
||||
"withdrawal_link_id": link.id,
|
||||
},
|
||||
extra={"tag": "withdraw", "withdrawal_link_id": link.id},
|
||||
)
|
||||
await increment_withdraw_link(link)
|
||||
# If the payment succeeds, delete the record with the unique_hash.
|
||||
|
|
@ -219,9 +195,6 @@ async def api_lnurl_multi_response(
|
|||
if not link:
|
||||
return LnurlErrorResponse(reason="Withdraw link does not exist.")
|
||||
|
||||
if not link.enabled:
|
||||
return LnurlErrorResponse(reason="Withdraw link is disabled.")
|
||||
|
||||
if link.is_spent:
|
||||
return LnurlErrorResponse(reason="Withdraw is spent.")
|
||||
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue