lotus/cli/cmd.go

244 lines
5.4 KiB
Go
Raw Permalink Normal View History

package cli
import (
"context"
"fmt"
2019-07-23 18:49:09 +00:00
"net/http"
"os"
"os/signal"
"strings"
"syscall"
logging "github.com/ipfs/go-log/v2"
2019-11-21 16:10:04 +00:00
"github.com/mitchellh/go-homedir"
"github.com/multiformats/go-multiaddr"
2019-07-10 17:28:49 +00:00
manet "github.com/multiformats/go-multiaddr-net"
2019-07-26 11:45:25 +00:00
"golang.org/x/xerrors"
"gopkg.in/urfave/cli.v2"
2019-07-10 17:28:49 +00:00
2020-05-20 17:43:22 +00:00
"github.com/filecoin-project/go-jsonrpc"
"github.com/filecoin-project/lotus/api"
"github.com/filecoin-project/lotus/api/client"
"github.com/filecoin-project/lotus/node/repo"
)
2019-07-23 18:49:09 +00:00
var log = logging.Logger("cli")
const (
metadataTraceConetxt = "traceContext"
)
// custom CLI error
type ErrCmdFailed struct {
msg string
}
func (e *ErrCmdFailed) Error() string {
return e.msg
}
func NewCliError(s string) error {
return &ErrCmdFailed{s}
}
// ApiConnector returns API instance
2019-07-24 00:09:34 +00:00
type ApiConnector func() api.FullNode
2019-07-08 19:07:16 +00:00
type APIInfo struct {
Addr multiaddr.Multiaddr
Token []byte
}
func (a APIInfo) DialArgs() (string, error) {
_, addr, err := manet.DialArgs(a.Addr)
return "ws://" + addr + "/rpc/v0", err
}
func (a APIInfo) AuthHeader() http.Header {
if len(a.Token) != 0 {
headers := http.Header{}
headers.Add("Authorization", "Bearer "+string(a.Token))
return headers
}
log.Warn("API Token not set and requested, capabilities might be limited.")
return nil
}
func flagForRepo(t repo.RepoType) string {
switch t {
case repo.FullNode:
return "repo"
case repo.StorageMiner:
return "storagerepo"
default:
panic(fmt.Sprintf("Unknown repo type: %v", t))
}
}
func envForRepo(t repo.RepoType) string {
switch t {
case repo.FullNode:
return "FULLNODE_API_INFO"
case repo.StorageMiner:
return "STORAGE_API_INFO"
default:
panic(fmt.Sprintf("Unknown repo type: %v", t))
}
}
func GetAPIInfo(ctx *cli.Context, t repo.RepoType) (APIInfo, error) {
if env, ok := os.LookupEnv(envForRepo(t)); ok {
sp := strings.SplitN(env, ":", 2)
if len(sp) != 2 {
log.Warnf("invalid env(%s) value, missing token or address", envForRepo(t))
} else {
ma, err := multiaddr.NewMultiaddr(sp[1])
if err != nil {
return APIInfo{}, xerrors.Errorf("could not parse multiaddr from env(%s): %w", envForRepo(t), err)
}
return APIInfo{
Addr: ma,
Token: []byte(sp[0]),
}, nil
}
}
repoFlag := flagForRepo(t)
2019-11-21 16:10:04 +00:00
p, err := homedir.Expand(ctx.String(repoFlag))
2019-07-10 17:28:49 +00:00
if err != nil {
2020-03-24 06:54:39 +00:00
return APIInfo{}, xerrors.Errorf("cound not expand home dir (%s): %w", repoFlag, err)
2019-11-21 16:10:04 +00:00
}
r, err := repo.NewFS(p)
if err != nil {
return APIInfo{}, xerrors.Errorf("could not open repo at path: %s; %w", p, err)
2019-07-10 17:28:49 +00:00
}
ma, err := r.APIEndpoint()
if err != nil {
2020-03-24 06:54:39 +00:00
return APIInfo{}, xerrors.Errorf("could not get api endpoint: %w", err)
2019-07-10 17:28:49 +00:00
}
token, err := r.APIToken()
2019-11-21 16:10:04 +00:00
if err != nil {
log.Warnf("Couldn't load CLI token, capabilities may be limited: %v", err)
2019-11-21 16:10:04 +00:00
}
return APIInfo{
Addr: ma,
Token: token,
}, nil
2019-11-21 16:10:04 +00:00
}
func GetRawAPI(ctx *cli.Context, t repo.RepoType) (string, http.Header, error) {
ainfo, err := GetAPIInfo(ctx, t)
2019-07-11 11:52:07 +00:00
if err != nil {
return "", nil, xerrors.Errorf("could not get API info: %w", err)
2019-07-11 11:52:07 +00:00
}
2019-11-21 16:10:04 +00:00
addr, err := ainfo.DialArgs()
2019-07-23 18:49:09 +00:00
if err != nil {
return "", nil, xerrors.Errorf("could not get DialArgs: %w", err)
2019-07-23 18:49:09 +00:00
}
return addr, ainfo.AuthHeader(), nil
}
2019-10-03 18:12:30 +00:00
func GetAPI(ctx *cli.Context) (api.Common, jsonrpc.ClientCloser, error) {
ti, ok := ctx.App.Metadata["repoType"]
if !ok {
log.Errorf("unknown repo type, are you sure you want to use GetAPI?")
ti = repo.FullNode
}
t, ok := ti.(repo.RepoType)
if !ok {
log.Errorf("repoType type does not match the type of repo.RepoType")
2019-08-02 16:18:44 +00:00
}
addr, headers, err := GetRawAPI(ctx, t)
2019-08-02 16:18:44 +00:00
if err != nil {
2019-10-03 18:12:30 +00:00
return nil, nil, err
2019-08-02 16:18:44 +00:00
}
return client.NewCommonRPC(addr, headers)
}
2019-10-03 18:12:30 +00:00
func GetFullNodeAPI(ctx *cli.Context) (api.FullNode, jsonrpc.ClientCloser, error) {
addr, headers, err := GetRawAPI(ctx, repo.FullNode)
if err != nil {
2019-10-03 18:12:30 +00:00
return nil, nil, err
}
return client.NewFullNodeRPC(addr, headers)
}
2019-10-03 18:12:30 +00:00
func GetStorageMinerAPI(ctx *cli.Context) (api.StorageMiner, jsonrpc.ClientCloser, error) {
addr, headers, err := GetRawAPI(ctx, repo.StorageMiner)
if err != nil {
2019-10-03 18:12:30 +00:00
return nil, nil, err
}
return client.NewStorageMinerRPC(addr, headers)
}
2019-09-17 18:36:06 +00:00
func DaemonContext(cctx *cli.Context) context.Context {
if mtCtx, ok := cctx.App.Metadata[metadataTraceConetxt]; ok {
return mtCtx.(context.Context)
}
return context.Background()
}
2019-07-18 23:16:23 +00:00
// ReqContext returns context for cli execution. Calling it for the first time
// installs SIGTERM handler that will close returned context.
// Not safe for concurrent execution.
2019-07-18 23:16:23 +00:00
func ReqContext(cctx *cli.Context) context.Context {
2019-09-17 18:36:06 +00:00
tCtx := DaemonContext(cctx)
ctx, done := context.WithCancel(tCtx)
sigChan := make(chan os.Signal, 2)
go func() {
<-sigChan
done()
}()
2020-03-09 22:03:33 +00:00
signal.Notify(sigChan, syscall.SIGTERM, syscall.SIGINT, syscall.SIGHUP)
return ctx
2019-07-08 19:07:16 +00:00
}
2020-03-23 12:29:24 +00:00
var CommonCommands = []*cli.Command{
netCmd,
2020-05-13 11:08:59 +00:00
authCmd,
2020-03-23 12:29:24 +00:00
logCmd,
waitApiCmd,
2020-05-13 11:08:59 +00:00
fetchParamCmd,
versionCmd,
2020-03-23 12:29:24 +00:00
}
var Commands = []*cli.Command{
2020-05-13 11:08:59 +00:00
withCategory("basic", sendCmd),
withCategory("basic", walletCmd),
withCategory("basic", clientCmd),
withCategory("basic", multisigCmd),
withCategory("basic", paychCmd),
withCategory("developer", authCmd),
withCategory("developer", mpoolCmd),
withCategory("developer", stateCmd),
withCategory("developer", chainCmd),
withCategory("developer", logCmd),
withCategory("developer", waitApiCmd),
withCategory("developer", fetchParamCmd),
withCategory("network", netCmd),
withCategory("network", syncCmd),
versionCmd,
2020-05-13 11:08:59 +00:00
}
func withCategory(cat string, cmd *cli.Command) *cli.Command {
cmd.Category = cat
return cmd
}