mirror of
https://github.com/ElementsProject/lightning.git
synced 2025-01-18 13:25:43 +01:00
1327 lines
43 KiB
Python
1327 lines
43 KiB
Python
from decimal import Decimal
|
|
from math import floor, log10
|
|
from typing import Optional, Union
|
|
import json
|
|
import logging
|
|
import os
|
|
import socket
|
|
import warnings
|
|
from json import JSONEncoder
|
|
|
|
|
|
def _patched_default(self, obj):
|
|
return getattr(obj.__class__, "to_json", _patched_default.default)(obj)
|
|
|
|
|
|
def monkey_patch_json(patch=True):
|
|
is_patched = JSONEncoder.default == _patched_default
|
|
|
|
if patch and not is_patched:
|
|
_patched_default.default = JSONEncoder.default # Save unmodified
|
|
JSONEncoder.default = _patched_default # Replace it.
|
|
elif not patch and is_patched:
|
|
JSONEncoder.default = _patched_default.default
|
|
|
|
|
|
class RpcError(ValueError):
|
|
def __init__(self, method: str, payload: dict, error: str):
|
|
super(ValueError, self).__init__(
|
|
"RPC call failed: method: {}, payload: {}, error: {}".format(
|
|
method, payload, error
|
|
)
|
|
)
|
|
|
|
self.method = method
|
|
self.payload = payload
|
|
self.error = error
|
|
|
|
|
|
class Millisatoshi:
|
|
"""
|
|
A subtype to represent thousandths of a satoshi.
|
|
|
|
Many JSON API fields are expressed in millisatoshis: these automatically
|
|
get turned into Millisatoshi types. Converts to and from int.
|
|
"""
|
|
def __init__(self, v: Union[int, str, Decimal]):
|
|
"""
|
|
Takes either a string ending in 'msat', 'sat', 'btc' or an integer.
|
|
"""
|
|
if isinstance(v, str):
|
|
if v.endswith("msat"):
|
|
parsed = Decimal(v[0:-4])
|
|
elif v.endswith("sat"):
|
|
parsed = Decimal(v[0:-3]) * 1000
|
|
elif v.endswith("btc"):
|
|
parsed = Decimal(v[0:-3]) * 1000 * 10**8
|
|
else:
|
|
raise TypeError(
|
|
"Millisatoshi must be string with msat/sat/btc suffix or"
|
|
" int"
|
|
)
|
|
if parsed != int(parsed):
|
|
raise ValueError("Millisatoshi must be a whole number")
|
|
self.millisatoshis = int(parsed)
|
|
|
|
elif isinstance(v, Millisatoshi):
|
|
self.millisatoshis = v.millisatoshis
|
|
|
|
elif int(v) == v:
|
|
self.millisatoshis = int(v)
|
|
|
|
elif isinstance(v, float):
|
|
raise TypeError("Millisatoshi by float is currently not supported")
|
|
|
|
else:
|
|
raise TypeError(
|
|
"Millisatoshi must be string with msat/sat/btc suffix or int"
|
|
)
|
|
|
|
if self.millisatoshis < 0:
|
|
raise ValueError("Millisatoshi must be >= 0")
|
|
|
|
def __repr__(self) -> str:
|
|
"""
|
|
Appends the 'msat' as expected for this type.
|
|
"""
|
|
return str(self.millisatoshis) + "msat"
|
|
|
|
def to_satoshi(self) -> Decimal:
|
|
"""
|
|
Return a Decimal representing the number of satoshis.
|
|
"""
|
|
return Decimal(self.millisatoshis) / 1000
|
|
|
|
def to_btc(self) -> Decimal:
|
|
"""
|
|
Return a Decimal representing the number of bitcoin.
|
|
"""
|
|
return Decimal(self.millisatoshis) / 1000 / 10**8
|
|
|
|
def to_satoshi_str(self) -> str:
|
|
"""
|
|
Return a string of form 1234sat or 1234.567sat.
|
|
"""
|
|
if self.millisatoshis % 1000:
|
|
return '{:.3f}sat'.format(self.to_satoshi())
|
|
else:
|
|
return '{:.0f}sat'.format(self.to_satoshi())
|
|
|
|
def to_btc_str(self) -> str:
|
|
"""
|
|
Return a string of form 12.34567890btc or 12.34567890123btc.
|
|
"""
|
|
if self.millisatoshis % 1000:
|
|
return '{:.11f}btc'.format(self.to_btc())
|
|
else:
|
|
return '{:.8f}btc'.format(self.to_btc())
|
|
|
|
def to_approx_str(self, digits: int = 3) -> str:
|
|
"""Returns the shortmost string using common units representation.
|
|
|
|
Rounds to significant `digits`. Default: 3
|
|
"""
|
|
def round_to_n(x: int, n: int) -> float:
|
|
return round(x, -int(floor(log10(x))) + (n - 1))
|
|
result = self.to_satoshi_str()
|
|
|
|
# we try to increase digits to check if we did loose out on precision
|
|
# without gaining a shorter string, since this is a rarely used UI
|
|
# function, performance is not an issue. Adds at least one iteration.
|
|
while True:
|
|
# first round everything down to effective digits
|
|
amount_rounded = round_to_n(self.millisatoshis, digits)
|
|
# try different units and take shortest resulting normalized string
|
|
amounts_str = [
|
|
"%gbtc" % (amount_rounded / 1000 / 10**8),
|
|
"%gsat" % (amount_rounded / 1000),
|
|
"%gmsat" % (amount_rounded),
|
|
]
|
|
test_result = min(amounts_str, key=len)
|
|
|
|
# check result and do another run if necessary
|
|
if test_result == result:
|
|
return result
|
|
elif not result or len(test_result) <= len(result):
|
|
digits = digits + 1
|
|
result = test_result
|
|
else:
|
|
return result
|
|
|
|
def to_json(self) -> str:
|
|
return self.__repr__()
|
|
|
|
def __int__(self) -> int:
|
|
return self.millisatoshis
|
|
|
|
def __lt__(self, other: 'Millisatoshi') -> bool:
|
|
return self.millisatoshis < other.millisatoshis
|
|
|
|
def __le__(self, other: 'Millisatoshi') -> bool:
|
|
return self.millisatoshis <= other.millisatoshis
|
|
|
|
def __eq__(self, other: object) -> bool:
|
|
if isinstance(other, Millisatoshi):
|
|
return self.millisatoshis == other.millisatoshis
|
|
elif isinstance(other, int):
|
|
return self.millisatoshis == other
|
|
else:
|
|
return False
|
|
|
|
def __gt__(self, other: 'Millisatoshi') -> bool:
|
|
return self.millisatoshis > other.millisatoshis
|
|
|
|
def __ge__(self, other: 'Millisatoshi') -> bool:
|
|
return self.millisatoshis >= other.millisatoshis
|
|
|
|
def __add__(self, other: 'Millisatoshi') -> 'Millisatoshi':
|
|
return Millisatoshi(int(self) + int(other))
|
|
|
|
def __sub__(self, other: 'Millisatoshi') -> 'Millisatoshi':
|
|
return Millisatoshi(int(self) - int(other))
|
|
|
|
def __mul__(self, other: Union[int, float]) -> 'Millisatoshi':
|
|
if isinstance(other, Millisatoshi):
|
|
raise TypeError("Resulting unit msat^2 is not supported")
|
|
return Millisatoshi(floor(self.millisatoshis * other))
|
|
|
|
def __truediv__(self, other: Union[int, float, 'Millisatoshi']) -> Union['Millisatoshi', float]:
|
|
if isinstance(other, Millisatoshi):
|
|
return self.millisatoshis / other.millisatoshis
|
|
return Millisatoshi(floor(self.millisatoshis / other))
|
|
|
|
def __floordiv__(self, other: Union[int, float, 'Millisatoshi']) -> Union['Millisatoshi', int]:
|
|
if isinstance(other, Millisatoshi):
|
|
return self.millisatoshis // other.millisatoshis
|
|
return Millisatoshi(floor(self.millisatoshis // float(other)))
|
|
|
|
def __mod__(self, other: Union[float, int]) -> 'Millisatoshi':
|
|
return Millisatoshi(int(self.millisatoshis % other))
|
|
|
|
def __radd__(self, other: 'Millisatoshi') -> 'Millisatoshi':
|
|
return Millisatoshi(int(self) + int(other))
|
|
|
|
|
|
class UnixSocket(object):
|
|
"""A wrapper for socket.socket that is specialized to unix sockets.
|
|
|
|
Some OS implementations impose restrictions on the Unix sockets.
|
|
|
|
- On linux OSs the socket path must be shorter than the in-kernel buffer
|
|
size (somewhere around 100 bytes), thus long paths may end up failing
|
|
the `socket.connect` call.
|
|
|
|
This is a small wrapper that tries to work around these limitations.
|
|
|
|
"""
|
|
|
|
def __init__(self, path: str):
|
|
self.path = path
|
|
self.sock: Optional[socket.SocketType] = None
|
|
self.connect()
|
|
|
|
def connect(self) -> None:
|
|
try:
|
|
self.sock = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)
|
|
self.sock.connect(self.path)
|
|
except OSError as e:
|
|
self.close()
|
|
|
|
if (e.args[0] == "AF_UNIX path too long" and os.uname()[0] == "Linux"):
|
|
# If this is a Linux system we may be able to work around this
|
|
# issue by opening our directory and using `/proc/self/fd/` to
|
|
# get a short alias for the socket file.
|
|
#
|
|
# This was heavily inspired by the Open vSwitch code see here:
|
|
# https://github.com/openvswitch/ovs/blob/master/python/ovs/socket_util.py
|
|
|
|
dirname = os.path.dirname(self.path)
|
|
basename = os.path.basename(self.path)
|
|
|
|
# Open an fd to our home directory, that we can then find
|
|
# through `/proc/self/fd` and access the contents.
|
|
dirfd = os.open(dirname, os.O_DIRECTORY | os.O_RDONLY)
|
|
short_path = "/proc/self/fd/%d/%s" % (dirfd, basename)
|
|
self.sock = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)
|
|
self.sock.connect(short_path)
|
|
else:
|
|
# There is no good way to recover from this.
|
|
raise
|
|
|
|
def close(self) -> None:
|
|
if self.sock is not None:
|
|
self.sock.close()
|
|
self.sock = None
|
|
|
|
def sendall(self, b: bytes) -> None:
|
|
if self.sock is None:
|
|
raise socket.error("not connected")
|
|
|
|
self.sock.sendall(b)
|
|
|
|
def recv(self, length: int) -> bytes:
|
|
if self.sock is None:
|
|
raise socket.error("not connected")
|
|
|
|
return self.sock.recv(length)
|
|
|
|
def __del__(self) -> None:
|
|
self.close()
|
|
|
|
|
|
class UnixDomainSocketRpc(object):
|
|
def __init__(self, socket_path, executor=None, logger=logging, encoder_cls=json.JSONEncoder, decoder=json.JSONDecoder()):
|
|
self.socket_path = socket_path
|
|
self.encoder_cls = encoder_cls
|
|
self.decoder = decoder
|
|
self.executor = executor
|
|
self.logger = logger
|
|
|
|
self.next_id = 0
|
|
|
|
def _writeobj(self, sock, obj):
|
|
s = json.dumps(obj, ensure_ascii=False, cls=self.encoder_cls)
|
|
sock.sendall(bytearray(s, 'UTF-8'))
|
|
|
|
def _readobj(self, sock, buff=b''):
|
|
"""Read a JSON object, starting with buff; returns object and any buffer left over."""
|
|
while True:
|
|
parts = buff.split(b'\n\n', 1)
|
|
if len(parts) == 1:
|
|
# Didn't read enough.
|
|
b = sock.recv(max(1024, len(buff)))
|
|
buff += b
|
|
if len(b) == 0:
|
|
return {'error': 'Connection to RPC server lost.'}, buff
|
|
else:
|
|
buff = parts[1]
|
|
obj, _ = self.decoder.raw_decode(parts[0].decode("UTF-8"))
|
|
return obj, buff
|
|
|
|
def __getattr__(self, name):
|
|
"""Intercept any call that is not explicitly defined and call @call.
|
|
|
|
We might still want to define the actual methods in the subclasses for
|
|
documentation purposes.
|
|
"""
|
|
name = name.replace('_', '-')
|
|
|
|
def wrapper(*args, **kwargs):
|
|
if len(args) != 0 and len(kwargs) != 0:
|
|
raise RpcError("Cannot mix positional and non-positional arguments")
|
|
elif len(args) != 0:
|
|
return self.call(name, payload=args)
|
|
else:
|
|
return self.call(name, payload=kwargs)
|
|
return wrapper
|
|
|
|
def call(self, method, payload=None):
|
|
self.logger.debug("Calling %s with payload %r", method, payload)
|
|
|
|
if payload is None:
|
|
payload = {}
|
|
# Filter out arguments that are None
|
|
if isinstance(payload, dict):
|
|
payload = {k: v for k, v in payload.items() if v is not None}
|
|
|
|
# FIXME: we open a new socket for every readobj call...
|
|
sock = UnixSocket(self.socket_path)
|
|
this_id = self.next_id
|
|
self._writeobj(sock, {
|
|
"jsonrpc": "2.0",
|
|
"method": method,
|
|
"params": payload,
|
|
"id": this_id,
|
|
})
|
|
self.next_id += 1
|
|
buf = b''
|
|
while True:
|
|
resp, buf = self._readobj(sock, buf)
|
|
# FIXME: We should offer a callback for notifications.
|
|
if 'method' not in resp or 'id' in resp:
|
|
break
|
|
|
|
self.logger.debug("Received response for %s call: %r", method, resp)
|
|
if 'id' in resp and resp['id'] != this_id:
|
|
raise ValueError("Malformed response, id is not {}: {}.".format(this_id, resp))
|
|
sock.close()
|
|
|
|
if not isinstance(resp, dict):
|
|
raise ValueError("Malformed response, response is not a dictionary %s." % resp)
|
|
elif "error" in resp:
|
|
raise RpcError(method, payload, resp['error'])
|
|
elif "result" not in resp:
|
|
raise ValueError("Malformed response, \"result\" missing.")
|
|
return resp["result"]
|
|
|
|
|
|
class LightningRpc(UnixDomainSocketRpc):
|
|
"""
|
|
RPC client for the `lightningd` daemon.
|
|
|
|
This RPC client connects to the `lightningd` daemon through a unix
|
|
domain socket and passes calls through. Since some of the calls
|
|
are blocking, the corresponding python methods include an `async`
|
|
keyword argument. If `async` is set to true then the method
|
|
returns a future immediately, instead of blocking indefinitely.
|
|
|
|
This implementation is thread safe in that it locks the socket
|
|
between calls, but it does not (yet) support concurrent calls.
|
|
"""
|
|
|
|
class LightningJSONEncoder(json.JSONEncoder):
|
|
def default(self, o):
|
|
try:
|
|
return o.to_json()
|
|
except NameError:
|
|
pass
|
|
return json.JSONEncoder.default(self, o)
|
|
|
|
class LightningJSONDecoder(json.JSONDecoder):
|
|
def __init__(self, *, object_hook=None, parse_float=None,
|
|
parse_int=None, parse_constant=None,
|
|
strict=True, object_pairs_hook=None,
|
|
patch_json=True):
|
|
self.object_hook_next = object_hook
|
|
super().__init__(object_hook=self.millisatoshi_hook, parse_float=parse_float, parse_int=parse_int, parse_constant=parse_constant, strict=strict, object_pairs_hook=object_pairs_hook)
|
|
|
|
@staticmethod
|
|
def replace_amounts(obj):
|
|
"""
|
|
Recursively replace _msat fields with appropriate values with Millisatoshi.
|
|
"""
|
|
if isinstance(obj, dict):
|
|
for k, v in obj.items():
|
|
if k.endswith('msat'):
|
|
if isinstance(v, str) and v.endswith('msat'):
|
|
obj[k] = Millisatoshi(v)
|
|
# Special case for array of msat values
|
|
elif isinstance(v, list) and all(isinstance(e, str) and e.endswith('msat') for e in v):
|
|
obj[k] = [Millisatoshi(e) for e in v]
|
|
else:
|
|
obj[k] = LightningRpc.LightningJSONDecoder.replace_amounts(v)
|
|
elif isinstance(obj, list):
|
|
obj = [LightningRpc.LightningJSONDecoder.replace_amounts(e) for e in obj]
|
|
|
|
return obj
|
|
|
|
def millisatoshi_hook(self, obj):
|
|
obj = LightningRpc.LightningJSONDecoder.replace_amounts(obj)
|
|
if self.object_hook_next:
|
|
obj = self.object_hook_next(obj)
|
|
return obj
|
|
|
|
def __init__(self, socket_path, executor=None, logger=logging,
|
|
patch_json=True):
|
|
super().__init__(
|
|
socket_path,
|
|
executor,
|
|
logger,
|
|
self.LightningJSONEncoder,
|
|
self.LightningJSONDecoder()
|
|
)
|
|
|
|
if patch_json:
|
|
monkey_patch_json(patch=True)
|
|
|
|
def autocleaninvoice(self, cycle_seconds=None, expired_by=None):
|
|
"""
|
|
Sets up automatic cleaning of expired invoices. {cycle_seconds} sets
|
|
the cleaning frequency in seconds (defaults to 3600) and {expired_by}
|
|
sets the minimum time an invoice should have been expired for to be
|
|
cleaned in seconds (defaults to 86400).
|
|
"""
|
|
payload = {
|
|
"cycle_seconds": cycle_seconds,
|
|
"expired_by": expired_by
|
|
}
|
|
return self.call("autocleaninvoice", payload)
|
|
|
|
def check(self, command_to_check, **kwargs):
|
|
"""
|
|
Checks if a command is valid without running it.
|
|
"""
|
|
payload = {"command_to_check": command_to_check}
|
|
payload.update({k: v for k, v in kwargs.items()})
|
|
return self.call("check", payload)
|
|
|
|
def _deprecated_close(self, peer_id, force=None, timeout=None):
|
|
warnings.warn("close now takes unilateraltimeout arg: expect removal"
|
|
" in early 2020",
|
|
DeprecationWarning)
|
|
payload = {
|
|
"id": peer_id,
|
|
"force": force,
|
|
"timeout": timeout
|
|
}
|
|
return self.call("close", payload)
|
|
|
|
def close(self, peer_id, *args, **kwargs):
|
|
"""
|
|
Close the channel with peer {id}, forcing a unilateral
|
|
close after {unilateraltimeout} seconds if non-zero, and
|
|
the to-local output will be sent to {destination}.
|
|
|
|
Deprecated usage has {force} and {timeout} args.
|
|
"""
|
|
|
|
if 'force' in kwargs or 'timeout' in kwargs:
|
|
return self._deprecated_close(peer_id, *args, **kwargs)
|
|
|
|
# Single arg is ambigious.
|
|
if len(args) >= 1:
|
|
if isinstance(args[0], bool):
|
|
return self._deprecated_close(peer_id, *args, **kwargs)
|
|
if len(args) == 2:
|
|
if args[0] is None and isinstance(args[1], int):
|
|
return self._deprecated_close(peer_id, *args, **kwargs)
|
|
|
|
def _close(peer_id, unilateraltimeout=None, destination=None, fee_negotiation_step=None):
|
|
payload = {
|
|
"id": peer_id,
|
|
"unilateraltimeout": unilateraltimeout,
|
|
"destination": destination,
|
|
"fee_negotiation_step": fee_negotiation_step
|
|
}
|
|
return self.call("close", payload)
|
|
|
|
return _close(peer_id, *args, **kwargs)
|
|
|
|
def connect(self, peer_id, host=None, port=None):
|
|
"""
|
|
Connect to {peer_id} at {host} and {port}.
|
|
"""
|
|
payload = {
|
|
"id": peer_id,
|
|
"host": host,
|
|
"port": port
|
|
}
|
|
return self.call("connect", payload)
|
|
|
|
def decodepay(self, bolt11, description=None):
|
|
"""
|
|
Decode {bolt11}, using {description} if necessary.
|
|
"""
|
|
payload = {
|
|
"bolt11": bolt11,
|
|
"description": description
|
|
}
|
|
return self.call("decodepay", payload)
|
|
|
|
def delexpiredinvoice(self, maxexpirytime=None):
|
|
"""
|
|
Delete all invoices that have expired on or before the given {maxexpirytime}.
|
|
"""
|
|
payload = {
|
|
"maxexpirytime": maxexpirytime
|
|
}
|
|
return self.call("delexpiredinvoice", payload)
|
|
|
|
def delinvoice(self, label, status):
|
|
"""
|
|
Delete unpaid invoice {label} with {status}.
|
|
"""
|
|
payload = {
|
|
"label": label,
|
|
"status": status
|
|
}
|
|
return self.call("delinvoice", payload)
|
|
|
|
def dev_crash(self):
|
|
"""
|
|
Crash lightningd by calling fatal().
|
|
"""
|
|
payload = {
|
|
"subcommand": "crash"
|
|
}
|
|
return self.call("dev", payload)
|
|
|
|
def dev_fail(self, peer_id):
|
|
"""
|
|
Fail with peer {peer_id}.
|
|
"""
|
|
payload = {
|
|
"id": peer_id
|
|
}
|
|
return self.call("dev-fail", payload)
|
|
|
|
def dev_forget_channel(self, peerid, force=False):
|
|
""" Forget the channel with id=peerid.
|
|
"""
|
|
return self.call(
|
|
"dev-forget-channel",
|
|
payload={"id": peerid, "force": force}
|
|
)
|
|
|
|
def dev_memdump(self):
|
|
"""
|
|
Show memory objects currently in use.
|
|
"""
|
|
return self.call("dev-memdump")
|
|
|
|
def dev_memleak(self):
|
|
"""
|
|
Show unreferenced memory objects.
|
|
"""
|
|
return self.call("dev-memleak")
|
|
|
|
def dev_pay(self, bolt11, msatoshi=None, label=None, riskfactor=None,
|
|
description=None, maxfeepercent=None, retry_for=None,
|
|
maxdelay=None, exemptfee=None, use_shadow=True):
|
|
"""
|
|
A developer version of `pay`, with the possibility to deactivate
|
|
shadow routing (used for testing).
|
|
"""
|
|
payload = {
|
|
"bolt11": bolt11,
|
|
"msatoshi": msatoshi,
|
|
"label": label,
|
|
"riskfactor": riskfactor,
|
|
"maxfeepercent": maxfeepercent,
|
|
"retry_for": retry_for,
|
|
"maxdelay": maxdelay,
|
|
"exemptfee": exemptfee,
|
|
"use_shadow": use_shadow,
|
|
# Deprecated.
|
|
"description": description,
|
|
}
|
|
return self.call("pay", payload)
|
|
|
|
def dev_reenable_commit(self, peer_id):
|
|
"""
|
|
Re-enable the commit timer on peer {id}.
|
|
"""
|
|
payload = {
|
|
"id": peer_id
|
|
}
|
|
return self.call("dev-reenable-commit", payload)
|
|
|
|
def dev_rescan_outputs(self):
|
|
"""
|
|
Synchronize the state of our funds with bitcoind.
|
|
"""
|
|
return self.call("dev-rescan-outputs")
|
|
|
|
def dev_rhash(self, secret):
|
|
"""
|
|
Show SHA256 of {secret}
|
|
"""
|
|
payload = {
|
|
"subcommand": "rhash",
|
|
"secret": secret
|
|
}
|
|
return self.call("dev", payload)
|
|
|
|
def dev_sign_last_tx(self, peer_id):
|
|
"""
|
|
Sign and show the last commitment transaction with peer {id}.
|
|
"""
|
|
payload = {
|
|
"id": peer_id
|
|
}
|
|
return self.call("dev-sign-last-tx", payload)
|
|
|
|
def dev_slowcmd(self, msec=None):
|
|
"""
|
|
Torture test for slow commands, optional {msec}.
|
|
"""
|
|
payload = {
|
|
"subcommand": "slowcmd",
|
|
"msec": msec
|
|
}
|
|
return self.call("dev", payload)
|
|
|
|
def disconnect(self, peer_id, force=False):
|
|
"""
|
|
Disconnect from peer with {peer_id}, optional {force} even if has active channel.
|
|
"""
|
|
payload = {
|
|
"id": peer_id,
|
|
"force": force,
|
|
}
|
|
return self.call("disconnect", payload)
|
|
|
|
def feerates(self, style, urgent=None, normal=None, slow=None):
|
|
"""
|
|
Supply feerate estimates manually.
|
|
"""
|
|
payload = {
|
|
"style": style,
|
|
"urgent": urgent,
|
|
"normal": normal,
|
|
"slow": slow
|
|
}
|
|
return self.call("feerates", payload)
|
|
|
|
def _deprecated_fundchannel(self, node_id, satoshi, feerate=None, announce=True, minconf=None, utxos=None):
|
|
warnings.warn("fundchannel: the 'satoshi' field is renamed 'amount' : expect removal"
|
|
" in Mid-2020",
|
|
DeprecationWarning)
|
|
|
|
payload = {
|
|
"id": node_id,
|
|
"satoshi": satoshi,
|
|
"feerate": feerate,
|
|
"announce": announce,
|
|
"minconf": minconf,
|
|
"utxos": utxos
|
|
}
|
|
return self.call("fundchannel", payload)
|
|
|
|
def fundchannel(self, node_id, *args, **kwargs):
|
|
"""
|
|
Fund channel with {id} using {amount} satoshis with feerate
|
|
of {feerate} (uses default feerate if unset).
|
|
If {announce} is False, don't send channel announcements.
|
|
Only select outputs with {minconf} confirmations.
|
|
If {utxos} is specified (as a list of 'txid:vout' strings),
|
|
fund a channel from these specifics utxos.
|
|
{close_to} is a valid Bitcoin address.
|
|
"""
|
|
|
|
if 'satoshi' in kwargs:
|
|
return self._deprecated_fundchannel(node_id, *args, **kwargs)
|
|
|
|
def _fundchannel(node_id, amount, feerate=None, announce=True, minconf=None, utxos=None, push_msat=None, close_to=None):
|
|
payload = {
|
|
"id": node_id,
|
|
"amount": amount,
|
|
"feerate": feerate,
|
|
"announce": announce,
|
|
"minconf": minconf,
|
|
"utxos": utxos,
|
|
"push_msat": push_msat,
|
|
"close_to": close_to,
|
|
}
|
|
return self.call("fundchannel", payload)
|
|
|
|
return _fundchannel(node_id, *args, **kwargs)
|
|
|
|
def _deprecated_fundchannel_start(self, node_id, satoshi, feerate=None, announce=True):
|
|
warnings.warn("fundchannel_start: the 'satoshi' field is renamed 'amount' : expect removal"
|
|
" in Mid-2020",
|
|
DeprecationWarning)
|
|
|
|
payload = {
|
|
"id": node_id,
|
|
"satoshi": satoshi,
|
|
"feerate": feerate,
|
|
"announce": announce,
|
|
}
|
|
return self.call("fundchannel_start", payload)
|
|
|
|
def fundchannel_start(self, node_id, *args, **kwargs):
|
|
"""
|
|
Start channel funding with {id} for {amount} satoshis
|
|
with feerate of {feerate} (uses default feerate if unset).
|
|
If {announce} is False, don't send channel announcements.
|
|
Returns a Bech32 {funding_address} for an external wallet
|
|
to create a funding transaction for. Requires a call to
|
|
'fundchannel_complete' to complete channel establishment
|
|
with peer.
|
|
"""
|
|
|
|
if 'satoshi' in kwargs:
|
|
return self._deprecated_fundchannel_start(node_id, *args, **kwargs)
|
|
|
|
def _fundchannel_start(node_id, amount, feerate=None, announce=True, close_to=None):
|
|
payload = {
|
|
"id": node_id,
|
|
"amount": amount,
|
|
"feerate": feerate,
|
|
"announce": announce,
|
|
"close_to": close_to,
|
|
}
|
|
return self.call("fundchannel_start", payload)
|
|
|
|
return _fundchannel_start(node_id, *args, **kwargs)
|
|
|
|
def fundchannel_cancel(self, node_id):
|
|
"""
|
|
Cancel a 'started' fundchannel with node {id}.
|
|
"""
|
|
payload = {
|
|
"id": node_id,
|
|
}
|
|
return self.call("fundchannel_cancel", payload)
|
|
|
|
def fundchannel_complete(self, node_id, funding_txid, funding_txout):
|
|
"""
|
|
Complete channel establishment with {id}, using {funding_txid} at {funding_txout}.
|
|
"""
|
|
payload = {
|
|
"id": node_id,
|
|
"txid": funding_txid,
|
|
"txout": funding_txout,
|
|
}
|
|
return self.call("fundchannel_complete", payload)
|
|
|
|
def getinfo(self):
|
|
"""
|
|
Show information about this node.
|
|
"""
|
|
return self.call("getinfo")
|
|
|
|
def getlog(self, level=None):
|
|
"""
|
|
Show logs, with optional log {level} (info|unusual|debug|io).
|
|
"""
|
|
payload = {
|
|
"level": level
|
|
}
|
|
return self.call("getlog", payload)
|
|
|
|
def getpeer(self, peer_id, level=None):
|
|
"""
|
|
Show peer with {peer_id}, if {level} is set, include {log}s.
|
|
"""
|
|
payload = {
|
|
"id": peer_id,
|
|
"level": level
|
|
}
|
|
res = self.call("listpeers", payload)
|
|
return res.get("peers") and res["peers"][0] or None
|
|
|
|
def getroute(self, node_id, msatoshi, riskfactor, cltv=9, fromid=None, fuzzpercent=None, exclude=[], maxhops=20):
|
|
"""
|
|
Show route to {id} for {msatoshi}, using {riskfactor} and optional
|
|
{cltv} (default 9). If specified search from {fromid} otherwise use
|
|
this node as source. Randomize the route with up to {fuzzpercent}
|
|
(0.0 -> 100.0, default 5.0). {exclude} is an optional array of
|
|
scid/direction or node-id to exclude. Limit the number of hops in the
|
|
route to {maxhops}.
|
|
"""
|
|
payload = {
|
|
"id": node_id,
|
|
"msatoshi": msatoshi,
|
|
"riskfactor": riskfactor,
|
|
"cltv": cltv,
|
|
"fromid": fromid,
|
|
"fuzzpercent": fuzzpercent,
|
|
"exclude": exclude,
|
|
"maxhops": maxhops
|
|
}
|
|
return self.call("getroute", payload)
|
|
|
|
def help(self, command=None):
|
|
"""
|
|
Show available commands, or just {command} if supplied.
|
|
"""
|
|
payload = {
|
|
"command": command,
|
|
}
|
|
return self.call("help", payload)
|
|
|
|
def invoice(self, msatoshi, label, description, expiry=None, fallbacks=None, preimage=None, exposeprivatechannels=None):
|
|
"""
|
|
Create an invoice for {msatoshi} with {label} and {description} with
|
|
optional {expiry} seconds (default 1 week).
|
|
"""
|
|
payload = {
|
|
"msatoshi": msatoshi,
|
|
"label": label,
|
|
"description": description,
|
|
"expiry": expiry,
|
|
"fallbacks": fallbacks,
|
|
"preimage": preimage,
|
|
"exposeprivatechannels": exposeprivatechannels
|
|
}
|
|
return self.call("invoice", payload)
|
|
|
|
def listchannels(self, short_channel_id=None, source=None):
|
|
"""
|
|
Show all known channels, accept optional {short_channel_id} or {source}.
|
|
"""
|
|
payload = {
|
|
"short_channel_id": short_channel_id,
|
|
"source": source
|
|
}
|
|
return self.call("listchannels", payload)
|
|
|
|
def listconfigs(self, config=None):
|
|
"""List this node's config.
|
|
"""
|
|
payload = {
|
|
"config": config
|
|
}
|
|
return self.call("listconfigs", payload)
|
|
|
|
def listforwards(self):
|
|
"""List all forwarded payments and their information.
|
|
"""
|
|
return self.call("listforwards")
|
|
|
|
def listfunds(self, spent=False):
|
|
"""
|
|
Show funds available for opening channels
|
|
or both unspent and spent funds if {spent} is True.
|
|
"""
|
|
|
|
payload = {
|
|
"spent": spent
|
|
}
|
|
return self.call("listfunds", payload)
|
|
|
|
def listtransactions(self):
|
|
"""
|
|
Show wallet history.
|
|
"""
|
|
return self.call("listtransactions")
|
|
|
|
def listinvoices(self, label=None):
|
|
"""
|
|
Show invoice {label} (or all, if no {label)).
|
|
"""
|
|
payload = {
|
|
"label": label
|
|
}
|
|
return self.call("listinvoices", payload)
|
|
|
|
def listnodes(self, node_id=None):
|
|
"""
|
|
Show all nodes in our local network view, filter on node {id}
|
|
if provided.
|
|
"""
|
|
payload = {
|
|
"id": node_id
|
|
}
|
|
return self.call("listnodes", payload)
|
|
|
|
def listpays(self, bolt11=None, payment_hash=None):
|
|
"""
|
|
Show outgoing payments, regarding {bolt11} or {payment_hash} if set
|
|
Can only specify one of {bolt11} or {payment_hash}.
|
|
"""
|
|
assert not (bolt11 and payment_hash)
|
|
payload = {
|
|
"bolt11": bolt11,
|
|
"payment_hash": payment_hash
|
|
}
|
|
return self.call("listpays", payload)
|
|
|
|
def listpeers(self, peerid=None, level=None):
|
|
"""
|
|
Show current peers, if {level} is set, include {log}s".
|
|
"""
|
|
payload = {
|
|
"id": peerid,
|
|
"level": level,
|
|
}
|
|
return self.call("listpeers", payload)
|
|
|
|
def listsendpays(self, bolt11=None, payment_hash=None):
|
|
"""Show all sendpays results, or only for `bolt11` or `payment_hash`."""
|
|
payload = {
|
|
"bolt11": bolt11,
|
|
"payment_hash": payment_hash
|
|
}
|
|
return self.call("listsendpays", payload)
|
|
|
|
def multifundchannel(self, destinations, feerate=None, minconf=None, utxos=None, minchannels=None, **kwargs):
|
|
"""
|
|
Fund channels to an array of {destinations},
|
|
each entry of which is a dict of node {id}
|
|
and {amount} to fund, and optionally whether
|
|
to {announce} and how much {push_msat} to
|
|
give outright to the node.
|
|
You may optionally specify {feerate},
|
|
{minconf} depth, and the {utxos} set to use
|
|
for the single transaction that funds all
|
|
the channels.
|
|
"""
|
|
payload = {
|
|
"destinations": destinations,
|
|
"feerate": feerate,
|
|
"minconf": minconf,
|
|
"utxos": utxos,
|
|
"minchannels": minchannels,
|
|
}
|
|
payload.update({k: v for k, v in kwargs.items()})
|
|
return self.call("multifundchannel", payload)
|
|
|
|
def multiwithdraw(self, outputs, feerate=None, minconf=None, utxos=None, **kwargs):
|
|
"""
|
|
Send to {outputs}
|
|
via Bitcoin transaction. Only select outputs
|
|
with {minconf} confirmations.
|
|
"""
|
|
payload = {
|
|
"outputs": outputs,
|
|
"feerate": feerate,
|
|
"minconf": minconf,
|
|
"utxos": utxos,
|
|
}
|
|
payload.update({k: v for k, v in kwargs.items()})
|
|
return self.call("multiwithdraw", payload)
|
|
|
|
def newaddr(self, addresstype=None):
|
|
"""Get a new address of type {addresstype} of the internal wallet.
|
|
"""
|
|
return self.call("newaddr", {"addresstype": addresstype})
|
|
|
|
def pay(self, bolt11, msatoshi=None, label=None, riskfactor=None,
|
|
description=None, maxfeepercent=None, retry_for=None,
|
|
maxdelay=None, exemptfee=None):
|
|
"""
|
|
Send payment specified by {bolt11} with {msatoshi}
|
|
(ignored if {bolt11} has an amount), optional {label}
|
|
and {riskfactor} (default 1.0).
|
|
"""
|
|
payload = {
|
|
"bolt11": bolt11,
|
|
"msatoshi": msatoshi,
|
|
"label": label,
|
|
"riskfactor": riskfactor,
|
|
"maxfeepercent": maxfeepercent,
|
|
"retry_for": retry_for,
|
|
"maxdelay": maxdelay,
|
|
"exemptfee": exemptfee,
|
|
# Deprecated.
|
|
"description": description,
|
|
}
|
|
return self.call("pay", payload)
|
|
|
|
def openchannel_init(self, node_id, channel_amount, psbt, feerate=None, funding_feerate=None, announce=True, close_to=None, *args, **kwargs):
|
|
"""Initiate an openchannel with a peer """
|
|
payload = {
|
|
"id": node_id,
|
|
"amount": channel_amount,
|
|
"initialpsbt": psbt,
|
|
"commitment_feerate": feerate,
|
|
"funding_feerate": funding_feerate,
|
|
"announce": announce,
|
|
"close_to": close_to,
|
|
}
|
|
return self.call("openchannel_init", payload)
|
|
|
|
def openchannel_signed(self, channel_id, signed_psbt, *args, **kwargs):
|
|
""" Send the funding transaction signatures to the peer, finish
|
|
the channel open """
|
|
payload = {
|
|
"channel_id": channel_id,
|
|
"signed_psbt": signed_psbt,
|
|
}
|
|
return self.call("openchannel_signed", payload)
|
|
|
|
def openchannel_update(self, channel_id, psbt, *args, **kwargs):
|
|
"""Update an openchannel with a peer """
|
|
payload = {
|
|
"channel_id": channel_id,
|
|
"psbt": psbt,
|
|
}
|
|
return self.call("openchannel_update", payload)
|
|
|
|
def paystatus(self, bolt11=None):
|
|
"""Detail status of attempts to pay {bolt11} or any."""
|
|
payload = {
|
|
"bolt11": bolt11
|
|
}
|
|
return self.call("paystatus", payload)
|
|
|
|
def ping(self, peer_id, length=128, pongbytes=128):
|
|
"""
|
|
Send {peer_id} a ping of length {len} asking for {pongbytes}.
|
|
"""
|
|
payload = {
|
|
"id": peer_id,
|
|
"len": length,
|
|
"pongbytes": pongbytes
|
|
}
|
|
return self.call("ping", payload)
|
|
|
|
def plugin_start(self, plugin, **kwargs):
|
|
"""
|
|
Adds a plugin to lightningd.
|
|
"""
|
|
payload = {
|
|
"subcommand": "start",
|
|
"plugin": plugin,
|
|
}
|
|
payload.update({k: v for k, v in kwargs.items()})
|
|
return self.call("plugin", payload)
|
|
|
|
def plugin_startdir(self, directory):
|
|
"""
|
|
Adds all plugins from a directory to lightningd.
|
|
"""
|
|
payload = {
|
|
"subcommand": "startdir",
|
|
"directory": directory
|
|
}
|
|
return self.call("plugin", payload)
|
|
|
|
def plugin_stop(self, plugin):
|
|
"""
|
|
Stops a lightningd plugin, will fail if plugin is not dynamic.
|
|
"""
|
|
payload = {
|
|
"subcommand": "stop",
|
|
"plugin": plugin
|
|
}
|
|
return self.call("plugin", payload)
|
|
|
|
def plugin_list(self):
|
|
"""
|
|
Lists all plugins lightningd knows about.
|
|
"""
|
|
payload = {
|
|
"subcommand": "list"
|
|
}
|
|
return self.call("plugin", payload)
|
|
|
|
def plugin_rescan(self):
|
|
payload = {
|
|
"subcommand": "rescan"
|
|
}
|
|
return self.call("plugin", payload)
|
|
|
|
def _deprecated_sendpay(self, route, payment_hash, description, msatoshi=None):
|
|
warnings.warn("sendpay: the 'description' field is renamed 'label' : expect removal"
|
|
" in early-2020",
|
|
DeprecationWarning)
|
|
payload = {
|
|
"route": route,
|
|
"payment_hash": payment_hash,
|
|
"label": description,
|
|
"msatoshi": msatoshi,
|
|
}
|
|
return self.call("sendpay", payload)
|
|
|
|
def sendpay(self, route, payment_hash, *args, **kwargs):
|
|
"""
|
|
Send along {route} in return for preimage of {payment_hash}.
|
|
"""
|
|
|
|
if 'description' in kwargs:
|
|
return self._deprecated_sendpay(route, payment_hash, *args, **kwargs)
|
|
|
|
def _sendpay(route, payment_hash, label=None, msatoshi=None, bolt11=None, payment_secret=None, partid=None):
|
|
payload = {
|
|
"route": route,
|
|
"payment_hash": payment_hash,
|
|
"label": label,
|
|
"msatoshi": msatoshi,
|
|
"bolt11": bolt11,
|
|
"payment_secret": payment_secret,
|
|
"partid": partid,
|
|
}
|
|
return self.call("sendpay", payload)
|
|
|
|
return _sendpay(route, payment_hash, *args, **kwargs)
|
|
|
|
def setchannelfee(self, id, base=None, ppm=None):
|
|
"""
|
|
Set routing fees for a channel/peer {id} (or 'all'). {base} is a value in millisatoshi
|
|
that is added as base fee to any routed payment. {ppm} is a value added proportionally
|
|
per-millionths to any routed payment volume in satoshi.
|
|
"""
|
|
payload = {
|
|
"id": id,
|
|
"base": base,
|
|
"ppm": ppm
|
|
}
|
|
return self.call("setchannelfee", payload)
|
|
|
|
def stop(self):
|
|
"""
|
|
Shut down the lightningd process.
|
|
"""
|
|
return self.call("stop")
|
|
|
|
def waitanyinvoice(self, lastpay_index=None, timeout=None, **kwargs):
|
|
"""
|
|
Wait for the next invoice to be paid, after {lastpay_index}
|
|
(if supplied).
|
|
Fail after {timeout} seconds has passed without an invoice
|
|
being paid.
|
|
"""
|
|
payload = {
|
|
"lastpay_index": lastpay_index,
|
|
"timeout": timeout
|
|
}
|
|
payload.update({k: v for k, v in kwargs.items()})
|
|
return self.call("waitanyinvoice", payload)
|
|
|
|
def waitblockheight(self, blockheight, timeout=None):
|
|
"""
|
|
Wait for the blockchain to reach the specified block height.
|
|
"""
|
|
payload = {
|
|
"blockheight": blockheight,
|
|
"timeout": timeout
|
|
}
|
|
return self.call("waitblockheight", payload)
|
|
|
|
def waitinvoice(self, label):
|
|
"""
|
|
Wait for an incoming payment matching the invoice with {label}.
|
|
"""
|
|
payload = {
|
|
"label": label
|
|
}
|
|
return self.call("waitinvoice", payload)
|
|
|
|
def waitsendpay(self, payment_hash, timeout=None, partid=None):
|
|
"""
|
|
Wait for payment for preimage of {payment_hash} to complete.
|
|
"""
|
|
payload = {
|
|
"payment_hash": payment_hash,
|
|
"timeout": timeout,
|
|
"partid": partid,
|
|
}
|
|
return self.call("waitsendpay", payload)
|
|
|
|
def withdraw(self, destination, satoshi, feerate=None, minconf=None, utxos=None):
|
|
"""
|
|
Send to {destination} address {satoshi} (or "all")
|
|
amount via Bitcoin transaction. Only select outputs
|
|
with {minconf} confirmations.
|
|
"""
|
|
payload = {
|
|
"destination": destination,
|
|
"satoshi": satoshi,
|
|
"feerate": feerate,
|
|
"minconf": minconf,
|
|
"utxos": utxos,
|
|
}
|
|
|
|
return self.call("withdraw", payload)
|
|
|
|
def txprepare(self, outputs, feerate=None, minconf=None, utxos=None):
|
|
"""
|
|
Prepare a Bitcoin transaction which sends to [outputs].
|
|
The format of output is like [{address1: amount1},
|
|
{address2: amount2}], or [{address: "all"}]).
|
|
Only select outputs with {minconf} confirmations.
|
|
|
|
Outputs will be reserved until you call txdiscard or txsend, or
|
|
lightningd restarts.
|
|
"""
|
|
payload = {
|
|
"outputs": outputs,
|
|
"feerate": feerate,
|
|
"minconf": minconf,
|
|
"utxos": utxos,
|
|
}
|
|
return self.call("txprepare", payload)
|
|
|
|
def txdiscard(self, txid):
|
|
"""
|
|
Cancel a Bitcoin transaction returned from txprepare. The outputs
|
|
it was spending are released for other use.
|
|
"""
|
|
payload = {
|
|
"txid": txid
|
|
}
|
|
return self.call("txdiscard", payload)
|
|
|
|
def txsend(self, txid):
|
|
"""
|
|
Sign and broadcast a Bitcoin transaction returned from txprepare.
|
|
"""
|
|
payload = {
|
|
"txid": txid
|
|
}
|
|
return self.call("txsend", payload)
|
|
|
|
def reserveinputs(self, psbt, exclusive=True):
|
|
"""
|
|
Reserve any inputs in this psbt.
|
|
"""
|
|
payload = {
|
|
"psbt": psbt,
|
|
"exclusive": exclusive,
|
|
}
|
|
return self.call("reserveinputs", payload)
|
|
|
|
def unreserveinputs(self, psbt):
|
|
"""
|
|
Unreserve (or reduce reservation) on any UTXOs in this psbt were previously reserved.
|
|
"""
|
|
payload = {
|
|
"psbt": psbt,
|
|
}
|
|
return self.call("unreserveinputs", payload)
|
|
|
|
def fundpsbt(self, satoshi, feerate, startweight, minconf=None, reserve=True, locktime=None):
|
|
"""
|
|
Create a PSBT with inputs sufficient to give an output of satoshi.
|
|
"""
|
|
payload = {
|
|
"satoshi": satoshi,
|
|
"feerate": feerate,
|
|
"startweight": startweight,
|
|
"minconf": minconf,
|
|
"reserve": reserve,
|
|
"locktime": locktime,
|
|
}
|
|
return self.call("fundpsbt", payload)
|
|
|
|
def utxopsbt(self, satoshi, feerate, startweight, utxos, reserve=True, reservedok=False, locktime=None):
|
|
"""
|
|
Create a PSBT with given inputs, to give an output of satoshi.
|
|
"""
|
|
payload = {
|
|
"satoshi": satoshi,
|
|
"feerate": feerate,
|
|
"startweight": startweight,
|
|
"utxos": utxos,
|
|
"reserve": reserve,
|
|
"reservedok": reservedok,
|
|
"locktime": locktime,
|
|
}
|
|
return self.call("utxopsbt", payload)
|
|
|
|
def signpsbt(self, psbt, signonly=None):
|
|
"""
|
|
Add internal wallet's signatures to PSBT
|
|
"""
|
|
payload = {
|
|
"psbt": psbt,
|
|
"signonly": signonly,
|
|
}
|
|
return self.call("signpsbt", payload)
|
|
|
|
def sendpsbt(self, psbt):
|
|
"""
|
|
Finalize extract and broadcast a PSBT
|
|
"""
|
|
payload = {
|
|
"psbt": psbt,
|
|
}
|
|
return self.call("sendpsbt", payload)
|
|
|
|
def signmessage(self, message):
|
|
"""
|
|
Sign a message with this node's secret key.
|
|
"""
|
|
payload = {
|
|
"message": message
|
|
}
|
|
return self.call("signmessage", payload)
|
|
|
|
def checkmessage(self, message, zbase, pubkey=None):
|
|
"""
|
|
Check if a message was signed (with a specific key).
|
|
Use returned field ['verified'] to get result.
|
|
"""
|
|
payload = {
|
|
"message": message,
|
|
"zbase": zbase,
|
|
"pubkey": pubkey,
|
|
}
|
|
return self.call("checkmessage", payload)
|
|
|
|
def getsharedsecret(self, point, **kwargs):
|
|
"""
|
|
Compute the hash of the Elliptic Curve Diffie Hellman shared
|
|
secret point from this node private key and an
|
|
input {point}.
|
|
"""
|
|
payload = {
|
|
"point": point
|
|
}
|
|
payload.update({k: v for k, v in kwargs.items()})
|
|
return self.call("getsharedsecret", payload)
|