package main import ( "context" "net/http" _ "net/http/pprof" "os" "os/signal" "syscall" mux "github.com/gorilla/mux" "github.com/multiformats/go-multiaddr" manet "github.com/multiformats/go-multiaddr-net" "golang.org/x/xerrors" "gopkg.in/urfave/cli.v2" "github.com/filecoin-project/lotus/api" "github.com/filecoin-project/lotus/build" lcli "github.com/filecoin-project/lotus/cli" "github.com/filecoin-project/lotus/lib/auth" "github.com/filecoin-project/lotus/lib/jsonrpc" "github.com/filecoin-project/lotus/node" "github.com/filecoin-project/lotus/node/impl" "github.com/filecoin-project/lotus/node/repo" ) const defaultListen = "/ip4/127.0.0.1/tcp/" var runCmd = &cli.Command{ Name: "run", Usage: "Start a lotus storage miner process", Flags: []cli.Flag{ &cli.StringFlag{ Name: "api", Value: "2345", }, &cli.BoolFlag{ Name: "enable-gpu-proving", Usage: "enable use of GPU for mining operations", Value: true, }, &cli.BoolFlag{ Name: "nosync", Usage: "don't check full-node sync status", }, }, Action: func(cctx *cli.Context) error { if err := build.GetParams(true, false); err != nil { return xerrors.Errorf("fetching proof parameters: %w", err) } if !cctx.Bool("enable-gpu-proving") { os.Setenv("BELLMAN_NO_GPU", "true") } nodeApi, ncloser, err := lcli.GetFullNodeAPI(cctx) if err != nil { return err } defer ncloser() ctx := lcli.DaemonContext(cctx) v, err := nodeApi.Version(ctx) if err != nil { return err } if v.APIVersion != build.APIVersion { return xerrors.Errorf("lotus-daemon API version doesn't match: local: ", api.Version{APIVersion: build.APIVersion}) } log.Info("Checking full node sync status") if !cctx.Bool("nosync") { if err := lcli.SyncWait(ctx, nodeApi); err != nil { return xerrors.Errorf("sync wait: %w", err) } } storageRepoPath := cctx.String(FlagStorageRepo) r, err := repo.NewFS(storageRepoPath) if err != nil { return err } ok, err := r.Exists() if err != nil { return err } if !ok { return xerrors.Errorf("repo at '%s' is not initialized, run 'lotus-storage-miner init' to set it up", storageRepoPath) } var minerapi api.StorageMiner stop, err := node.New(ctx, node.StorageMiner(&minerapi), node.Online(), node.Repo(r), node.Override(node.SetApiEndpointKey, func(lr repo.LockedRepo) error { apima, err := parseApi(cctx.String("api")) if err != nil { return err } return lr.SetAPIEndpoint(apima) }), node.Override(new(api.FullNode), nodeApi), ) if err != nil { return err } endpoint, err := r.APIEndpoint() if err != nil { return err } // Bootstrap with full node remoteAddrs, err := nodeApi.NetAddrsListen(ctx) if err != nil { return err } if err := minerapi.NetConnect(ctx, remoteAddrs); err != nil { return err } log.Infof("Remote version %s", v) lst, err := manet.Listen(endpoint) if err != nil { return xerrors.Errorf("could not listen: %w", err) } mux := mux.NewRouter() rpcServer := jsonrpc.NewServer() rpcServer.Register("Filecoin", api.PermissionedStorMinerAPI(minerapi)) mux.Handle("/rpc/v0", rpcServer) mux.PathPrefix("/remote").HandlerFunc(minerapi.(*impl.StorageMinerAPI).ServeRemote) mux.PathPrefix("/").Handler(http.DefaultServeMux) // pprof ah := &auth.Handler{ Verify: minerapi.AuthVerify, Next: mux.ServeHTTP, } srv := &http.Server{Handler: ah} sigChan := make(chan os.Signal, 2) go func() { <-sigChan log.Warn("Shutting down..") if err := stop(context.TODO()); err != nil { log.Errorf("graceful shutting down failed: %s", err) } if err := srv.Shutdown(context.TODO()); err != nil { log.Errorf("shutting down RPC server failed: %s", err) } log.Warn("Graceful shutdown successful") }() signal.Notify(sigChan, syscall.SIGTERM, syscall.SIGINT) return srv.Serve(manet.NetListener(lst)) }, } func parseApi(api string) (multiaddr.Multiaddr, error) { if api == "" { return nil, xerrors.New("empty --api") } if api[0] != '/' { api = defaultListen + api } return multiaddr.NewMultiaddr(api) }