2020-07-24 13:54:03 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2021-06-28 14:01:31 +00:00
|
|
|
"context"
|
2020-07-24 13:54:03 +00:00
|
|
|
"flag"
|
|
|
|
"fmt"
|
2021-05-11 22:59:05 +00:00
|
|
|
"net/http"
|
2020-07-24 13:54:03 +00:00
|
|
|
"os"
|
2021-06-28 14:01:31 +00:00
|
|
|
"os/signal"
|
2020-11-13 12:19:33 +00:00
|
|
|
"strings"
|
2021-06-28 14:01:31 +00:00
|
|
|
"syscall"
|
2020-07-24 13:54:03 +00:00
|
|
|
|
2020-11-13 12:19:33 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
2020-07-24 13:54:03 +00:00
|
|
|
"github.com/nspcc-dev/neofs-node/misc"
|
|
|
|
"github.com/nspcc-dev/neofs-node/pkg/innerring"
|
2021-05-11 22:59:05 +00:00
|
|
|
httputil "github.com/nspcc-dev/neofs-node/pkg/util/http"
|
2020-07-24 13:54:03 +00:00
|
|
|
"github.com/nspcc-dev/neofs-node/pkg/util/logger"
|
2021-05-11 22:59:05 +00:00
|
|
|
"github.com/prometheus/client_golang/prometheus/promhttp"
|
2021-05-12 07:35:18 +00:00
|
|
|
"github.com/spf13/viper"
|
2020-10-13 15:40:35 +00:00
|
|
|
"go.uber.org/zap"
|
2020-07-24 13:54:03 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
// ErrorReturnCode returns when application crashed at initialization stage
|
|
|
|
ErrorReturnCode = 1
|
|
|
|
|
|
|
|
// SuccessReturnCode returns when application closed without panic
|
|
|
|
SuccessReturnCode = 0
|
|
|
|
)
|
|
|
|
|
|
|
|
func exitErr(err error) {
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
os.Exit(ErrorReturnCode)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
configFile := flag.String("config", "", "path to config")
|
2020-11-13 12:19:33 +00:00
|
|
|
validators := flag.String("vote", "", "hex encoded public keys split with comma")
|
2020-07-24 13:54:03 +00:00
|
|
|
versionFlag := flag.Bool("version", false, "neofs-ir node version")
|
|
|
|
flag.Parse()
|
|
|
|
|
|
|
|
if *versionFlag {
|
2021-06-09 11:06:02 +00:00
|
|
|
fmt.Println("neofs-ir", misc.Version)
|
|
|
|
fmt.Println("debug:", misc.Debug)
|
2020-07-24 13:54:03 +00:00
|
|
|
os.Exit(SuccessReturnCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg, err := newConfig(*configFile)
|
|
|
|
exitErr(err)
|
|
|
|
|
2021-05-11 08:54:59 +00:00
|
|
|
var logPrm logger.Prm
|
|
|
|
|
|
|
|
err = logPrm.SetLevelString(
|
|
|
|
cfg.GetString("logger.level"),
|
|
|
|
)
|
|
|
|
exitErr(err)
|
|
|
|
|
|
|
|
log, err := logger.NewLogger(logPrm)
|
2020-07-24 13:54:03 +00:00
|
|
|
exitErr(err)
|
|
|
|
|
2021-06-28 14:01:31 +00:00
|
|
|
ctx, cancel := signal.NotifyContext(context.Background(), syscall.SIGINT, syscall.SIGTERM, syscall.SIGHUP)
|
|
|
|
defer cancel()
|
|
|
|
|
2020-10-13 15:40:35 +00:00
|
|
|
intErr := make(chan error) // internal inner ring errors
|
2020-07-24 13:54:03 +00:00
|
|
|
|
2021-05-12 07:35:18 +00:00
|
|
|
httpServers := initHTTPServers(cfg)
|
2020-07-24 13:54:03 +00:00
|
|
|
|
|
|
|
innerRing, err := innerring.New(ctx, log, cfg)
|
2021-05-31 05:16:10 +00:00
|
|
|
exitErr(err)
|
2020-07-24 13:54:03 +00:00
|
|
|
|
2020-11-13 12:19:33 +00:00
|
|
|
if len(*validators) != 0 {
|
|
|
|
validatorKeys, err := parsePublicKeysFromString(*validators)
|
|
|
|
exitErr(err)
|
|
|
|
|
|
|
|
err = innerRing.InitAndVoteForSidechainValidator(validatorKeys)
|
|
|
|
exitErr(err)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-05-11 22:59:05 +00:00
|
|
|
// start HTTP servers
|
|
|
|
for i := range httpServers {
|
|
|
|
srv := httpServers[i]
|
|
|
|
go func() {
|
|
|
|
exitErr(srv.Serve())
|
|
|
|
}()
|
2020-07-24 13:54:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// start inner ring
|
2020-10-13 15:40:35 +00:00
|
|
|
err = innerRing.Start(ctx, intErr)
|
2021-05-31 05:16:10 +00:00
|
|
|
exitErr(err)
|
2020-07-24 13:54:03 +00:00
|
|
|
|
2021-05-11 09:25:14 +00:00
|
|
|
log.Info("application started",
|
|
|
|
zap.String("build time", misc.Build),
|
|
|
|
zap.String("version", misc.Version),
|
|
|
|
zap.String("debug", misc.Debug),
|
|
|
|
)
|
2020-07-24 13:54:03 +00:00
|
|
|
|
2020-10-13 15:40:35 +00:00
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
case err := <-intErr:
|
|
|
|
// todo: restart application instead of shutdown
|
|
|
|
log.Info("internal error", zap.String("msg", err.Error()))
|
|
|
|
}
|
2020-07-24 13:54:03 +00:00
|
|
|
|
|
|
|
innerRing.Stop()
|
|
|
|
|
2021-05-11 22:59:05 +00:00
|
|
|
// shut down HTTP servers
|
|
|
|
for i := range httpServers {
|
|
|
|
srv := httpServers[i]
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
err := srv.Shutdown()
|
|
|
|
if err != nil {
|
|
|
|
log.Debug("could not shutdown HTTP server",
|
|
|
|
zap.String("error", err.Error()),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2020-07-24 13:54:03 +00:00
|
|
|
log.Info("application stopped")
|
|
|
|
}
|
2020-11-13 12:19:33 +00:00
|
|
|
|
2021-03-25 07:34:27 +00:00
|
|
|
func parsePublicKeysFromString(argument string) (keys.PublicKeys, error) {
|
2020-11-13 12:19:33 +00:00
|
|
|
publicKeysString := strings.Split(argument, ",")
|
|
|
|
|
|
|
|
return innerring.ParsePublicKeysFromStrings(publicKeysString)
|
|
|
|
}
|
2021-05-12 07:35:18 +00:00
|
|
|
|
|
|
|
func initHTTPServers(cfg *viper.Viper) []*httputil.Server {
|
2021-05-12 07:37:58 +00:00
|
|
|
items := []struct {
|
2021-05-12 07:35:18 +00:00
|
|
|
cfgPrefix string
|
|
|
|
handler func() http.Handler
|
|
|
|
}{
|
|
|
|
{"profiler", httputil.Handler},
|
|
|
|
{"metrics", promhttp.Handler},
|
2021-05-12 07:37:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
httpServers := make([]*httputil.Server, 0, len(items))
|
|
|
|
|
|
|
|
for _, item := range items {
|
2021-05-12 07:35:18 +00:00
|
|
|
addr := cfg.GetString(item.cfgPrefix + ".address")
|
|
|
|
if addr == "" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
var prm httputil.Prm
|
|
|
|
|
|
|
|
prm.Address = addr
|
|
|
|
prm.Handler = item.handler()
|
|
|
|
|
|
|
|
httpServers = append(httpServers,
|
|
|
|
httputil.New(prm,
|
|
|
|
httputil.WithShutdownTimeout(
|
|
|
|
cfg.GetDuration(item.cfgPrefix+".shutdown_timeout"),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
return httpServers
|
|
|
|
}
|