package node import ( "context" "crypto/rand" "encoding/base64" "errors" "go.uber.org/fx" "gitlab.dms3.io/dms3/go-dms3/core/node/dms3p2p" "gitlab.dms3.io/dms3/go-dms3/core/node/helpers" "gitlab.dms3.io/dms3/go-dms3/repo" ds "gitlab.dms3.io/dms3/go-datastore" dsync "gitlab.dms3.io/dms3/go-datastore/sync" cfg "gitlab.dms3.io/dms3/go-dms3-config" ci "gitlab.dms3.io/p2p/go-p2p-core/crypto" peer "gitlab.dms3.io/p2p/go-p2p-core/peer" ) type BuildCfg struct { // If online is set, the node will have networking enabled Online bool // ExtraOpts is a map of extra options used to configure the dms3 nodes creation ExtraOpts map[string]bool // If permanent then node should run more expensive processes // that will improve performance in long run Permanent bool // DisableEncryptedConnections disables connection encryption *entirely*. // DO NOT SET THIS UNLESS YOU'RE TESTING. DisableEncryptedConnections bool // If NilRepo is set, a Repo backed by a nil datastore will be constructed NilRepo bool Routing dms3p2p.RoutingOption Host dms3p2p.HostOption Repo repo.Repo } func (cfg *BuildCfg) getOpt(key string) bool { if cfg.ExtraOpts == nil { return false } return cfg.ExtraOpts[key] } func (cfg *BuildCfg) fillDefaults() error { if cfg.Repo != nil && cfg.NilRepo { return errors.New("cannot set a Repo and specify nilrepo at the same time") } if cfg.Repo == nil { var d ds.Datastore if cfg.NilRepo { d = ds.NewNullDatastore() } else { d = ds.NewMapDatastore() } r, err := defaultRepo(dsync.MutexWrap(d)) if err != nil { return err } cfg.Repo = r } if cfg.Routing == nil { cfg.Routing = dms3p2p.DHTOption } if cfg.Host == nil { cfg.Host = dms3p2p.DefaultHostOption } return nil } // options creates fx option group from this build config func (cfg *BuildCfg) options(ctx context.Context) (fx.Option, *cfg.Config) { err := cfg.fillDefaults() if err != nil { return fx.Error(err), nil } repoOption := fx.Provide(func(lc fx.Lifecycle) repo.Repo { lc.Append(fx.Hook{ OnStop: func(ctx context.Context) error { return cfg.Repo.Close() }, }) return cfg.Repo }) metricsCtx := fx.Provide(func() helpers.MetricsCtx { return helpers.MetricsCtx(ctx) }) hostOption := fx.Provide(func() dms3p2p.HostOption { return cfg.Host }) routingOption := fx.Provide(func() dms3p2p.RoutingOption { return cfg.Routing }) conf, err := cfg.Repo.Config() if err != nil { return fx.Error(err), nil } return fx.Options( repoOption, hostOption, routingOption, metricsCtx, ), conf } func defaultRepo(dstore repo.Datastore) (repo.Repo, error) { c := cfg.Config{} priv, pub, err := ci.GenerateKeyPairWithReader(ci.RSA, 2048, rand.Reader) if err != nil { return nil, err } pid, err := peer.IDFromPublicKey(pub) if err != nil { return nil, err } privkeyb, err := priv.Bytes() if err != nil { return nil, err } c.Bootstrap = cfg.DefaultBootstrapAddresses c.Addresses.Swarm = []string{"/ip4/0.0.0.0/tcp/4001", "/ip4/0.0.0.0/udp/4001/quic"} c.Identity.PeerID = pid.Pretty() c.Identity.PrivKey = base64.StdEncoding.EncodeToString(privkeyb) return &repo.Mock{ D: dstore, C: c, }, nil }