2022-08-01 16:12:25 +02:00
|
|
|
import hashlib
|
|
|
|
|
2022-08-03 14:10:32 +02:00
|
|
|
import pytest
|
|
|
|
import pytest_asyncio
|
|
|
|
|
2022-08-01 16:12:25 +02:00
|
|
|
from lnbits import bolt11
|
2022-06-27 00:11:46 +02:00
|
|
|
from lnbits.core.crud import get_wallet
|
2022-08-03 14:10:32 +02:00
|
|
|
from lnbits.core.views.api import (
|
|
|
|
CreateInvoiceData,
|
|
|
|
api_payment,
|
|
|
|
api_payments_create_invoice,
|
|
|
|
)
|
2022-10-05 11:15:11 +02:00
|
|
|
from lnbits.settings import get_wallet_class
|
2022-08-01 16:12:25 +02:00
|
|
|
|
Boltz.exchange Extension (#922)
* initial commit and still draft, ready for review
* forgot to uncomment this line
* fee estimation and blockheight
* resolve conversation with michael, to use mempool websockets instead of boltz status event
* Update lnbits/extensions/boltz/boltz.py
Co-authored-by: michael1011 <me@michael1011.at>
* add status to swaps, add sorting and data into listing
* add swap status checks, change urls to docker test setup, dynamic minimum and maximum limits
* fix docker hosts for development
* add api endpoints to _api_docs
* add wallet name and id, to list and status information
* fix status_update for reverse_swaps
* chore: format with black
* more blackformatting and refactoring create_swap()
* fix variable bug
* check if swap is already refunded
* use create_task instead of ensure_future
* add mempool and boltz urls depending on DEBUG .env
* raise exception in mempool fails
* fix onchain txs, sending funds to wrong address and add a refund address for normal swaps beforehand
* add status to swaps, add sorting and data into listing
* add swap status checks, change urls to docker test setup, dynamic minimum and maximum limits
* add wallet name and id, to list and status information
* fix status_update for reverse_swaps
* chore: format with black
* use create_task instead of ensure_future
* add mempool and boltz urls depending on DEBUG .env
* fix onchain txs, sending funds to wrong address and add a refund address for normal swaps beforehand
* black formatting
* add some logging with loguru, and remove function duplication
* cleanup readme
* updates/suggestions from calle
Co-authored-by: calle <93376500+callebtc@users.noreply.github.com>
* remove unused comments
* Update API Endpoints
Co-authored-by: calle <93376500+callebtc@users.noreply.github.com>
* un-factor get_boltz_pairs
* added a explaination for the onchain tx
* remove unused template file
* rename api endpoints
* fix isort and prettier
* more verbose logging!!
* add boltz to mock_data.zip
* new mockdata
* remove comment
* better readme
* fix mempool urls
* change /refund /check /status to post requests
* first step in tests2
* add first tests
* change refund,check,status to post requests
* next try on tests
* overall code improvements
* just testing tests
* throw http exceptions in views_api
* require admincheck for refund,check,status and added fastapi documentation for those
* added more tests
* black
* many code improvements
* adding tests
* temp fix test
* fix race condition when pay_invoice fails
* test are working
* add boltz env variables
* add startup check, bugfixes, improvements
* improve on status checking
* remove check_invoice_status
* more fixes and tests
* testing testing testing
* make tests run again inside regtest
* fix bad error :O
* fix postgres boolean bug and add swap test
* Update README.md
Update README.md
Update README.md
Update README.md
* some mypy
* blacked
* the missing commit?
* fix api_docs readme link
* better refunding error catching
fix
* check swaps now also shows pending reverse swap, ui improvements, tooltips
* add backend check for boltz limits
fixup
* many improvements, startup check for swaps working, reverse needs more testing
* little last fixes
* remove unused logic
* fastapi documentation
fixup
* formatting and remove unused tests
* fix test
* fix swapstatus model
* Update lnbits/extensions/boltz/tasks.py
Co-authored-by: calle <93376500+callebtc@users.noreply.github.com>
* Update lnbits/extensions/boltz/views_api.py
Co-authored-by: calle <93376500+callebtc@users.noreply.github.com>
* balance check msg, format
* fix mypy data override
* fix swapstatus, remove can refund column
* Update lnbits/extensions/boltz/README.md
Co-authored-by: michael1011 <me@michael1011.at>
* empty lines
* fix error message when swap is not found
* remove preimage_hash from database
* fix api_docs html
fix api_docs html
* catch boltz network exceptions better
* formatting
* check for timeout on swap at get request
Co-authored-by: michael1011 <me@michael1011.at>
Co-authored-by: fusion44 <some.fusion@gmail.com>
Co-authored-by: calle <93376500+callebtc@users.noreply.github.com>
2022-08-30 12:51:17 +02:00
|
|
|
from ...helpers import get_random_invoice_data, is_regtest
|
2022-06-27 00:11:46 +02:00
|
|
|
|
2022-10-05 11:15:11 +02:00
|
|
|
WALLET = get_wallet_class()
|
2022-08-03 14:10:32 +02:00
|
|
|
|
2023-01-21 15:32:25 +00:00
|
|
|
|
2022-06-27 00:11:46 +02:00
|
|
|
# check if the client is working
|
|
|
|
@pytest.mark.asyncio
|
|
|
|
async def test_core_views_generic(client):
|
|
|
|
response = await client.get("/")
|
|
|
|
assert response.status_code == 200
|
|
|
|
|
|
|
|
|
2022-07-23 10:39:58 +02:00
|
|
|
# check GET /api/v1/wallet with inkey: wallet info, no balance
|
2022-06-27 00:11:46 +02:00
|
|
|
@pytest.mark.asyncio
|
2022-07-23 10:39:58 +02:00
|
|
|
async def test_get_wallet_inkey(client, inkey_headers_to):
|
2022-06-27 00:11:46 +02:00
|
|
|
response = await client.get("/api/v1/wallet", headers=inkey_headers_to)
|
2022-07-23 10:39:58 +02:00
|
|
|
assert response.status_code == 200
|
|
|
|
result = response.json()
|
|
|
|
assert "name" in result
|
|
|
|
assert "balance" in result
|
|
|
|
assert "id" not in result
|
|
|
|
|
|
|
|
|
|
|
|
# check GET /api/v1/wallet with adminkey: wallet info with balance
|
|
|
|
@pytest.mark.asyncio
|
|
|
|
async def test_get_wallet_adminkey(client, adminkey_headers_to):
|
|
|
|
response = await client.get("/api/v1/wallet", headers=adminkey_headers_to)
|
|
|
|
assert response.status_code == 200
|
|
|
|
result = response.json()
|
|
|
|
assert "name" in result
|
|
|
|
assert "balance" in result
|
|
|
|
assert "id" in result
|
2022-06-27 00:11:46 +02:00
|
|
|
|
|
|
|
|
2022-08-16 17:01:05 +02:00
|
|
|
# check PUT /api/v1/wallet/newwallet: empty request where admin key is needed
|
2022-08-13 14:47:29 +02:00
|
|
|
@pytest.mark.asyncio
|
2022-08-16 17:01:05 +02:00
|
|
|
async def test_put_empty_request_expected_admin_keys(client):
|
|
|
|
response = await client.put("/api/v1/wallet/newwallet")
|
|
|
|
assert response.status_code == 401
|
|
|
|
|
|
|
|
|
|
|
|
# check POST /api/v1/payments: empty request where invoice key is needed
|
|
|
|
@pytest.mark.asyncio
|
|
|
|
async def test_post_empty_request_expected_invoice_keys(client):
|
2022-08-13 14:47:29 +02:00
|
|
|
response = await client.post("/api/v1/payments")
|
|
|
|
assert response.status_code == 401
|
|
|
|
|
|
|
|
|
2022-06-27 00:11:46 +02:00
|
|
|
# check POST /api/v1/payments: invoice creation
|
|
|
|
@pytest.mark.asyncio
|
|
|
|
async def test_create_invoice(client, inkey_headers_to):
|
|
|
|
data = await get_random_invoice_data()
|
|
|
|
response = await client.post(
|
|
|
|
"/api/v1/payments", json=data, headers=inkey_headers_to
|
|
|
|
)
|
2022-07-17 14:34:25 +02:00
|
|
|
assert response.status_code == 201
|
|
|
|
invoice = response.json()
|
|
|
|
assert "payment_hash" in invoice
|
|
|
|
assert len(invoice["payment_hash"]) == 64
|
|
|
|
assert "payment_request" in invoice
|
|
|
|
assert "checking_id" in invoice
|
|
|
|
assert len(invoice["checking_id"])
|
|
|
|
return invoice
|
|
|
|
|
|
|
|
|
|
|
|
# check POST /api/v1/payments: invoice creation for internal payments only
|
|
|
|
@pytest.mark.asyncio
|
|
|
|
async def test_create_internal_invoice(client, inkey_headers_to):
|
|
|
|
data = await get_random_invoice_data()
|
|
|
|
data["internal"] = True
|
|
|
|
response = await client.post(
|
|
|
|
"/api/v1/payments", json=data, headers=inkey_headers_to
|
|
|
|
)
|
|
|
|
invoice = response.json()
|
|
|
|
assert response.status_code == 201
|
|
|
|
assert "payment_hash" in invoice
|
|
|
|
assert len(invoice["payment_hash"]) == 64
|
|
|
|
assert "payment_request" in invoice
|
|
|
|
assert "checking_id" in invoice
|
|
|
|
assert len(invoice["checking_id"])
|
|
|
|
return invoice
|
2022-06-27 00:11:46 +02:00
|
|
|
|
|
|
|
|
2023-01-26 11:08:40 +01:00
|
|
|
# check POST /api/v1/payments: invoice with custom expiry
|
|
|
|
@pytest.mark.asyncio
|
|
|
|
async def test_create_invoice_custom_expiry(client, inkey_headers_to):
|
|
|
|
data = await get_random_invoice_data()
|
|
|
|
expiry_seconds = 600 * 6 * 24 * 31 # 31 days in the future
|
|
|
|
data["expiry"] = expiry_seconds
|
|
|
|
response = await client.post(
|
|
|
|
"/api/v1/payments", json=data, headers=inkey_headers_to
|
|
|
|
)
|
|
|
|
assert response.status_code == 201
|
|
|
|
invoice = response.json()
|
|
|
|
bolt11_invoice = bolt11.decode(invoice["payment_request"])
|
|
|
|
assert bolt11_invoice.expiry == expiry_seconds
|
|
|
|
|
|
|
|
|
2022-06-27 00:11:46 +02:00
|
|
|
# check POST /api/v1/payments: make payment
|
|
|
|
@pytest.mark.asyncio
|
|
|
|
async def test_pay_invoice(client, invoice, adminkey_headers_from):
|
|
|
|
data = {"out": True, "bolt11": invoice["payment_request"]}
|
|
|
|
response = await client.post(
|
|
|
|
"/api/v1/payments", json=data, headers=adminkey_headers_from
|
|
|
|
)
|
|
|
|
assert response.status_code < 300
|
|
|
|
assert len(response.json()["payment_hash"]) == 64
|
|
|
|
assert len(response.json()["checking_id"]) > 0
|
|
|
|
|
|
|
|
|
|
|
|
# check GET /api/v1/payments/<hash>: payment status
|
|
|
|
@pytest.mark.asyncio
|
|
|
|
async def test_check_payment_without_key(client, invoice):
|
|
|
|
# check the payment status
|
|
|
|
response = await client.get(f"/api/v1/payments/{invoice['payment_hash']}")
|
|
|
|
assert response.status_code < 300
|
2023-01-21 15:02:35 +00:00
|
|
|
assert response.json()["paid"] is True
|
2022-06-27 00:11:46 +02:00
|
|
|
assert invoice
|
|
|
|
# not key, that's why no "details"
|
|
|
|
assert "details" not in response.json()
|
|
|
|
|
|
|
|
|
|
|
|
# check GET /api/v1/payments/<hash>: payment status
|
2022-07-07 18:29:26 +02:00
|
|
|
# NOTE: this test is sensitive to which db is used.
|
|
|
|
# If postgres: it will succeed only with inkey_headers_from
|
|
|
|
# If sqlite: it will succeed only with adminkey_headers_to
|
|
|
|
# TODO: fix this
|
2022-06-27 00:11:46 +02:00
|
|
|
@pytest.mark.asyncio
|
2022-07-07 18:29:26 +02:00
|
|
|
async def test_check_payment_with_key(client, invoice, inkey_headers_from):
|
2022-06-27 00:11:46 +02:00
|
|
|
# check the payment status
|
|
|
|
response = await client.get(
|
2022-07-07 18:29:26 +02:00
|
|
|
f"/api/v1/payments/{invoice['payment_hash']}", headers=inkey_headers_from
|
2022-06-27 00:11:46 +02:00
|
|
|
)
|
|
|
|
assert response.status_code < 300
|
2023-01-21 15:02:35 +00:00
|
|
|
assert response.json()["paid"] is True
|
2022-06-27 00:11:46 +02:00
|
|
|
assert invoice
|
|
|
|
# with key, that's why with "details"
|
|
|
|
assert "details" in response.json()
|
|
|
|
|
|
|
|
|
|
|
|
# check POST /api/v1/payments: payment with wrong key type
|
|
|
|
@pytest.mark.asyncio
|
|
|
|
async def test_pay_invoice_wrong_key(client, invoice, adminkey_headers_from):
|
|
|
|
data = {"out": True, "bolt11": invoice["payment_request"]}
|
|
|
|
# try payment with wrong key
|
|
|
|
wrong_adminkey_headers = adminkey_headers_from.copy()
|
|
|
|
wrong_adminkey_headers["X-Api-Key"] = "wrong_key"
|
|
|
|
response = await client.post(
|
|
|
|
"/api/v1/payments", json=data, headers=wrong_adminkey_headers
|
|
|
|
)
|
|
|
|
assert response.status_code >= 300 # should fail
|
|
|
|
|
|
|
|
|
|
|
|
# check POST /api/v1/payments: payment with invoice key [should fail]
|
|
|
|
@pytest.mark.asyncio
|
|
|
|
async def test_pay_invoice_invoicekey(client, invoice, inkey_headers_from):
|
|
|
|
data = {"out": True, "bolt11": invoice["payment_request"]}
|
|
|
|
# try payment with invoice key
|
|
|
|
response = await client.post(
|
|
|
|
"/api/v1/payments", json=data, headers=inkey_headers_from
|
|
|
|
)
|
|
|
|
assert response.status_code >= 300 # should fail
|
|
|
|
|
|
|
|
|
|
|
|
# check POST /api/v1/payments: payment with admin key [should pass]
|
|
|
|
@pytest.mark.asyncio
|
Boltz.exchange Extension (#922)
* initial commit and still draft, ready for review
* forgot to uncomment this line
* fee estimation and blockheight
* resolve conversation with michael, to use mempool websockets instead of boltz status event
* Update lnbits/extensions/boltz/boltz.py
Co-authored-by: michael1011 <me@michael1011.at>
* add status to swaps, add sorting and data into listing
* add swap status checks, change urls to docker test setup, dynamic minimum and maximum limits
* fix docker hosts for development
* add api endpoints to _api_docs
* add wallet name and id, to list and status information
* fix status_update for reverse_swaps
* chore: format with black
* more blackformatting and refactoring create_swap()
* fix variable bug
* check if swap is already refunded
* use create_task instead of ensure_future
* add mempool and boltz urls depending on DEBUG .env
* raise exception in mempool fails
* fix onchain txs, sending funds to wrong address and add a refund address for normal swaps beforehand
* add status to swaps, add sorting and data into listing
* add swap status checks, change urls to docker test setup, dynamic minimum and maximum limits
* add wallet name and id, to list and status information
* fix status_update for reverse_swaps
* chore: format with black
* use create_task instead of ensure_future
* add mempool and boltz urls depending on DEBUG .env
* fix onchain txs, sending funds to wrong address and add a refund address for normal swaps beforehand
* black formatting
* add some logging with loguru, and remove function duplication
* cleanup readme
* updates/suggestions from calle
Co-authored-by: calle <93376500+callebtc@users.noreply.github.com>
* remove unused comments
* Update API Endpoints
Co-authored-by: calle <93376500+callebtc@users.noreply.github.com>
* un-factor get_boltz_pairs
* added a explaination for the onchain tx
* remove unused template file
* rename api endpoints
* fix isort and prettier
* more verbose logging!!
* add boltz to mock_data.zip
* new mockdata
* remove comment
* better readme
* fix mempool urls
* change /refund /check /status to post requests
* first step in tests2
* add first tests
* change refund,check,status to post requests
* next try on tests
* overall code improvements
* just testing tests
* throw http exceptions in views_api
* require admincheck for refund,check,status and added fastapi documentation for those
* added more tests
* black
* many code improvements
* adding tests
* temp fix test
* fix race condition when pay_invoice fails
* test are working
* add boltz env variables
* add startup check, bugfixes, improvements
* improve on status checking
* remove check_invoice_status
* more fixes and tests
* testing testing testing
* make tests run again inside regtest
* fix bad error :O
* fix postgres boolean bug and add swap test
* Update README.md
Update README.md
Update README.md
Update README.md
* some mypy
* blacked
* the missing commit?
* fix api_docs readme link
* better refunding error catching
fix
* check swaps now also shows pending reverse swap, ui improvements, tooltips
* add backend check for boltz limits
fixup
* many improvements, startup check for swaps working, reverse needs more testing
* little last fixes
* remove unused logic
* fastapi documentation
fixup
* formatting and remove unused tests
* fix test
* fix swapstatus model
* Update lnbits/extensions/boltz/tasks.py
Co-authored-by: calle <93376500+callebtc@users.noreply.github.com>
* Update lnbits/extensions/boltz/views_api.py
Co-authored-by: calle <93376500+callebtc@users.noreply.github.com>
* balance check msg, format
* fix mypy data override
* fix swapstatus, remove can refund column
* Update lnbits/extensions/boltz/README.md
Co-authored-by: michael1011 <me@michael1011.at>
* empty lines
* fix error message when swap is not found
* remove preimage_hash from database
* fix api_docs html
fix api_docs html
* catch boltz network exceptions better
* formatting
* check for timeout on swap at get request
Co-authored-by: michael1011 <me@michael1011.at>
Co-authored-by: fusion44 <some.fusion@gmail.com>
Co-authored-by: calle <93376500+callebtc@users.noreply.github.com>
2022-08-30 12:51:17 +02:00
|
|
|
@pytest.mark.skipif(is_regtest, reason="this only works in fakewallet")
|
2022-06-27 00:11:46 +02:00
|
|
|
async def test_pay_invoice_adminkey(client, invoice, adminkey_headers_from):
|
|
|
|
data = {"out": True, "bolt11": invoice["payment_request"]}
|
|
|
|
# try payment with admin key
|
|
|
|
response = await client.post(
|
|
|
|
"/api/v1/payments", json=data, headers=adminkey_headers_from
|
|
|
|
)
|
|
|
|
assert response.status_code < 300 # should pass
|
|
|
|
|
|
|
|
|
|
|
|
# check POST /api/v1/payments/decode
|
|
|
|
@pytest.mark.asyncio
|
|
|
|
async def test_decode_invoice(client, invoice):
|
|
|
|
data = {"data": invoice["payment_request"]}
|
|
|
|
response = await client.post(
|
|
|
|
"/api/v1/payments/decode",
|
|
|
|
json=data,
|
|
|
|
)
|
|
|
|
assert response.status_code < 300
|
|
|
|
assert response.json()["payment_hash"] == invoice["payment_hash"]
|
2022-07-20 11:52:49 +02:00
|
|
|
|
|
|
|
|
|
|
|
# check api_payment() internal function call (NOT API): payment status
|
|
|
|
@pytest.mark.asyncio
|
|
|
|
async def test_api_payment_without_key(invoice):
|
|
|
|
# check the payment status
|
|
|
|
response = await api_payment(invoice["payment_hash"])
|
|
|
|
assert type(response) == dict
|
2023-01-21 15:02:35 +00:00
|
|
|
assert response["paid"] is True
|
2022-07-20 13:15:18 +02:00
|
|
|
# no key, that's why no "details"
|
2022-07-20 11:52:49 +02:00
|
|
|
assert "details" not in response
|
|
|
|
|
|
|
|
|
|
|
|
# check api_payment() internal function call (NOT API): payment status
|
|
|
|
@pytest.mark.asyncio
|
|
|
|
async def test_api_payment_with_key(invoice, inkey_headers_from):
|
|
|
|
# check the payment status
|
|
|
|
response = await api_payment(
|
|
|
|
invoice["payment_hash"], inkey_headers_from["X-Api-Key"]
|
|
|
|
)
|
|
|
|
assert type(response) == dict
|
2023-01-21 15:02:35 +00:00
|
|
|
assert response["paid"] is True
|
2022-07-20 11:52:49 +02:00
|
|
|
assert "details" in response
|
2022-08-01 16:12:25 +02:00
|
|
|
|
|
|
|
|
|
|
|
# check POST /api/v1/payments: invoice creation with a description hash
|
2022-08-13 14:29:04 +02:00
|
|
|
@pytest.mark.skipif(
|
2022-10-05 11:15:11 +02:00
|
|
|
WALLET.__class__.__name__ in ["CoreLightningWallet"],
|
2022-08-13 14:29:04 +02:00
|
|
|
reason="wallet does not support description_hash",
|
|
|
|
)
|
2022-08-01 16:12:25 +02:00
|
|
|
@pytest.mark.asyncio
|
|
|
|
async def test_create_invoice_with_description_hash(client, inkey_headers_to):
|
|
|
|
data = await get_random_invoice_data()
|
2022-12-29 16:50:05 +01:00
|
|
|
descr_hash = hashlib.sha256("asdasdasd".encode()).hexdigest()
|
2022-08-13 14:29:04 +02:00
|
|
|
data["description_hash"] = descr_hash
|
|
|
|
|
|
|
|
response = await client.post(
|
|
|
|
"/api/v1/payments", json=data, headers=inkey_headers_to
|
|
|
|
)
|
|
|
|
invoice = response.json()
|
|
|
|
|
|
|
|
invoice_bolt11 = bolt11.decode(invoice["payment_request"])
|
|
|
|
assert invoice_bolt11.description_hash == descr_hash
|
|
|
|
assert invoice_bolt11.description is None
|
|
|
|
return invoice
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
|
|
async def test_create_invoice_with_unhashed_description(client, inkey_headers_to):
|
|
|
|
data = await get_random_invoice_data()
|
2022-12-29 16:50:05 +01:00
|
|
|
descr_hash = hashlib.sha256("asdasdasd".encode()).hexdigest()
|
|
|
|
data["unhashed_description"] = "asdasdasd".encode().hex()
|
2022-08-01 16:12:25 +02:00
|
|
|
|
|
|
|
response = await client.post(
|
|
|
|
"/api/v1/payments", json=data, headers=inkey_headers_to
|
|
|
|
)
|
|
|
|
invoice = response.json()
|
|
|
|
|
|
|
|
invoice_bolt11 = bolt11.decode(invoice["payment_request"])
|
|
|
|
assert invoice_bolt11.description_hash == descr_hash
|
|
|
|
assert invoice_bolt11.description is None
|
|
|
|
return invoice
|