//go:build autopilotrpc // +build autopilotrpc package autopilotrpc import ( "context" "encoding/hex" "sync/atomic" "github.com/btcsuite/btcd/btcec/v2" "github.com/grpc-ecosystem/grpc-gateway/v2/runtime" "github.com/lightningnetwork/lnd/autopilot" "github.com/lightningnetwork/lnd/lnrpc" "google.golang.org/grpc" "gopkg.in/macaroon-bakery.v2/bakery" ) const ( // subServerName is the name of the sub rpc server. We'll use this name // to register ourselves, and we also require that the main // SubServerConfigDispatcher instance recognize tt as the name of our // RPC service. subServerName = "AutopilotRPC" ) var ( // macPermissions maps RPC calls to the permissions they require. macPermissions = map[string][]bakery.Op{ "/autopilotrpc.Autopilot/Status": {{ Entity: "info", Action: "read", }}, "/autopilotrpc.Autopilot/ModifyStatus": {{ Entity: "onchain", Action: "write", }, { Entity: "offchain", Action: "write", }}, "/autopilotrpc.Autopilot/QueryScores": {{ Entity: "info", Action: "read", }}, "/autopilotrpc.Autopilot/SetScores": {{ Entity: "onchain", Action: "write", }, { Entity: "offchain", Action: "write", }}, } ) // ServerShell is a shell struct holding a reference to the actual sub-server. // It is used to register the gRPC sub-server with the root server before we // have the necessary dependencies to populate the actual sub-server. type ServerShell struct { AutopilotServer } // Server is a sub-server of the main RPC server: the autopilot RPC. This sub // RPC server allows external callers to access the status of the autopilot // currently active within lnd, as well as configuring it at runtime. type Server struct { started int32 // To be used atomically. shutdown int32 // To be used atomically. // Required by the grpc-gateway/v2 library for forward compatibility. // Must be after the atomically used variables to not break struct // alignment. UnimplementedAutopilotServer cfg *Config manager *autopilot.Manager } // A compile time check to ensure that Server fully implements the // AutopilotServer gRPC service. var _ AutopilotServer = (*Server)(nil) // New returns a new instance of the autopilotrpc Autopilot sub-server. We also // return the set of permissions for the macaroons that we may create within // this method. If the macaroons we need aren't found in the filepath, then // we'll create them on start up. If we're unable to locate, or create the // macaroons we need, then we'll return with an error. func New(cfg *Config) (*Server, lnrpc.MacaroonPerms, error) { // We don't create any new macaroons for this subserver, instead reuse // existing onchain/offchain permissions. server := &Server{ cfg: cfg, manager: cfg.Manager, } return server, macPermissions, nil } // Start launches any helper goroutines required for the Server to function. // // NOTE: This is part of the lnrpc.SubServer interface. func (s *Server) Start() error { if atomic.AddInt32(&s.started, 1) != 1 { return nil } return s.manager.Start() } // Stop signals any active goroutines for a graceful closure. // // NOTE: This is part of the lnrpc.SubServer interface. func (s *Server) Stop() error { if atomic.AddInt32(&s.shutdown, 1) != 1 { return nil } return s.manager.Stop() } // Name returns a unique string representation of the sub-server. This can be // used to identify the sub-server and also de-duplicate them. // // NOTE: This is part of the lnrpc.SubServer interface. func (s *Server) Name() string { return subServerName } // RegisterWithRootServer will be called by the root gRPC server to direct a // sub RPC server to register itself with the main gRPC root server. Until this // is called, each sub-server won't be able to have // requests routed towards it. // // NOTE: This is part of the lnrpc.GrpcHandler interface. func (r *ServerShell) RegisterWithRootServer(grpcServer *grpc.Server) error { // We make sure that we register it with the main gRPC server to ensure // all our methods are routed properly. RegisterAutopilotServer(grpcServer, r) log.Debugf("Autopilot RPC server successfully register with root " + "gRPC server") return nil } // RegisterWithRestServer will be called by the root REST mux to direct a sub // RPC server to register itself with the main REST mux server. Until this is // called, each sub-server won't be able to have requests routed towards it. // // NOTE: This is part of the lnrpc.GrpcHandler interface. func (r *ServerShell) RegisterWithRestServer(ctx context.Context, mux *runtime.ServeMux, dest string, opts []grpc.DialOption) error { // We make sure that we register it with the main REST server to ensure // all our methods are routed properly. err := RegisterAutopilotHandlerFromEndpoint(ctx, mux, dest, opts) if err != nil { log.Errorf("Could not register Autopilot REST server "+ "with root REST server: %v", err) return err } log.Debugf("Autopilot REST server successfully registered with " + "root REST server") return nil } // CreateSubServer populates the subserver's dependencies using the passed // SubServerConfigDispatcher. This method should fully initialize the // sub-server instance, making it ready for action. It returns the macaroon // permissions that the sub-server wishes to pass on to the root server for all // methods routed towards it. // // NOTE: This is part of the lnrpc.GrpcHandler interface. func (r *ServerShell) CreateSubServer(configRegistry lnrpc.SubServerConfigDispatcher) ( lnrpc.SubServer, lnrpc.MacaroonPerms, error) { subServer, macPermissions, err := createNewSubServer(configRegistry) if err != nil { return nil, nil, err } r.AutopilotServer = subServer return subServer, macPermissions, nil } // Status returns the current status of the autopilot agent. // // NOTE: Part of the AutopilotServer interface. func (s *Server) Status(ctx context.Context, in *StatusRequest) (*StatusResponse, error) { return &StatusResponse{ Active: s.manager.IsActive(), }, nil } // ModifyStatus activates the current autopilot agent, if active. // // NOTE: Part of the AutopilotServer interface. func (s *Server) ModifyStatus(ctx context.Context, in *ModifyStatusRequest) (*ModifyStatusResponse, error) { log.Debugf("Setting agent enabled=%v", in.Enable) var err error if in.Enable { err = s.manager.StartAgent() } else { err = s.manager.StopAgent() } return &ModifyStatusResponse{}, err } // QueryScores queries all available autopilot heuristics, in addition to any // active combination of these heruristics, for the scores they would give to // the given nodes. // // NOTE: Part of the AutopilotServer interface. func (s *Server) QueryScores(ctx context.Context, in *QueryScoresRequest) ( *QueryScoresResponse, error) { var nodes []autopilot.NodeID for _, pubStr := range in.Pubkeys { pubHex, err := hex.DecodeString(pubStr) if err != nil { return nil, err } pubKey, err := btcec.ParsePubKey(pubHex) if err != nil { return nil, err } nID := autopilot.NewNodeID(pubKey) nodes = append(nodes, nID) } // Query the heuristics. heuristicScores, err := s.manager.QueryHeuristics( nodes, !in.IgnoreLocalState, ) if err != nil { return nil, err } resp := &QueryScoresResponse{} for heuristic, scores := range heuristicScores { result := &QueryScoresResponse_HeuristicResult{ Heuristic: heuristic, Scores: make(map[string]float64), } for pub, score := range scores { pubkeyHex := hex.EncodeToString(pub[:]) result.Scores[pubkeyHex] = score } // Since a node not being part of the internally returned // scores imply a zero score, we add these before we return the // RPC results. for _, node := range nodes { if _, ok := scores[node]; ok { continue } pubkeyHex := hex.EncodeToString(node[:]) result.Scores[pubkeyHex] = 0.0 } resp.Results = append(resp.Results, result) } return resp, nil } // SetScores sets the scores of the external score heuristic, if active. // // NOTE: Part of the AutopilotServer interface. func (s *Server) SetScores(ctx context.Context, in *SetScoresRequest) (*SetScoresResponse, error) { scores := make(map[autopilot.NodeID]float64) for pubStr, score := range in.Scores { pubHex, err := hex.DecodeString(pubStr) if err != nil { return nil, err } pubKey, err := btcec.ParsePubKey(pubHex) if err != nil { return nil, err } nID := autopilot.NewNodeID(pubKey) scores[nID] = score } if err := s.manager.SetNodeScores(in.Heuristic, scores); err != nil { return nil, err } return &SetScoresResponse{}, nil }