-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathviews_api.py
187 lines (154 loc) · 5.73 KB
/
views_api.py
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
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
from http import HTTPStatus
from fastapi import APIRouter, Depends, HTTPException, Query
from lnbits.core.crud import get_standalone_payment, get_user
from lnbits.core.models import Payment, WalletTypeInfo
from lnbits.core.services import create_invoice
from lnbits.decorators import require_admin_key
from lnbits.utils.exchange_rates import fiat_amount_as_satoshis
from loguru import logger
from pydantic import BaseModel
from .crud import (
create_invoice_internal,
create_invoice_items,
delete_invoice,
delete_invoice_items,
get_invoice,
get_invoice_items,
get_invoice_payments,
get_invoice_total,
get_invoices,
get_payments_total,
update_invoice_internal,
)
from .models import (
CreateInvoiceData,
Invoice,
InvoiceFull,
UpdateInvoiceData,
)
invoices_api_router = APIRouter()
@invoices_api_router.get("/api/v1/invoices")
async def api_invoices(
all_wallets: bool = Query(None),
key_info: WalletTypeInfo = Depends(require_admin_key),
) -> list[Invoice]:
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 []
invoices = await get_invoices(wallet_ids)
return invoices
@invoices_api_router.get("/api/v1/invoice/{invoice_id}")
async def api_invoice(invoice_id: str) -> InvoiceFull:
invoice = await get_invoice(invoice_id)
if not invoice:
raise HTTPException(
status_code=HTTPStatus.NOT_FOUND, detail="Invoice does not exist."
)
invoice_items = await get_invoice_items(invoice_id)
invoice_payments = await get_invoice_payments(invoice_id)
payments_total = await get_payments_total(invoice_payments)
invoice = InvoiceFull(
**invoice.dict(), items=invoice_items, payments=payments_total
)
return invoice
@invoices_api_router.post("/api/v1/invoice", status_code=HTTPStatus.CREATED)
async def api_invoice_create(
data: CreateInvoiceData, key_info: WalletTypeInfo = Depends(require_admin_key)
) -> Invoice:
invoice = await create_invoice_internal(wallet_id=key_info.wallet.id, data=data)
await create_invoice_items(invoice_id=invoice.id, data=data.items)
return invoice
@invoices_api_router.delete(
"/api/v1/invoice/{invoice_id}", dependencies=[Depends(require_admin_key)]
)
async def api_invoice_delete(invoice_id: str):
try:
status = await delete_invoice(invoice_id=invoice_id)
return {"status": status}
except Exception as exc:
raise HTTPException(
status_code=HTTPStatus.INTERNAL_SERVER_ERROR, detail=str(exc)
) from exc
@invoices_api_router.put("/api/v1/invoice/{invoice_id}")
async def api_invoice_update(
data: UpdateInvoiceData,
invoice_id: str,
key_info: WalletTypeInfo = Depends(require_admin_key),
) -> Invoice:
invoice = await get_invoice(invoice_id)
if not invoice:
raise HTTPException(
status_code=HTTPStatus.NOT_FOUND, detail="Invoice does not exist."
)
if invoice.wallet != key_info.wallet.id:
raise HTTPException(
status_code=HTTPStatus.FORBIDDEN,
detail="You do not have permission to update this invoice.",
)
for key, value in data.dict().items():
if key == "items":
continue
setattr(invoice, key, value)
await delete_invoice_items(invoice_id=invoice_id)
await create_invoice_items(invoice_id=invoice_id, data=data.items)
await update_invoice_internal(invoice)
return invoice
class InvoiceAmountPayment(BaseModel):
famount: int = Query(..., ge=1, description="Amount to pay in fiat currency.")
@invoices_api_router.post(
"/api/v1/invoice/{invoice_id}/payments", status_code=HTTPStatus.CREATED
)
async def api_invoices_create_payment(
invoice_id: str, data: InvoiceAmountPayment
) -> Payment:
invoice = await get_invoice(invoice_id)
invoice_items = await get_invoice_items(invoice_id)
invoice_total = await get_invoice_total(invoice_items)
invoice_payments = await get_invoice_payments(invoice_id)
payments_total = await get_payments_total(invoice_payments)
if payments_total + data.famount > invoice_total:
raise HTTPException(
status_code=HTTPStatus.BAD_REQUEST, detail="Amount exceeds invoice due."
)
if not invoice:
raise HTTPException(
status_code=HTTPStatus.NOT_FOUND, detail="Invoice does not exist."
)
price_in_sats = await fiat_amount_as_satoshis(data.famount / 100, invoice.currency)
try:
payment = await create_invoice(
wallet_id=invoice.wallet,
amount=price_in_sats,
memo=f"Payment for invoice {invoice_id}",
extra={
"tag": "invoices",
"invoice_id": invoice_id,
"famount": data.famount,
},
)
except Exception as exc:
raise HTTPException(
status_code=HTTPStatus.INTERNAL_SERVER_ERROR, detail=str(exc)
) from exc
return payment
@invoices_api_router.get(
"/api/v1/invoice/{invoice_id}/payments/{payment_hash}",
dependencies=[Depends(require_admin_key)],
)
async def api_invoices_check_payment(invoice_id: str, payment_hash: str):
invoice = await get_invoice(invoice_id)
if not invoice:
raise HTTPException(
status_code=HTTPStatus.NOT_FOUND, detail="Invoice does not exist."
)
try:
payment = await get_standalone_payment(payment_hash, incoming=True)
if not payment:
raise HTTPException(
status_code=HTTPStatus.NOT_FOUND, detail="Payment does not exist."
)
return {"paid": payment.success}
except Exception as exc:
logger.error(exc)
return {"paid": False}