2015-12-26 09:09:17 +03:00
|
|
|
package main
|
2015-12-30 02:09:38 +03:00
|
|
|
|
|
|
|
import (
|
2016-09-21 03:15:26 +03:00
|
|
|
"bytes"
|
2016-06-21 21:52:09 +03:00
|
|
|
"encoding/hex"
|
2015-12-31 08:40:41 +03:00
|
|
|
"fmt"
|
2016-07-13 03:46:25 +03:00
|
|
|
"io"
|
2016-09-19 22:04:56 +03:00
|
|
|
"time"
|
2016-01-16 22:09:12 +03:00
|
|
|
|
2016-01-17 06:12:36 +03:00
|
|
|
"sync"
|
|
|
|
"sync/atomic"
|
2015-12-30 02:09:38 +03:00
|
|
|
|
2016-09-21 03:15:26 +03:00
|
|
|
"github.com/BitfuryLightning/tools/rt/graph"
|
2016-09-19 22:04:56 +03:00
|
|
|
"github.com/btcsuite/fastsha256"
|
2016-09-21 03:15:26 +03:00
|
|
|
"github.com/davecgh/go-spew/spew"
|
|
|
|
"github.com/lightningnetwork/lightning-onion"
|
2016-09-19 22:04:56 +03:00
|
|
|
"github.com/lightningnetwork/lnd/channeldb"
|
2016-01-17 21:45:07 +03:00
|
|
|
"github.com/lightningnetwork/lnd/lndc"
|
2016-01-16 21:38:48 +03:00
|
|
|
"github.com/lightningnetwork/lnd/lnrpc"
|
2016-08-13 01:53:18 +03:00
|
|
|
"github.com/lightningnetwork/lnd/lnwallet"
|
2016-07-13 03:46:25 +03:00
|
|
|
"github.com/lightningnetwork/lnd/lnwire"
|
2016-09-21 03:15:26 +03:00
|
|
|
"github.com/roasbeef/btcd/btcec"
|
2016-05-15 17:17:44 +03:00
|
|
|
"github.com/roasbeef/btcd/txscript"
|
|
|
|
"github.com/roasbeef/btcd/wire"
|
|
|
|
"github.com/roasbeef/btcutil"
|
|
|
|
"github.com/roasbeef/btcwallet/waddrmgr"
|
2015-12-30 02:09:38 +03:00
|
|
|
"golang.org/x/net/context"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
defaultAccount uint32 = waddrmgr.DefaultAccountNum
|
|
|
|
)
|
|
|
|
|
2016-06-21 21:52:09 +03:00
|
|
|
// rpcServer is a gRPC, RPC front end to the lnd daemon.
|
2016-01-17 06:07:44 +03:00
|
|
|
type rpcServer struct {
|
|
|
|
started int32 // To be used atomically.
|
|
|
|
shutdown int32 // To be used atomically.
|
|
|
|
|
|
|
|
server *server
|
|
|
|
|
|
|
|
wg sync.WaitGroup
|
|
|
|
|
|
|
|
quit chan struct{}
|
2015-12-30 02:09:38 +03:00
|
|
|
}
|
|
|
|
|
2016-06-21 21:52:09 +03:00
|
|
|
// A compile time check to ensure that rpcServer fully implements the
|
|
|
|
// LightningServer gRPC service.
|
2015-12-30 02:21:42 +03:00
|
|
|
var _ lnrpc.LightningServer = (*rpcServer)(nil)
|
|
|
|
|
2016-06-21 21:52:09 +03:00
|
|
|
// newRpcServer creates and returns a new instance of the rpcServer.
|
2016-01-17 06:07:44 +03:00
|
|
|
func newRpcServer(s *server) *rpcServer {
|
|
|
|
return &rpcServer{server: s, quit: make(chan struct{}, 1)}
|
2015-12-30 02:09:38 +03:00
|
|
|
}
|
|
|
|
|
2016-06-21 21:52:09 +03:00
|
|
|
// Start launches any helper goroutines required for the rpcServer
|
|
|
|
// to function.
|
2016-01-17 06:12:36 +03:00
|
|
|
func (r *rpcServer) Start() error {
|
|
|
|
if atomic.AddInt32(&r.started, 1) != 1 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-01-14 08:41:46 +03:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-06-21 21:52:09 +03:00
|
|
|
// Stop signals any active goroutines for a graceful closure.
|
2016-01-17 06:12:36 +03:00
|
|
|
func (r *rpcServer) Stop() error {
|
|
|
|
if atomic.AddInt32(&r.shutdown, 1) != 1 {
|
|
|
|
return nil
|
2016-01-02 07:27:40 +03:00
|
|
|
}
|
2016-01-17 06:12:36 +03:00
|
|
|
|
2016-04-25 06:26:32 +03:00
|
|
|
close(r.quit)
|
|
|
|
|
2016-01-17 06:12:36 +03:00
|
|
|
return nil
|
2016-01-02 07:27:40 +03:00
|
|
|
}
|
|
|
|
|
2016-06-29 21:31:29 +03:00
|
|
|
// addrPairsToOutputs converts a map describing a set of outputs to be created,
|
|
|
|
// the outputs themselves. The passed map pairs up an address, to a desired
|
|
|
|
// output value amount. Each address is converted to its corresponding pkScript
|
|
|
|
// to be used within the constructed output(s).
|
|
|
|
func addrPairsToOutputs(addrPairs map[string]int64) ([]*wire.TxOut, error) {
|
|
|
|
outputs := make([]*wire.TxOut, 0, len(addrPairs))
|
|
|
|
for addr, amt := range addrPairs {
|
2016-07-14 04:37:50 +03:00
|
|
|
addr, err := btcutil.DecodeAddress(addr, activeNetParams.Params)
|
2016-03-23 04:48:46 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
pkscript, err := txscript.PayToAddrScript(addr)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
outputs = append(outputs, wire.NewTxOut(amt, pkscript))
|
2015-12-30 02:09:38 +03:00
|
|
|
}
|
|
|
|
|
2016-06-29 21:31:29 +03:00
|
|
|
return outputs, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// sendCoinsOnChain makes an on-chain transaction in or to send coins to one or
|
|
|
|
// more addresses specified in the passed payment map. The payment map maps an
|
|
|
|
// address to a specified output value to be sent to that address.
|
|
|
|
func (r *rpcServer) sendCoinsOnChain(paymentMap map[string]int64) (*wire.ShaHash, error) {
|
|
|
|
outputs, err := addrPairsToOutputs(paymentMap)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2016-08-13 01:53:18 +03:00
|
|
|
return r.server.lnwallet.SendOutputs(outputs)
|
2016-06-29 21:31:29 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// SendCoins executes a request to send coins to a particular address. Unlike
|
|
|
|
// SendMany, this RPC call only allows creating a single output at a time.
|
|
|
|
func (r *rpcServer) SendCoins(ctx context.Context,
|
|
|
|
in *lnrpc.SendCoinsRequest) (*lnrpc.SendCoinsResponse, error) {
|
|
|
|
|
|
|
|
rpcsLog.Infof("[sendcoins] addr=%v, amt=%v", in.Addr, btcutil.Amount(in.Amount))
|
|
|
|
|
|
|
|
paymentMap := map[string]int64{in.Addr: in.Amount}
|
|
|
|
txid, err := r.sendCoinsOnChain(paymentMap)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
rpcsLog.Infof("[sendcoins] spend generated txid: %v", txid.String())
|
|
|
|
|
|
|
|
return &lnrpc.SendCoinsResponse{Txid: txid.String()}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// SendMany handles a request for a transaction create multiple specified
|
|
|
|
// outputs in parallel.
|
|
|
|
func (r *rpcServer) SendMany(ctx context.Context,
|
|
|
|
in *lnrpc.SendManyRequest) (*lnrpc.SendManyResponse, error) {
|
|
|
|
|
|
|
|
txid, err := r.sendCoinsOnChain(in.AddrToAmount)
|
2015-12-30 02:09:38 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2016-06-29 21:31:29 +03:00
|
|
|
rpcsLog.Infof("[sendmany] spend generated txid: %v", txid.String())
|
2016-04-25 06:26:32 +03:00
|
|
|
|
|
|
|
return &lnrpc.SendManyResponse{Txid: txid.String()}, nil
|
2015-12-30 02:09:38 +03:00
|
|
|
}
|
|
|
|
|
2016-06-21 21:52:09 +03:00
|
|
|
// NewAddress creates a new address under control of the local wallet.
|
2016-04-25 06:26:32 +03:00
|
|
|
func (r *rpcServer) NewAddress(ctx context.Context,
|
|
|
|
in *lnrpc.NewAddressRequest) (*lnrpc.NewAddressResponse, error) {
|
2015-12-30 02:09:38 +03:00
|
|
|
|
2016-04-25 06:26:32 +03:00
|
|
|
// Translate the gRPC proto address type to the wallet controller's
|
|
|
|
// available address types.
|
2016-08-13 01:53:18 +03:00
|
|
|
var addrType lnwallet.AddressType
|
2016-04-25 06:26:32 +03:00
|
|
|
switch in.Type {
|
|
|
|
case lnrpc.NewAddressRequest_WITNESS_PUBKEY_HASH:
|
2016-08-13 01:53:18 +03:00
|
|
|
addrType = lnwallet.WitnessPubKey
|
2016-04-25 06:26:32 +03:00
|
|
|
case lnrpc.NewAddressRequest_NESTED_PUBKEY_HASH:
|
2016-08-13 01:53:18 +03:00
|
|
|
addrType = lnwallet.NestedWitnessPubKey
|
2016-04-25 06:26:32 +03:00
|
|
|
case lnrpc.NewAddressRequest_PUBKEY_HASH:
|
2016-08-13 01:53:18 +03:00
|
|
|
addrType = lnwallet.PubKeyHash
|
2016-04-25 06:26:32 +03:00
|
|
|
}
|
|
|
|
|
2016-08-13 01:53:18 +03:00
|
|
|
addr, err := r.server.lnwallet.NewAddress(addrType, false)
|
2015-12-30 02:09:38 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2016-06-29 21:31:29 +03:00
|
|
|
rpcsLog.Infof("[newaddress] addr=%v", addr.String())
|
2015-12-30 02:09:38 +03:00
|
|
|
return &lnrpc.NewAddressResponse{Address: addr.String()}, nil
|
|
|
|
}
|
2015-12-31 06:56:57 +03:00
|
|
|
|
2016-06-21 21:52:09 +03:00
|
|
|
// ConnectPeer attempts to establish a connection to a remote peer.
|
2016-01-17 06:12:36 +03:00
|
|
|
func (r *rpcServer) ConnectPeer(ctx context.Context,
|
|
|
|
in *lnrpc.ConnectPeerRequest) (*lnrpc.ConnectPeerResponse, error) {
|
2015-12-31 06:56:57 +03:00
|
|
|
|
2016-06-21 21:52:09 +03:00
|
|
|
if in.Addr == nil {
|
2016-01-02 07:27:40 +03:00
|
|
|
return nil, fmt.Errorf("need: lnc pubkeyhash@hostname")
|
|
|
|
}
|
|
|
|
|
2016-06-21 21:52:09 +03:00
|
|
|
idAtHost := fmt.Sprintf("%v@%v", in.Addr.PubKeyHash, in.Addr.Host)
|
2016-06-29 21:31:29 +03:00
|
|
|
rpcsLog.Debugf("[connectpeer] peer=%v", idAtHost)
|
2016-06-21 21:52:09 +03:00
|
|
|
|
2016-07-17 04:00:52 +03:00
|
|
|
peerAddr, err := lndc.LnAddrFromString(idAtHost, activeNetParams.Params)
|
2016-06-21 21:52:09 +03:00
|
|
|
if err != nil {
|
|
|
|
rpcsLog.Errorf("(connectpeer): error parsing ln addr: %v", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
peerID, err := r.server.ConnectToPeer(peerAddr)
|
|
|
|
if err != nil {
|
|
|
|
rpcsLog.Errorf("(connectpeer): error connecting to peer: %v", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
rpcsLog.Debugf("Connected to peer: %v", peerAddr.String())
|
|
|
|
return &lnrpc.ConnectPeerResponse{peerID}, nil
|
|
|
|
}
|
2016-06-21 22:32:32 +03:00
|
|
|
|
|
|
|
// OpenChannel attempts to open a singly funded channel specified in the
|
|
|
|
// request to a remote peer.
|
2016-07-08 01:30:55 +03:00
|
|
|
func (r *rpcServer) OpenChannel(in *lnrpc.OpenChannelRequest,
|
|
|
|
updateStream lnrpc.Lightning_OpenChannelServer) error {
|
2016-06-21 22:32:32 +03:00
|
|
|
|
2016-06-29 21:31:29 +03:00
|
|
|
rpcsLog.Tracef("[openchannel] request to peerid(%v) "+
|
2016-06-21 22:32:32 +03:00
|
|
|
"allocation(us=%v, them=%v) numconfs=%v", in.TargetPeerId,
|
|
|
|
in.LocalFundingAmount, in.RemoteFundingAmount, in.NumConfs)
|
|
|
|
|
|
|
|
localFundingAmt := btcutil.Amount(in.LocalFundingAmount)
|
|
|
|
remoteFundingAmt := btcutil.Amount(in.RemoteFundingAmount)
|
2016-09-14 01:35:41 +03:00
|
|
|
updateChan, errChan := r.server.OpenChannel(in.TargetPeerId,
|
|
|
|
in.TargetNode, localFundingAmt, remoteFundingAmt, in.NumConfs)
|
2016-01-02 07:27:40 +03:00
|
|
|
|
2016-08-31 02:52:53 +03:00
|
|
|
var outpoint wire.OutPoint
|
|
|
|
out:
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case err := <-errChan:
|
2016-09-14 01:35:41 +03:00
|
|
|
rpcsLog.Errorf("unable to open channel to "+
|
2016-09-21 02:07:22 +03:00
|
|
|
"lightningID(%x) nor peerID(%v): %v",
|
2016-09-14 01:35:41 +03:00
|
|
|
in.TargetNode, in.TargetPeerId, err)
|
2016-07-08 01:30:55 +03:00
|
|
|
return err
|
2016-08-31 02:52:53 +03:00
|
|
|
case fundingUpdate := <-updateChan:
|
|
|
|
rpcsLog.Tracef("[openchannel] sending update: %v",
|
|
|
|
fundingUpdate)
|
|
|
|
if err := updateStream.Send(fundingUpdate); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// If a final channel open update is being sent, then
|
|
|
|
// we can break out of our recv loop as we no longer
|
|
|
|
// need to process any further updates.
|
|
|
|
switch update := fundingUpdate.Update.(type) {
|
|
|
|
case *lnrpc.OpenStatusUpdate_ChanOpen:
|
|
|
|
chanPoint := update.ChanOpen.ChannelPoint
|
|
|
|
h, _ := wire.NewShaHash(chanPoint.FundingTxid)
|
|
|
|
outpoint = wire.OutPoint{
|
|
|
|
Hash: *h,
|
|
|
|
Index: chanPoint.OutputIndex,
|
|
|
|
}
|
|
|
|
|
|
|
|
break out
|
|
|
|
}
|
|
|
|
case <-r.quit:
|
|
|
|
return nil
|
2016-07-08 01:30:55 +03:00
|
|
|
}
|
|
|
|
}
|
2016-08-31 02:52:53 +03:00
|
|
|
|
2016-06-29 21:31:29 +03:00
|
|
|
rpcsLog.Tracef("[openchannel] success peerid(%v), ChannelPoint(%v)",
|
2016-07-08 01:30:55 +03:00
|
|
|
in.TargetPeerId, outpoint)
|
|
|
|
return nil
|
2016-06-21 22:32:32 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// CloseChannel attempts to close an active channel identified by its channel
|
|
|
|
// point. The actions of this method can additionally be augmented to attempt
|
|
|
|
// a force close after a timeout period in the case of an inactive peer.
|
2016-07-08 01:30:55 +03:00
|
|
|
func (r *rpcServer) CloseChannel(in *lnrpc.CloseChannelRequest,
|
|
|
|
updateStream lnrpc.Lightning_CloseChannelServer) error {
|
2016-06-21 22:32:32 +03:00
|
|
|
|
2016-09-12 22:42:26 +03:00
|
|
|
force := in.Force
|
2016-06-21 22:32:32 +03:00
|
|
|
index := in.ChannelPoint.OutputIndex
|
|
|
|
txid, err := wire.NewShaHash(in.ChannelPoint.FundingTxid)
|
|
|
|
if err != nil {
|
2016-06-29 21:31:29 +03:00
|
|
|
rpcsLog.Errorf("[closechannel] invalid txid: %v", err)
|
2016-07-08 01:30:55 +03:00
|
|
|
return err
|
2016-01-02 07:27:40 +03:00
|
|
|
}
|
2016-06-21 22:32:32 +03:00
|
|
|
targetChannelPoint := wire.NewOutPoint(txid, index)
|
|
|
|
|
2016-06-29 21:31:29 +03:00
|
|
|
rpcsLog.Tracef("[closechannel] request for ChannelPoint(%v)",
|
2016-06-21 22:32:32 +03:00
|
|
|
targetChannelPoint)
|
|
|
|
|
2016-09-12 22:42:26 +03:00
|
|
|
updateChan, errChan := r.server.htlcSwitch.CloseLink(targetChannelPoint, force)
|
2016-07-08 01:30:55 +03:00
|
|
|
|
2016-08-31 02:52:53 +03:00
|
|
|
out:
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case err := <-errChan:
|
|
|
|
rpcsLog.Errorf("[closechannel] unable to close "+
|
|
|
|
"ChannelPoint(%v): %v", targetChannelPoint, err)
|
2016-07-08 01:30:55 +03:00
|
|
|
return err
|
2016-08-31 02:52:53 +03:00
|
|
|
case closingUpdate := <-updateChan:
|
2016-08-31 05:34:13 +03:00
|
|
|
rpcsLog.Tracef("[closechannel] sending update: %v",
|
|
|
|
closingUpdate)
|
2016-08-31 02:52:53 +03:00
|
|
|
if err := updateStream.Send(closingUpdate); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// If a final channel closing updates is being sent,
|
|
|
|
// then we can break out of our dispatch loop as we no
|
|
|
|
// longer need to process any further updates.
|
|
|
|
switch closeUpdate := closingUpdate.Update.(type) {
|
|
|
|
case *lnrpc.CloseStatusUpdate_ChanClose:
|
|
|
|
h, _ := wire.NewShaHash(closeUpdate.ChanClose.ClosingTxid)
|
2016-09-12 22:42:26 +03:00
|
|
|
rpcsLog.Infof("[closechannel] close completed: "+
|
2016-08-31 02:52:53 +03:00
|
|
|
"txid(%v)", h)
|
|
|
|
break out
|
|
|
|
}
|
|
|
|
case <-r.quit:
|
|
|
|
return nil
|
2016-07-08 01:30:55 +03:00
|
|
|
}
|
2016-06-21 22:32:32 +03:00
|
|
|
}
|
|
|
|
|
2016-07-08 01:30:55 +03:00
|
|
|
return nil
|
2016-06-21 22:32:32 +03:00
|
|
|
}
|
2016-01-02 07:27:40 +03:00
|
|
|
|
2016-07-06 04:57:08 +03:00
|
|
|
// GetInfo serves a request to the "getinfo" RPC call. This call returns
|
|
|
|
// general information concerning the lightning node including it's LN ID,
|
|
|
|
// identity address, and information concerning the number of open+pending
|
|
|
|
// channels.
|
|
|
|
func (r *rpcServer) GetInfo(ctx context.Context,
|
|
|
|
in *lnrpc.GetInfoRequest) (*lnrpc.GetInfoResponse, error) {
|
|
|
|
|
|
|
|
var activeChannels uint32
|
|
|
|
serverPeers := r.server.Peers()
|
|
|
|
for _, serverPeer := range serverPeers {
|
|
|
|
activeChannels += uint32(len(serverPeer.ChannelSnapshots()))
|
|
|
|
}
|
|
|
|
|
|
|
|
pendingChannels := r.server.fundingMgr.NumPendingChannels()
|
|
|
|
|
|
|
|
idPub := r.server.identityPriv.PubKey().SerializeCompressed()
|
2016-07-14 04:37:50 +03:00
|
|
|
idAddr, err := btcutil.NewAddressPubKeyHash(btcutil.Hash160(idPub), activeNetParams.Params)
|
2016-07-06 04:57:08 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &lnrpc.GetInfoResponse{
|
|
|
|
LightningId: hex.EncodeToString(r.server.lightningID[:]),
|
2016-09-21 03:15:26 +03:00
|
|
|
IdentityPubkey: hex.EncodeToString(idPub),
|
2016-07-06 04:57:08 +03:00
|
|
|
IdentityAddress: idAddr.String(),
|
|
|
|
NumPendingChannels: pendingChannels,
|
|
|
|
NumActiveChannels: activeChannels,
|
|
|
|
NumPeers: uint32(len(serverPeers)),
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2016-06-21 21:52:09 +03:00
|
|
|
// ListPeers returns a verbose listing of all currently active peers.
|
|
|
|
func (r *rpcServer) ListPeers(ctx context.Context,
|
|
|
|
in *lnrpc.ListPeersRequest) (*lnrpc.ListPeersResponse, error) {
|
|
|
|
|
2016-06-29 21:31:29 +03:00
|
|
|
rpcsLog.Tracef("[listpeers] request")
|
2016-06-21 21:52:09 +03:00
|
|
|
|
|
|
|
serverPeers := r.server.Peers()
|
|
|
|
resp := &lnrpc.ListPeersResponse{
|
|
|
|
Peers: make([]*lnrpc.Peer, 0, len(serverPeers)),
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, serverPeer := range serverPeers {
|
|
|
|
// TODO(roasbeef): add a snapshot method which grabs peer read mtx
|
2016-06-23 08:22:40 +03:00
|
|
|
|
|
|
|
lnID := hex.EncodeToString(serverPeer.lightningID[:])
|
2016-06-21 21:52:09 +03:00
|
|
|
peer := &lnrpc.Peer{
|
2016-06-23 08:22:40 +03:00
|
|
|
LightningId: lnID,
|
2016-06-21 21:52:09 +03:00
|
|
|
PeerId: serverPeer.id,
|
|
|
|
Address: serverPeer.conn.RemoteAddr().String(),
|
|
|
|
Inbound: serverPeer.inbound,
|
|
|
|
BytesRecv: atomic.LoadUint64(&serverPeer.bytesReceived),
|
|
|
|
BytesSent: atomic.LoadUint64(&serverPeer.bytesSent),
|
|
|
|
}
|
|
|
|
|
2016-06-23 08:22:40 +03:00
|
|
|
chanSnapshots := serverPeer.ChannelSnapshots()
|
|
|
|
peer.Channels = make([]*lnrpc.ActiveChannel, 0, len(chanSnapshots))
|
|
|
|
for _, chanSnapshot := range chanSnapshots {
|
|
|
|
channel := &lnrpc.ActiveChannel{
|
|
|
|
RemoteId: lnID,
|
|
|
|
ChannelPoint: chanSnapshot.ChannelPoint.String(),
|
|
|
|
Capacity: int64(chanSnapshot.Capacity),
|
|
|
|
LocalBalance: int64(chanSnapshot.LocalBalance),
|
|
|
|
RemoteBalance: int64(chanSnapshot.RemoteBalance),
|
|
|
|
NumUpdates: chanSnapshot.NumUpdates,
|
|
|
|
}
|
|
|
|
peer.Channels = append(peer.Channels, channel)
|
|
|
|
}
|
|
|
|
|
2016-06-21 21:52:09 +03:00
|
|
|
resp.Peers = append(resp.Peers, peer)
|
|
|
|
}
|
|
|
|
|
2016-06-29 21:31:29 +03:00
|
|
|
rpcsLog.Debugf("[listpeers] yielded %v peers", serverPeers)
|
2016-06-21 21:52:09 +03:00
|
|
|
|
|
|
|
return resp, nil
|
|
|
|
}
|
|
|
|
|
2016-06-21 21:46:27 +03:00
|
|
|
// WalletBalance returns the sum of all confirmed unspent outputs under control
|
|
|
|
// by the wallet. This method can be modified by having the request specify
|
|
|
|
// only witness outputs should be factored into the final output sum.
|
|
|
|
// TODO(roasbeef): split into total and confirmed/unconfirmed
|
2016-08-31 02:52:53 +03:00
|
|
|
// TODO(roasbeef): add async hooks into wallet balance changes
|
2016-06-21 21:46:27 +03:00
|
|
|
func (r *rpcServer) WalletBalance(ctx context.Context,
|
|
|
|
in *lnrpc.WalletBalanceRequest) (*lnrpc.WalletBalanceResponse, error) {
|
|
|
|
|
2016-08-13 01:53:18 +03:00
|
|
|
balance, err := r.server.lnwallet.ConfirmedBalance(1, in.WitnessOnly)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2016-06-21 21:46:27 +03:00
|
|
|
}
|
|
|
|
|
2016-06-29 21:31:29 +03:00
|
|
|
rpcsLog.Debugf("[walletbalance] balance=%v", balance)
|
2016-06-21 21:46:27 +03:00
|
|
|
|
2016-09-06 22:44:05 +03:00
|
|
|
return &lnrpc.WalletBalanceResponse{balance.ToBTC()}, nil
|
2015-12-31 08:40:41 +03:00
|
|
|
}
|
2016-07-08 01:33:52 +03:00
|
|
|
|
2016-09-15 22:24:52 +03:00
|
|
|
// ChannelBalance returns the total available channel flow across all open
|
|
|
|
// channels in satoshis.
|
2016-09-15 21:59:51 +03:00
|
|
|
func (r *rpcServer) ChannelBalance(ctx context.Context,
|
|
|
|
in *lnrpc.ChannelBalanceRequest) (*lnrpc.ChannelBalanceResponse, error) {
|
|
|
|
|
|
|
|
var balance btcutil.Amount
|
|
|
|
for _, peer := range r.server.Peers() {
|
|
|
|
for _, snapshot := range peer.ChannelSnapshots() {
|
|
|
|
balance += snapshot.LocalBalance
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return &lnrpc.ChannelBalanceResponse{Balance: int64(balance)}, nil
|
|
|
|
}
|
|
|
|
|
2016-07-08 01:33:52 +03:00
|
|
|
// PendingChannels returns a list of all the channels that are currently
|
|
|
|
// considered "pending". A channel is pending if it has finished the funding
|
|
|
|
// workflow and is waiting for confirmations for the funding txn, or is in the
|
|
|
|
// process of closure, either initiated cooperatively or non-coopertively.
|
|
|
|
func (r *rpcServer) PendingChannels(ctx context.Context,
|
|
|
|
in *lnrpc.PendingChannelRequest) (*lnrpc.PendingChannelResponse, error) {
|
|
|
|
|
|
|
|
both := in.Status == lnrpc.ChannelStatus_ALL
|
|
|
|
includeOpen := (in.Status == lnrpc.ChannelStatus_OPENING) || both
|
|
|
|
includeClose := (in.Status == lnrpc.ChannelStatus_CLOSING) || both
|
|
|
|
rpcsLog.Debugf("[pendingchannels] %v", in.Status)
|
|
|
|
|
|
|
|
var pendingChannels []*lnrpc.PendingChannelResponse_PendingChannel
|
|
|
|
if includeOpen {
|
|
|
|
pendingOpenChans := r.server.fundingMgr.PendingChannels()
|
|
|
|
for _, pendingOpen := range pendingOpenChans {
|
|
|
|
// TODO(roasbeef): add confirmation progress
|
|
|
|
pendingChan := &lnrpc.PendingChannelResponse_PendingChannel{
|
|
|
|
PeerId: pendingOpen.peerId,
|
|
|
|
LightningId: hex.EncodeToString(pendingOpen.lightningID[:]),
|
|
|
|
ChannelPoint: pendingOpen.channelPoint.String(),
|
|
|
|
Capacity: int64(pendingOpen.capacity),
|
|
|
|
LocalBalance: int64(pendingOpen.localBalance),
|
|
|
|
RemoteBalance: int64(pendingOpen.remoteBalance),
|
|
|
|
Status: lnrpc.ChannelStatus_OPENING,
|
|
|
|
}
|
|
|
|
pendingChannels = append(pendingChannels, pendingChan)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if includeClose {
|
|
|
|
}
|
|
|
|
|
|
|
|
return &lnrpc.PendingChannelResponse{
|
|
|
|
PendingChannels: pendingChannels,
|
|
|
|
}, nil
|
|
|
|
}
|
2016-07-13 03:46:25 +03:00
|
|
|
|
|
|
|
// SendPayment dispatches a bi-directional streaming RPC for sending payments
|
|
|
|
// through the Lightning Network. A single RPC invocation creates a persistent
|
|
|
|
// bi-directional stream allowing clients to rapidly send payments through the
|
|
|
|
// Lightning Network with a single persistent connection.
|
|
|
|
func (r *rpcServer) SendPayment(paymentStream lnrpc.Lightning_SendPaymentServer) error {
|
2016-09-21 03:15:26 +03:00
|
|
|
queryTimeout := time.Duration(time.Minute)
|
2016-07-22 02:22:30 +03:00
|
|
|
errChan := make(chan error, 1)
|
2016-09-21 03:15:26 +03:00
|
|
|
|
2016-07-13 03:46:25 +03:00
|
|
|
for {
|
2016-07-22 02:22:30 +03:00
|
|
|
select {
|
|
|
|
case err := <-errChan:
|
2016-07-13 03:46:25 +03:00
|
|
|
return err
|
2016-07-22 02:22:30 +03:00
|
|
|
default:
|
|
|
|
// Receive the next pending payment within the stream sent by
|
|
|
|
// the client. If we read the EOF sentinel, then the client has
|
|
|
|
// closed the stream, and we can exit normally.
|
|
|
|
nextPayment, err := paymentStream.Recv()
|
|
|
|
if err == io.EOF {
|
|
|
|
return nil
|
|
|
|
} else if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2016-07-13 03:46:25 +03:00
|
|
|
|
2016-09-21 03:15:26 +03:00
|
|
|
// Query the routing table for a potential path to the
|
|
|
|
// destination node. If a path is ultimately
|
|
|
|
// unavailable, then an error will be returned.
|
|
|
|
destNode := hex.EncodeToString(nextPayment.Dest)
|
|
|
|
targetVertex := graph.NewID(destNode)
|
|
|
|
path, err := r.server.routingMgr.FindPath(targetVertex,
|
|
|
|
queryTimeout)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
rpcsLog.Tracef("[sendpayment] selected route: %v", path)
|
|
|
|
|
|
|
|
// Generate the raw encoded sphinx packet to be
|
|
|
|
// included along with the HTLC add message.
|
|
|
|
// We snip off the first hop from the path as within
|
|
|
|
// the routing table's star graph, we're always the
|
|
|
|
// first hop.
|
|
|
|
sphinxPacket, err := generateSphinxPacket(path[1:])
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2016-09-19 22:03:38 +03:00
|
|
|
// If we're in debug HTLC mode, then all outgoing
|
|
|
|
// HTLC's will pay to the same debug rHash. Otherwise,
|
|
|
|
// we pay to the rHash specified within the RPC
|
|
|
|
// request.
|
|
|
|
var rHash [32]byte
|
|
|
|
if cfg.DebugHTLC {
|
|
|
|
rHash = debugHash
|
|
|
|
} else {
|
|
|
|
copy(rHash[:], nextPayment.PaymentHash)
|
|
|
|
}
|
2016-09-21 03:15:26 +03:00
|
|
|
|
|
|
|
// Craft an HTLC packet to send to the routing
|
|
|
|
// sub-system. The meta-data within this packet will be
|
|
|
|
// used to route the payment through the network.
|
2016-07-22 02:22:30 +03:00
|
|
|
htlcAdd := &lnwire.HTLCAddRequest{
|
|
|
|
Amount: lnwire.CreditsAmount(nextPayment.Amt),
|
2016-09-19 22:03:38 +03:00
|
|
|
RedemptionHashes: [][32]byte{rHash},
|
2016-09-21 03:15:26 +03:00
|
|
|
OnionBlob: sphinxPacket,
|
2016-07-22 02:22:30 +03:00
|
|
|
}
|
2016-09-21 03:15:26 +03:00
|
|
|
destAddr := wire.ShaHash(fastsha256.Sum256(nextPayment.Dest))
|
2016-07-22 02:22:30 +03:00
|
|
|
htlcPkt := &htlcPacket{
|
2016-09-21 03:15:26 +03:00
|
|
|
dest: destAddr,
|
2016-07-22 02:22:30 +03:00
|
|
|
msg: htlcAdd,
|
|
|
|
}
|
2016-07-13 03:46:25 +03:00
|
|
|
|
2016-07-22 02:22:30 +03:00
|
|
|
// TODO(roasbeef): semaphore to limit num outstanding
|
|
|
|
// goroutines.
|
|
|
|
go func() {
|
|
|
|
// Finally, send this next packet to the routing layer in order
|
|
|
|
// to complete the next payment.
|
|
|
|
// TODO(roasbeef): this should go through the L3 router once
|
|
|
|
// multi-hop is in place.
|
|
|
|
if err := r.server.htlcSwitch.SendHTLC(htlcPkt); err != nil {
|
|
|
|
errChan <- err
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO(roasbeef): proper responses
|
|
|
|
resp := &lnrpc.SendResponse{}
|
|
|
|
if err := paymentStream.Send(resp); err != nil {
|
|
|
|
errChan <- err
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}()
|
2016-07-13 03:46:25 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2016-07-15 14:02:59 +03:00
|
|
|
|
2016-09-21 03:15:26 +03:00
|
|
|
// generateSphinxPacket generates then encodes a sphinx packet which encodes
|
|
|
|
// the onion route specified by the passed list of graph vertexes. The blob
|
|
|
|
// returned from this function can immediately be included within an HTLC add
|
|
|
|
// packet to be sent to the first hop within the route.
|
|
|
|
func generateSphinxPacket(vertexes []graph.ID) ([]byte, error) {
|
|
|
|
var dest sphinx.LightningAddress
|
|
|
|
e2eMessage := []byte("test")
|
|
|
|
|
|
|
|
route := make([]*btcec.PublicKey, len(vertexes))
|
|
|
|
for i, vertex := range vertexes {
|
|
|
|
vertexBytes, err := hex.DecodeString(vertex.String())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
pub, err := btcec.ParsePubKey(vertexBytes, btcec.S256())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
route[i] = pub
|
|
|
|
}
|
|
|
|
|
|
|
|
// Next generate the onion routing packet which allows
|
|
|
|
// us to perform privacy preserving source routing
|
|
|
|
// across the network.
|
|
|
|
var onionBlob bytes.Buffer
|
|
|
|
mixHeader, err := sphinx.NewOnionPacket(route, dest,
|
|
|
|
e2eMessage)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if err := mixHeader.Encode(&onionBlob); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
rpcsLog.Tracef("[sendpayment] generated sphinx packet: %v",
|
|
|
|
newLogClosure(func() string {
|
|
|
|
return spew.Sdump(mixHeader)
|
|
|
|
}))
|
|
|
|
|
|
|
|
return onionBlob.Bytes(), nil
|
|
|
|
}
|
|
|
|
|
2016-09-19 22:04:56 +03:00
|
|
|
// AddInvoice attempts to add a new invoice to the invoice database. Any
|
|
|
|
// duplicated invoices are rejected, therefore all invoices *must* have a
|
|
|
|
// unique payment preimage.
|
|
|
|
func (r *rpcServer) AddInvoice(ctx context.Context,
|
|
|
|
invoice *lnrpc.Invoice) (*lnrpc.AddInvoiceResponse, error) {
|
|
|
|
|
|
|
|
preImage := invoice.RPreimage
|
|
|
|
preimageLength := len(preImage)
|
|
|
|
if preimageLength != 32 {
|
|
|
|
return nil, fmt.Errorf("payment preimage must be exactly "+
|
|
|
|
"32 bytes, is instead %v", preimageLength)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(invoice.Memo) > channeldb.MaxMemoSize {
|
|
|
|
return nil, fmt.Errorf("memo too large: %v bytes "+
|
|
|
|
"(maxsize=%v)", len(invoice.Memo), channeldb.MaxMemoSize)
|
|
|
|
}
|
|
|
|
if len(invoice.Receipt) > channeldb.MaxReceiptSize {
|
|
|
|
return nil, fmt.Errorf("receipt too large: %v bytes "+
|
|
|
|
"(maxsize=%v)", len(invoice.Receipt), channeldb.MaxReceiptSize)
|
|
|
|
}
|
|
|
|
|
|
|
|
i := &channeldb.Invoice{
|
|
|
|
CreationDate: time.Now(),
|
|
|
|
Terms: channeldb.ContractTerm{
|
|
|
|
Value: btcutil.Amount(invoice.Value),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
copy(i.Memo[:], invoice.Memo)
|
|
|
|
copy(i.Receipt[:], invoice.Receipt)
|
|
|
|
copy(i.Terms.PaymentPreimage[:], preImage)
|
|
|
|
|
|
|
|
if err := r.server.invoices.AddInvoice(i); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
rHash := fastsha256.Sum256(preImage)
|
|
|
|
return &lnrpc.AddInvoiceResponse{
|
|
|
|
RHash: rHash[:],
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// LookupInvoice attemps to look up an invoice according to its payment hash.
|
|
|
|
// The passed payment hash *must* be exactly 32 bytes, if not an error is
|
|
|
|
// returned.
|
|
|
|
func (r *rpcServer) LookupInvoice(ctx context.Context,
|
|
|
|
req *lnrpc.PaymentHash) (*lnrpc.Invoice, error) {
|
|
|
|
|
|
|
|
if len(req.RHash) != 32 {
|
|
|
|
return nil, fmt.Errorf("payment hash must be exactly "+
|
|
|
|
"32 bytes, is instead %v", len(req.RHash))
|
|
|
|
}
|
|
|
|
|
|
|
|
var payHash [32]byte
|
|
|
|
copy(payHash[:], req.RHash)
|
|
|
|
|
|
|
|
invoice, err := r.server.invoices.LookupInvoice(payHash)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &lnrpc.Invoice{
|
|
|
|
Memo: string(invoice.Memo[:]),
|
|
|
|
Receipt: invoice.Receipt[:],
|
|
|
|
RPreimage: invoice.Terms.PaymentPreimage[:],
|
|
|
|
Value: int64(invoice.Terms.Value),
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ListInvoices returns a list of all the invoices currently stored within the
|
|
|
|
// database. Any active debug invoices are ignored.
|
|
|
|
func (r *rpcServer) ListInvoices(ctx context.Context,
|
|
|
|
req *lnrpc.ListInvoiceRequest) (*lnrpc.ListInvoiceResponse, error) {
|
|
|
|
|
|
|
|
dbInvoices, err := r.server.chanDB.FetchAllInvoices(req.PendingOnly)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
invoices := make([]*lnrpc.Invoice, len(dbInvoices))
|
|
|
|
for i, dbInvoice := range dbInvoices {
|
|
|
|
invoice := &lnrpc.Invoice{
|
|
|
|
Memo: string(dbInvoice.Memo[:]),
|
|
|
|
Receipt: dbInvoice.Receipt[:],
|
|
|
|
RPreimage: dbInvoice.Terms.PaymentPreimage[:],
|
|
|
|
Value: int64(dbInvoice.Terms.Value),
|
|
|
|
}
|
|
|
|
|
|
|
|
invoices[i] = invoice
|
|
|
|
}
|
|
|
|
|
|
|
|
return &lnrpc.ListInvoiceResponse{
|
|
|
|
Invoices: invoices,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2016-07-15 14:02:59 +03:00
|
|
|
func (r *rpcServer) ShowRoutingTable(ctx context.Context,
|
|
|
|
in *lnrpc.ShowRoutingTableRequest) (*lnrpc.ShowRoutingTableResponse, error) {
|
2016-09-21 03:15:26 +03:00
|
|
|
|
2016-07-15 14:02:59 +03:00
|
|
|
rpcsLog.Debugf("[ShowRoutingTable]")
|
2016-09-21 03:15:26 +03:00
|
|
|
|
2016-07-15 14:02:59 +03:00
|
|
|
rtCopy := r.server.routingMgr.GetRTCopy()
|
2016-09-21 03:15:26 +03:00
|
|
|
|
2016-08-21 17:46:54 +03:00
|
|
|
channels := make([]*lnrpc.RoutingTableLink, 0)
|
2016-08-20 23:49:35 +03:00
|
|
|
for _, channel := range rtCopy.AllChannels() {
|
|
|
|
channels = append(channels,
|
2016-08-21 17:46:54 +03:00
|
|
|
&lnrpc.RoutingTableLink{
|
2016-09-12 22:42:26 +03:00
|
|
|
Id1: channel.Id1.String(),
|
|
|
|
Id2: channel.Id2.String(),
|
|
|
|
Outpoint: channel.EdgeID.String(),
|
|
|
|
Capacity: channel.Info.Capacity(),
|
|
|
|
Weight: channel.Info.Weight(),
|
2016-08-20 23:49:35 +03:00
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
2016-09-21 03:15:26 +03:00
|
|
|
|
2016-07-15 14:02:59 +03:00
|
|
|
return &lnrpc.ShowRoutingTableResponse{
|
2016-08-20 23:49:35 +03:00
|
|
|
Channels: channels,
|
2016-07-15 14:02:59 +03:00
|
|
|
}, nil
|
|
|
|
}
|