2020-01-03 13:01:31 +00:00
|
|
|
package loopd
|
2019-03-06 20:13:50 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2019-05-03 08:17:26 +00:00
|
|
|
"errors"
|
2019-03-06 20:13:50 +00:00
|
|
|
"fmt"
|
|
|
|
"net"
|
2019-03-12 22:35:53 +00:00
|
|
|
"net/http"
|
2019-03-06 20:13:50 +00:00
|
|
|
"os"
|
|
|
|
"os/signal"
|
|
|
|
"runtime/pprof"
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
2019-03-12 22:35:53 +00:00
|
|
|
proxy "github.com/grpc-ecosystem/grpc-gateway/runtime"
|
2019-03-07 04:32:24 +00:00
|
|
|
"github.com/lightninglabs/loop"
|
2020-01-03 13:01:32 +00:00
|
|
|
"github.com/lightninglabs/loop/lndclient"
|
2019-03-06 23:53:17 +00:00
|
|
|
"github.com/lightninglabs/loop/looprpc"
|
2019-03-06 20:13:50 +00:00
|
|
|
"google.golang.org/grpc"
|
|
|
|
)
|
|
|
|
|
2020-01-03 13:01:31 +00:00
|
|
|
// listenerCfg holds closures used to retrieve listeners for the gRPC services.
|
|
|
|
type listenerCfg struct {
|
|
|
|
// grpcListener returns a listener to use for the gRPC server.
|
|
|
|
grpcListener func() (net.Listener, error)
|
|
|
|
|
|
|
|
// restListener returns a listener to use for the REST proxy.
|
|
|
|
restListener func() (net.Listener, error)
|
2020-01-03 13:01:32 +00:00
|
|
|
|
|
|
|
// getLnd returns a grpc connection to an lnd instance.
|
|
|
|
getLnd func(string, *lndConfig) (*lndclient.GrpcLndServices, error)
|
2020-01-03 13:01:31 +00:00
|
|
|
}
|
|
|
|
|
2019-03-07 10:37:28 +00:00
|
|
|
// daemon runs loopd in daemon mode. It will listen for grpc connections,
|
2019-03-06 20:13:50 +00:00
|
|
|
// execute commands and pass back swap status information.
|
2020-01-03 13:01:31 +00:00
|
|
|
func daemon(config *config, lisCfg *listenerCfg) error {
|
2020-01-03 13:01:32 +00:00
|
|
|
lnd, err := lisCfg.getLnd(config.Network, config.Lnd)
|
2019-03-06 20:13:50 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer lnd.Close()
|
|
|
|
|
2019-05-03 08:17:26 +00:00
|
|
|
// If no swap server is specified, use the default addresses for mainnet
|
|
|
|
// and testnet.
|
|
|
|
if config.SwapServer == "" {
|
|
|
|
switch config.Network {
|
|
|
|
case "mainnet":
|
|
|
|
config.SwapServer = mainnetServer
|
|
|
|
case "testnet":
|
|
|
|
config.SwapServer = testnetServer
|
|
|
|
default:
|
|
|
|
return errors.New("no swap server address specified")
|
|
|
|
}
|
2019-03-22 04:53:50 +00:00
|
|
|
}
|
|
|
|
|
2019-10-28 16:06:07 +00:00
|
|
|
log.Infof("Swap server address: %v", config.SwapServer)
|
2019-05-03 08:17:26 +00:00
|
|
|
|
2019-04-11 07:06:59 +00:00
|
|
|
// Create an instance of the loop client library.
|
2020-01-10 11:13:39 +00:00
|
|
|
swapClient, cleanup, err := getClient(config, &lnd.LndServices)
|
2019-03-06 20:13:50 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer cleanup()
|
|
|
|
|
2019-04-11 07:06:59 +00:00
|
|
|
// Retrieve all currently existing swaps from the database.
|
|
|
|
swapsList, err := swapClient.FetchSwaps()
|
2019-03-06 20:13:50 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-04-11 07:06:59 +00:00
|
|
|
for _, s := range swapsList {
|
|
|
|
swaps[s.SwapHash] = *s
|
2019-03-12 15:10:37 +00:00
|
|
|
}
|
|
|
|
|
2019-03-07 10:37:28 +00:00
|
|
|
// Instantiate the loopd gRPC server.
|
2019-03-06 20:13:50 +00:00
|
|
|
server := swapClientServer{
|
|
|
|
impl: swapClient,
|
|
|
|
lnd: &lnd.LndServices,
|
|
|
|
}
|
|
|
|
|
|
|
|
serverOpts := []grpc.ServerOption{}
|
|
|
|
grpcServer := grpc.NewServer(serverOpts...)
|
2019-03-06 23:53:17 +00:00
|
|
|
looprpc.RegisterSwapClientServer(grpcServer, &server)
|
2019-03-06 20:13:50 +00:00
|
|
|
|
2019-03-12 22:35:53 +00:00
|
|
|
// Next, start the gRPC server listening for HTTP/2 connections.
|
2019-10-28 16:06:07 +00:00
|
|
|
log.Infof("Starting gRPC listener")
|
2020-01-03 13:01:31 +00:00
|
|
|
grpcListener, err := lisCfg.grpcListener()
|
2019-03-06 20:13:50 +00:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("RPC server unable to listen on %s",
|
2019-03-12 22:35:53 +00:00
|
|
|
config.RPCListen)
|
2019-03-06 20:13:50 +00:00
|
|
|
|
|
|
|
}
|
2019-03-12 22:35:53 +00:00
|
|
|
defer grpcListener.Close()
|
|
|
|
|
|
|
|
// We'll also create and start an accompanying proxy to serve clients
|
|
|
|
// through REST.
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
defer cancel()
|
|
|
|
mux := proxy.NewServeMux()
|
|
|
|
proxyOpts := []grpc.DialOption{grpc.WithInsecure()}
|
|
|
|
err = looprpc.RegisterSwapClientHandlerFromEndpoint(
|
|
|
|
ctx, mux, config.RPCListen, proxyOpts,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-01-03 13:01:31 +00:00
|
|
|
restListener, err := lisCfg.restListener()
|
2019-03-12 22:35:53 +00:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("REST proxy unable to listen on %s",
|
|
|
|
config.RESTListen)
|
|
|
|
}
|
2020-01-03 13:01:31 +00:00
|
|
|
|
2020-01-03 13:01:32 +00:00
|
|
|
// A nil listener indicates REST is disabled.
|
|
|
|
if restListener != nil {
|
|
|
|
log.Infof("Starting REST proxy listener")
|
|
|
|
|
|
|
|
defer restListener.Close()
|
|
|
|
proxy := &http.Server{Handler: mux}
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
err := proxy.Serve(restListener)
|
|
|
|
// ErrServerClosed is always returned when the proxy is
|
|
|
|
// shut down, so don't log it.
|
|
|
|
if err != nil && err != http.ErrServerClosed {
|
|
|
|
log.Error(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
} else {
|
|
|
|
log.Infof("REST proxy disabled")
|
|
|
|
}
|
2019-03-06 20:13:50 +00:00
|
|
|
|
2019-03-07 04:32:24 +00:00
|
|
|
statusChan := make(chan loop.SwapInfo)
|
2019-03-06 20:13:50 +00:00
|
|
|
|
|
|
|
mainCtx, cancel := context.WithCancel(context.Background())
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
|
|
|
// Start the swap client itself.
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
|
2019-10-28 16:06:07 +00:00
|
|
|
log.Infof("Starting swap client")
|
2019-03-06 20:13:50 +00:00
|
|
|
err := swapClient.Run(mainCtx, statusChan)
|
|
|
|
if err != nil {
|
2019-10-28 16:06:07 +00:00
|
|
|
log.Error(err)
|
2019-03-06 20:13:50 +00:00
|
|
|
}
|
2019-10-28 16:06:07 +00:00
|
|
|
log.Infof("Swap client stopped")
|
2019-03-06 20:13:50 +00:00
|
|
|
|
2019-10-28 16:06:07 +00:00
|
|
|
log.Infof("Stopping gRPC server")
|
2019-03-06 20:13:50 +00:00
|
|
|
grpcServer.Stop()
|
|
|
|
|
|
|
|
cancel()
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Start a goroutine that broadcasts swap updates to clients.
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
|
2019-10-28 16:06:07 +00:00
|
|
|
log.Infof("Waiting for updates")
|
2019-03-06 20:13:50 +00:00
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case swap := <-statusChan:
|
|
|
|
swapsLock.Lock()
|
|
|
|
swaps[swap.SwapHash] = swap
|
|
|
|
|
|
|
|
for _, subscriber := range subscribers {
|
|
|
|
select {
|
|
|
|
case subscriber <- swap:
|
|
|
|
case <-mainCtx.Done():
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
swapsLock.Unlock()
|
|
|
|
case <-mainCtx.Done():
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Start the grpc server.
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
|
|
|
|
2019-10-28 16:06:07 +00:00
|
|
|
log.Infof("RPC server listening on %s", grpcListener.Addr())
|
2020-01-03 13:01:32 +00:00
|
|
|
|
|
|
|
if restListener != nil {
|
|
|
|
log.Infof("REST proxy listening on %s", restListener.Addr())
|
|
|
|
}
|
2019-03-06 20:13:50 +00:00
|
|
|
|
2019-03-12 22:35:53 +00:00
|
|
|
err = grpcServer.Serve(grpcListener)
|
2019-03-06 20:13:50 +00:00
|
|
|
if err != nil {
|
2019-10-28 16:06:07 +00:00
|
|
|
log.Error(err)
|
2019-03-06 20:13:50 +00:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
interruptChannel := make(chan os.Signal, 1)
|
|
|
|
signal.Notify(interruptChannel, os.Interrupt)
|
|
|
|
|
2019-03-07 10:37:28 +00:00
|
|
|
// Run until the users terminates loopd or an error occurred.
|
2019-03-06 20:13:50 +00:00
|
|
|
select {
|
|
|
|
case <-interruptChannel:
|
2019-10-28 16:06:07 +00:00
|
|
|
log.Infof("Received SIGINT (Ctrl+C).")
|
2019-03-06 20:13:50 +00:00
|
|
|
|
|
|
|
// TODO: Remove debug code.
|
|
|
|
// Debug code to dump goroutines on hanging exit.
|
|
|
|
go func() {
|
|
|
|
time.Sleep(5 * time.Second)
|
|
|
|
pprof.Lookup("goroutine").WriteTo(os.Stdout, 1)
|
|
|
|
}()
|
|
|
|
|
|
|
|
cancel()
|
|
|
|
case <-mainCtx.Done():
|
|
|
|
}
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|