splitting the modules
This commit is contained in:
parent
a53a500fd7
commit
8705f87504
12 changed files with 402 additions and 396 deletions
|
|
@ -1,4 +1,4 @@
|
||||||
package abciapp
|
package blockchain
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
141
blockchain/main.go
Normal file
141
blockchain/main.go
Normal file
|
|
@ -0,0 +1,141 @@
|
||||||
|
package blockchain
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"lbc/cfg"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
"github.com/dgraph-io/badger"
|
||||||
|
abci "github.com/tendermint/tendermint/abci/types"
|
||||||
|
"github.com/tendermint/tendermint/libs/log"
|
||||||
|
nm "github.com/tendermint/tendermint/node"
|
||||||
|
"github.com/tendermint/tendermint/p2p"
|
||||||
|
"github.com/tendermint/tendermint/privval"
|
||||||
|
"github.com/tendermint/tendermint/proxy"
|
||||||
|
tmTypes "github.com/tendermint/tendermint/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
func openBadger(path string) (*badger.DB, error) {
|
||||||
|
return badger.Open(badger.DefaultOptions(path).WithTruncate(true))
|
||||||
|
}
|
||||||
|
|
||||||
|
func newTendermint(app abci.Application, config *cfg.Config, laddrReturner chan string) (*nm.Node, error) {
|
||||||
|
config.P2P.ListenAddress = "tcp://" + <-laddrReturner
|
||||||
|
|
||||||
|
//if config.P2P.PersistentPeers == "" {
|
||||||
|
config.P2P.PersistentPeers = <-laddrReturner
|
||||||
|
//} else {
|
||||||
|
// <- laddrReturner
|
||||||
|
//}
|
||||||
|
|
||||||
|
var pv tmTypes.PrivValidator
|
||||||
|
if _, err := os.Stat(config.PrivValidatorKeyFile()); err == nil {
|
||||||
|
pv = privval.LoadFilePV(
|
||||||
|
config.PrivValidatorKeyFile(),
|
||||||
|
config.PrivValidatorStateFile(),
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
fmt.Println("⚠️ priv_validator_key.json not found. Node will run as non-validator.")
|
||||||
|
pv = tmTypes.NewMockPV()
|
||||||
|
}
|
||||||
|
|
||||||
|
nodeKey, err := p2p.LoadNodeKey(config.NodeKeyFile())
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("load node key: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
clientCreator := proxy.NewLocalClientCreator(app)
|
||||||
|
logger := log.NewTMLogger(log.NewSyncWriter(os.Stdout))
|
||||||
|
|
||||||
|
return nm.NewNode(
|
||||||
|
config,
|
||||||
|
pv,
|
||||||
|
nodeKey,
|
||||||
|
clientCreator,
|
||||||
|
nm.DefaultGenesisDocProviderFunc(config),
|
||||||
|
nm.DefaultDBProvider,
|
||||||
|
nm.DefaultMetricsProvider(config.Instrumentation),
|
||||||
|
logger,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetNodeInfo(config *cfg.Config, dbPath string) (p2p.NodeInfo, error) {
|
||||||
|
db, err := openBadger(dbPath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to open badger db to get node info", err)
|
||||||
|
}
|
||||||
|
defer db.Close()
|
||||||
|
|
||||||
|
app := NewKVStoreApplication(db)
|
||||||
|
|
||||||
|
nodeKey, err := p2p.LoadNodeKey(config.NodeKeyFile())
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("load node key: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
clientCreator := proxy.NewLocalClientCreator(app)
|
||||||
|
logger := log.NewTMLogger(log.NewSyncWriter(os.Stdout))
|
||||||
|
var pv tmTypes.PrivValidator
|
||||||
|
if _, err := os.Stat(config.PrivValidatorKeyFile()); err == nil {
|
||||||
|
pv = privval.LoadFilePV(
|
||||||
|
config.PrivValidatorKeyFile(),
|
||||||
|
config.PrivValidatorStateFile(),
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
fmt.Println("⚠️ priv_validator_key.json not found. Node will run as non-validator.")
|
||||||
|
pv = tmTypes.NewMockPV()
|
||||||
|
}
|
||||||
|
|
||||||
|
config.P2P.PersistentPeers = ""
|
||||||
|
|
||||||
|
node, err := nm.NewNode(
|
||||||
|
config,
|
||||||
|
pv,
|
||||||
|
nodeKey,
|
||||||
|
clientCreator,
|
||||||
|
nm.DefaultGenesisDocProviderFunc(config),
|
||||||
|
nm.DefaultDBProvider,
|
||||||
|
nm.DefaultMetricsProvider(config.Instrumentation),
|
||||||
|
logger,
|
||||||
|
)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return node.NodeInfo(), nil
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// В blockchain/run.go
|
||||||
|
func Run(ctx context.Context, dbPath string, config *cfg.Config, laddrReturner chan string) error {
|
||||||
|
db, err := openBadger(dbPath)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("open badger db: %w", err)
|
||||||
|
}
|
||||||
|
defer db.Close()
|
||||||
|
|
||||||
|
app := NewKVStoreApplication(db)
|
||||||
|
node, err := newTendermint(app, config, laddrReturner)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("build node: %w", err)
|
||||||
|
}
|
||||||
|
if err := node.Start(); err != nil {
|
||||||
|
return fmt.Errorf("start node: %w", err)
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
|
||||||
|
}()
|
||||||
|
|
||||||
|
// ждём контекст или внутренних ошибок узла
|
||||||
|
select {
|
||||||
|
case <-ctx.Done():
|
||||||
|
case <-node.Quit(): // Tendermint закрылся сам
|
||||||
|
return err
|
||||||
|
// Node quit signal received
|
||||||
|
}
|
||||||
|
node.Stop()
|
||||||
|
node.Wait()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
@ -1,10 +1,11 @@
|
||||||
package configfunctions
|
package cfg
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"flag"
|
"flag"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"io"
|
||||||
"lbc/yggdrasil"
|
"lbc/yggdrasil"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
|
@ -18,11 +19,44 @@ import (
|
||||||
tmTypes "github.com/tendermint/tendermint/types"
|
tmTypes "github.com/tendermint/tendermint/types"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type Config = cfg.Config
|
||||||
|
|
||||||
var (
|
var (
|
||||||
yggListenPort = 4224
|
yggListenPort = 4224
|
||||||
yggKeyPath = flag.String("ygg-key", "./config/yggdrasil.key", "Path to Yggdrasil key file")
|
yggKeyPath = flag.String("ygg-key", "./config/yggdrasil.key", "Path to Yggdrasil key file")
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func copyFile(src, dst string) error {
|
||||||
|
in, err := os.Open(src)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer in.Close()
|
||||||
|
|
||||||
|
if err = os.MkdirAll(filepath.Dir(dst), 0o700); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
out, err := os.Create(dst)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
_ = out.Sync()
|
||||||
|
_ = out.Close()
|
||||||
|
}()
|
||||||
|
|
||||||
|
_, err = io.Copy(out, in)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func LoadViperConfig(path string) (*viper.Viper, error) {
|
||||||
|
v := viper.New()
|
||||||
|
v.SetConfigFile(path)
|
||||||
|
err := v.ReadInConfig()
|
||||||
|
return v, err
|
||||||
|
}
|
||||||
|
|
||||||
func InitTendermintFiles(config *cfg.Config, isGenesis bool, chainName string) error {
|
func InitTendermintFiles(config *cfg.Config, isGenesis bool, chainName string) error {
|
||||||
if err := os.MkdirAll(filepath.Dir(config.PrivValidatorKeyFile()), 0700); err != nil {
|
if err := os.MkdirAll(filepath.Dir(config.PrivValidatorKeyFile()), 0700); err != nil {
|
||||||
return err
|
return err
|
||||||
|
|
@ -214,3 +248,59 @@ func UpdateGenesisJson(nodeInfo p2p.NodeInfo, v *viper.Viper, defaultConfigDirec
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func InitGenesis(chainName, defaultConfigPath string) (*cfg.Config, *viper.Viper) {
|
||||||
|
config := cfg.DefaultConfig()
|
||||||
|
config.RootDir = filepath.Dir(filepath.Dir(defaultConfigPath))
|
||||||
|
|
||||||
|
nodeinfo := p2p.DefaultNodeInfo{}
|
||||||
|
viper := WriteConfig(config, &defaultConfigPath, nodeinfo)
|
||||||
|
if err := InitTendermintFiles(config, true, chainName); err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "Failed to init files: %v\n", err)
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return config, viper
|
||||||
|
}
|
||||||
|
|
||||||
|
func InitJoiner(chainName, defaultConfigPath, path string) error {
|
||||||
|
config := cfg.DefaultConfig()
|
||||||
|
config.RootDir = filepath.Dir(filepath.Dir(defaultConfigPath))
|
||||||
|
|
||||||
|
if err := copyFile(path, config.GenesisFile()); err != nil {
|
||||||
|
fmt.Fprintln(os.Stderr, "не удалось скопировать genesis.json:", err)
|
||||||
|
os.Exit(3)
|
||||||
|
}
|
||||||
|
|
||||||
|
nodeinfo := p2p.DefaultNodeInfo{}
|
||||||
|
WriteConfig(config, &defaultConfigPath, nodeinfo)
|
||||||
|
//viper := cfg.WriteConfig(config, &defaultConfigPath, nodeinfo)
|
||||||
|
if err := InitTendermintFiles(config, false, chainName); err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "Failed to init files: %v\n", err)
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
//node, db, err := buildNode()
|
||||||
|
//if err != nil {
|
||||||
|
// panic(err)
|
||||||
|
//}
|
||||||
|
//defer db.Close()
|
||||||
|
|
||||||
|
//cfg.WriteConfig(config, &defaultConfigPath, node.NodeInfo())
|
||||||
|
|
||||||
|
//if err := os.MkdirAll(filepath.Join(config.RootDir, "data"), 0o700); err != nil {
|
||||||
|
// fmt.Fprintln(os.Stderr, "не удалось создать директорию data")
|
||||||
|
// os.Exit(1)
|
||||||
|
//}
|
||||||
|
|
||||||
|
if _, err := p2p.LoadOrGenNodeKey(config.NodeKeyFile()); err != nil {
|
||||||
|
return fmt.Errorf("ошибка генерации node_key.json")
|
||||||
|
}
|
||||||
|
|
||||||
|
pv := privval.GenFilePV(
|
||||||
|
config.PrivValidatorKeyFile(),
|
||||||
|
config.PrivValidatorStateFile(),
|
||||||
|
)
|
||||||
|
pv.Save()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
17
cli/init.go
17
cli/init.go
|
|
@ -2,7 +2,10 @@ package cli
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"lbc/blockchain"
|
||||||
|
"lbc/cfg"
|
||||||
"os"
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
)
|
)
|
||||||
|
|
@ -14,13 +17,23 @@ var initCmd = &cobra.Command{
|
||||||
Run: func(cmd *cobra.Command, args []string) {
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
switch args[0] {
|
switch args[0] {
|
||||||
case "genesis":
|
case "genesis":
|
||||||
initGenesis()
|
config, viper := cfg.InitGenesis(chainName, defaultConfigPath)
|
||||||
|
nodeinfo, err := blockchain.GetNodeInfo(config, dbPath)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("Error: %w", err)
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
cfg.UpdateGenesisJson(nodeinfo, viper, filepath.Dir(defaultConfigPath))
|
||||||
|
fmt.Println("Genesis node initialized.")
|
||||||
case "join":
|
case "join":
|
||||||
if len(args) < 2 {
|
if len(args) < 2 {
|
||||||
fmt.Println("Укажите путь к genesis.json")
|
fmt.Println("Укажите путь к genesis.json")
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
initJoiner(args[1])
|
cfg.InitJoiner(chainName, defaultConfigPath, args[1])
|
||||||
|
|
||||||
|
fmt.Println("Joiner node initialized.")
|
||||||
default:
|
default:
|
||||||
fmt.Printf("Неизвестный режим init: %s\n", args[0])
|
fmt.Printf("Неизвестный режим init: %s\n", args[0])
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
|
|
|
||||||
51
cli/root.go
51
cli/root.go
|
|
@ -1,17 +1,66 @@
|
||||||
package cli
|
package cli
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"lbc/blockchain"
|
||||||
|
"lbc/cfg"
|
||||||
|
"lbc/yggdrasil"
|
||||||
"os"
|
"os"
|
||||||
|
"os/signal"
|
||||||
|
"syscall"
|
||||||
|
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var defaultConfigPath string
|
||||||
|
var dbPath string
|
||||||
|
var chainName string
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
rootCmd.PersistentFlags().StringVar(&defaultConfigPath, "config", "./config/config.toml", "Путь к конфигурационному файлу")
|
||||||
|
rootCmd.PersistentFlags().StringVar(&dbPath, "badger", "./badger", "Путь к базе данных BadgerDB")
|
||||||
|
rootCmd.PersistentFlags().StringVar(&chainName, "chainname", "lbc-chain", "Название цепочки блоков")
|
||||||
|
}
|
||||||
|
|
||||||
var rootCmd = &cobra.Command{
|
var rootCmd = &cobra.Command{
|
||||||
Use: "lbc",
|
Use: "lbc",
|
||||||
Short: "Лёгкий блокчейн координатор",
|
Short: "Лёгкий блокчейн координатор",
|
||||||
Run: func(cmd *cobra.Command, args []string) {
|
Run: func(cmd *cobra.Command, args []string) {
|
||||||
runNode()
|
//Run()
|
||||||
|
v, err := cfg.LoadViperConfig(defaultConfigPath)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, `Конфигурационный файл не найден: %v
|
||||||
|
|
||||||
|
Похоже, что нода ещё не инициализирована.
|
||||||
|
|
||||||
|
Чтобы создать необходимые файлы, запусти одну из следующих команд:
|
||||||
|
|
||||||
|
lbc --init=genesis # если это новая цепочка
|
||||||
|
lbc --init=join # если ты присоединяешься к существующей
|
||||||
|
|
||||||
|
По умолчанию файл конфигурации ищется по пути: %s
|
||||||
|
`, err, defaultConfigPath)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
config, err := cfg.ReadConfig(defaultConfigPath)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("конфигурация не прочитана: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
laddrReturner := make(chan string, 2)
|
||||||
|
go yggdrasil.Yggdrasil(v, laddrReturner)
|
||||||
|
go blockchain.Run(ctx, dbPath, config, laddrReturner)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
// TODO exitWithError("error creating node", err)
|
||||||
|
}
|
||||||
|
sigCh := make(chan os.Signal, 1)
|
||||||
|
signal.Notify(sigCh, syscall.SIGINT, syscall.SIGTERM)
|
||||||
|
<-sigCh // ждём SIGINT/SIGTERM
|
||||||
|
cancel() // говорим горутинам завершаться
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
242
cli/run.go
242
cli/run.go
|
|
@ -1,242 +0,0 @@
|
||||||
package cli
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"io"
|
|
||||||
|
|
||||||
abciApp "lbc/abciapp"
|
|
||||||
cfg "lbc/configfunctions"
|
|
||||||
"lbc/yggdrasil"
|
|
||||||
"os"
|
|
||||||
"os/signal"
|
|
||||||
"path/filepath"
|
|
||||||
"syscall"
|
|
||||||
|
|
||||||
"github.com/dgraph-io/badger"
|
|
||||||
"github.com/spf13/viper"
|
|
||||||
abci "github.com/tendermint/tendermint/abci/types"
|
|
||||||
"github.com/tendermint/tendermint/libs/log"
|
|
||||||
nm "github.com/tendermint/tendermint/node"
|
|
||||||
"github.com/tendermint/tendermint/p2p"
|
|
||||||
"github.com/tendermint/tendermint/privval"
|
|
||||||
"github.com/tendermint/tendermint/proxy"
|
|
||||||
tmTypes "github.com/tendermint/tendermint/types"
|
|
||||||
)
|
|
||||||
|
|
||||||
var defaultConfigPath string
|
|
||||||
var dbPath string
|
|
||||||
var chainName string
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
rootCmd.PersistentFlags().StringVar(&defaultConfigPath, "config", "./config/config.toml", "Путь к конфигурационному файлу")
|
|
||||||
rootCmd.PersistentFlags().StringVar(&dbPath, "badger", "./badger", "Путь к базе данных BadgerDB")
|
|
||||||
rootCmd.PersistentFlags().StringVar(&chainName, "chainname", "lbc-chain", "Название цепочки блоков")
|
|
||||||
}
|
|
||||||
|
|
||||||
func newTendermint(app abci.Application, configFile string, v *viper.Viper) (*nm.Node, error) {
|
|
||||||
config, err := cfg.ReadConfig(configFile)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("конфигурация не прочитана: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
laddrReturner := make(chan string, 2)
|
|
||||||
|
|
||||||
config.P2P.PersistentPeers = cfg.ReadP2Peers(configFile)
|
|
||||||
//v.Set("p2p.persistent_peers", config.P2P.PersistentPeers)
|
|
||||||
//v.Set("")
|
|
||||||
|
|
||||||
go yggdrasil.Yggdrasil(v, laddrReturner)
|
|
||||||
config.P2P.ListenAddress = "tcp://" + <-laddrReturner
|
|
||||||
|
|
||||||
//if config.P2P.PersistentPeers == "" {
|
|
||||||
config.P2P.PersistentPeers = <-laddrReturner
|
|
||||||
//} else {
|
|
||||||
// <- laddrReturner
|
|
||||||
//}
|
|
||||||
|
|
||||||
var pv tmTypes.PrivValidator
|
|
||||||
if _, err := os.Stat(config.PrivValidatorKeyFile()); err == nil {
|
|
||||||
pv = privval.LoadFilePV(
|
|
||||||
config.PrivValidatorKeyFile(),
|
|
||||||
config.PrivValidatorStateFile(),
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
fmt.Println("⚠️ priv_validator_key.json not found. Node will run as non-validator.")
|
|
||||||
pv = tmTypes.NewMockPV()
|
|
||||||
}
|
|
||||||
|
|
||||||
nodeKey, err := p2p.LoadNodeKey(config.NodeKeyFile())
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("load node key: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
clientCreator := proxy.NewLocalClientCreator(app)
|
|
||||||
logger := log.NewTMLogger(log.NewSyncWriter(os.Stdout))
|
|
||||||
|
|
||||||
return nm.NewNode(
|
|
||||||
config,
|
|
||||||
pv,
|
|
||||||
nodeKey,
|
|
||||||
clientCreator,
|
|
||||||
nm.DefaultGenesisDocProviderFunc(config),
|
|
||||||
nm.DefaultDBProvider,
|
|
||||||
nm.DefaultMetricsProvider(config.Instrumentation),
|
|
||||||
logger,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
func exitWithError(msg string, err error) {
|
|
||||||
fmt.Fprintf(os.Stderr, "%s: %v\n", msg, err)
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
func loadViperConfig(path string) (*viper.Viper, error) {
|
|
||||||
v := viper.New()
|
|
||||||
v.SetConfigFile(path)
|
|
||||||
err := v.ReadInConfig()
|
|
||||||
return v, err
|
|
||||||
}
|
|
||||||
|
|
||||||
func openBadger(path string) (*badger.DB, error) {
|
|
||||||
return badger.Open(badger.DefaultOptions(path).WithTruncate(true))
|
|
||||||
}
|
|
||||||
|
|
||||||
func buildNode() (*nm.Node, *badger.DB, error) {
|
|
||||||
v, err := loadViperConfig(defaultConfigPath)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Fprintf(os.Stderr, `Конфигурационный файл не найден: %v
|
|
||||||
|
|
||||||
Похоже, что нода ещё не инициализирована.
|
|
||||||
|
|
||||||
Чтобы создать необходимые файлы, запусти одну из следующих команд:
|
|
||||||
|
|
||||||
lbc --init=genesis # если это новая цепочка
|
|
||||||
lbc --init=join # если ты присоединяешься к существующей
|
|
||||||
|
|
||||||
По умолчанию файл конфигурации ищется по пути: %s
|
|
||||||
`, err, defaultConfigPath)
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
db, err := openBadger(dbPath)
|
|
||||||
if err != nil {
|
|
||||||
exitWithError("failed to open badger db", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
app := abciApp.NewKVStoreApplication(db)
|
|
||||||
|
|
||||||
node, err := newTendermint(app, defaultConfigPath, v)
|
|
||||||
if err != nil {
|
|
||||||
exitWithError("error creating node", err)
|
|
||||||
}
|
|
||||||
return node, db, err
|
|
||||||
}
|
|
||||||
|
|
||||||
func copyFile(src, dst string) error {
|
|
||||||
in, err := os.Open(src)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
defer in.Close()
|
|
||||||
|
|
||||||
if err = os.MkdirAll(filepath.Dir(dst), 0o700); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
out, err := os.Create(dst)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
defer func() {
|
|
||||||
_ = out.Sync()
|
|
||||||
_ = out.Close()
|
|
||||||
}()
|
|
||||||
|
|
||||||
_, err = io.Copy(out, in)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
func runNode() {
|
|
||||||
node, db, err := buildNode()
|
|
||||||
if err != nil {
|
|
||||||
exitWithError("failed to build node", err)
|
|
||||||
}
|
|
||||||
defer db.Close()
|
|
||||||
|
|
||||||
if err := node.Start(); err != nil {
|
|
||||||
exitWithError("failed to start node", err)
|
|
||||||
}
|
|
||||||
defer func() {
|
|
||||||
node.Stop()
|
|
||||||
node.Wait()
|
|
||||||
}()
|
|
||||||
|
|
||||||
sigCh := make(chan os.Signal, 1)
|
|
||||||
signal.Notify(sigCh, syscall.SIGINT, syscall.SIGTERM)
|
|
||||||
<-sigCh
|
|
||||||
}
|
|
||||||
|
|
||||||
func initGenesis() {
|
|
||||||
config := cfg.DefaultConfig()
|
|
||||||
config.RootDir = filepath.Dir(filepath.Dir(defaultConfigPath))
|
|
||||||
|
|
||||||
nodeinfo := p2p.DefaultNodeInfo{}
|
|
||||||
viper := cfg.WriteConfig(config, &defaultConfigPath, nodeinfo)
|
|
||||||
if err := cfg.InitTendermintFiles(config, true, chainName); err != nil {
|
|
||||||
fmt.Fprintf(os.Stderr, "Failed to init files: %v\n", err)
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
node, db, err := buildNode()
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
db.Close()
|
|
||||||
|
|
||||||
cfg.UpdateGenesisJson(node.NodeInfo(), viper, filepath.Dir(defaultConfigPath))
|
|
||||||
fmt.Println("Genesis node initialized.")
|
|
||||||
}
|
|
||||||
|
|
||||||
func initJoiner(path string) {
|
|
||||||
config := cfg.DefaultConfig()
|
|
||||||
config.RootDir = filepath.Dir(filepath.Dir(defaultConfigPath))
|
|
||||||
|
|
||||||
if err := copyFile(path, config.GenesisFile()); err != nil {
|
|
||||||
fmt.Fprintln(os.Stderr, "не удалось скопировать genesis.json:", err)
|
|
||||||
os.Exit(3)
|
|
||||||
}
|
|
||||||
|
|
||||||
nodeinfo := p2p.DefaultNodeInfo{}
|
|
||||||
cfg.WriteConfig(config, &defaultConfigPath, nodeinfo)
|
|
||||||
//viper := cfg.WriteConfig(config, &defaultConfigPath, nodeinfo)
|
|
||||||
if err := cfg.InitTendermintFiles(config, false, chainName); err != nil {
|
|
||||||
fmt.Fprintf(os.Stderr, "Failed to init files: %v\n", err)
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
node, db, err := buildNode()
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
defer db.Close()
|
|
||||||
|
|
||||||
cfg.WriteConfig(config, &defaultConfigPath, node.NodeInfo())
|
|
||||||
|
|
||||||
//if err := os.MkdirAll(filepath.Join(config.RootDir, "data"), 0o700); err != nil {
|
|
||||||
// fmt.Fprintln(os.Stderr, "не удалось создать директорию data")
|
|
||||||
// os.Exit(1)
|
|
||||||
//}
|
|
||||||
|
|
||||||
if _, err := p2p.LoadOrGenNodeKey(config.NodeKeyFile()); err != nil {
|
|
||||||
fmt.Fprintln(os.Stderr, "ошибка генерации node_key.json")
|
|
||||||
os.Exit(2)
|
|
||||||
}
|
|
||||||
|
|
||||||
pv := privval.GenFilePV(
|
|
||||||
config.PrivValidatorKeyFile(),
|
|
||||||
config.PrivValidatorStateFile(),
|
|
||||||
)
|
|
||||||
pv.Save()
|
|
||||||
|
|
||||||
fmt.Println("Joiner node initialized.")
|
|
||||||
}
|
|
||||||
36
genesis.json
36
genesis.json
|
|
@ -1,36 +0,0 @@
|
||||||
{
|
|
||||||
"app_hash": "",
|
|
||||||
"chain_id": "lbc-chain",
|
|
||||||
"consensus_params": {
|
|
||||||
"block": {
|
|
||||||
"max_bytes": "22020096",
|
|
||||||
"max_gas": "-1",
|
|
||||||
"time_iota_ms": "1000"
|
|
||||||
},
|
|
||||||
"evidence": {
|
|
||||||
"max_age_duration": "172800000000000",
|
|
||||||
"max_age_num_blocks": "100000",
|
|
||||||
"max_bytes": "1048576"
|
|
||||||
},
|
|
||||||
"validator": {
|
|
||||||
"pub_key_types": [
|
|
||||||
"ed25519"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"version": {}
|
|
||||||
},
|
|
||||||
"genesis_time": "2025-07-12T17:34:34.7567451Z",
|
|
||||||
"initial_height": "0",
|
|
||||||
"p2peers": "f8366fe22e5a3cd64281ecfc6e50075903848ab9@ygg://[200:6687:b756:51a2:1147:aef0:9011:c142]:4224",
|
|
||||||
"validators": [
|
|
||||||
{
|
|
||||||
"address": "F1585752261F319FE0A9E2097825A2807622AE5A",
|
|
||||||
"name": "DESKTOP-HN9RCT8",
|
|
||||||
"power": "10",
|
|
||||||
"pub_key": {
|
|
||||||
"type": "tendermint/PubKeyEd25519",
|
|
||||||
"value": "gnPCTB+kqaiw2TmFwrZYoCcWiPO3SdTf1IB7rEdBulE="
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
package autopeering
|
package yggdrasil
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
|
@ -50,7 +50,7 @@ func readPeersFile(path string) []url.URL {
|
||||||
// connection strings. If fetching fails, it falls back to reading peers from
|
// connection strings. If fetching fails, it falls back to reading peers from
|
||||||
// peers.txt in the current working directory. It returns an empty slice if no
|
// peers.txt in the current working directory. It returns an empty slice if no
|
||||||
// peers can be retrieved.
|
// peers can be retrieved.
|
||||||
func GetPublicPeers() []url.URL {
|
func getPublicPeers() []url.URL {
|
||||||
tempDir, err := os.MkdirTemp("", "public-peers-*")
|
tempDir, err := os.MkdirTemp("", "public-peers-*")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return readPeersFile("peers.txt")
|
return readPeersFile("peers.txt")
|
||||||
97
yggdrasil/keys.go
Normal file
97
yggdrasil/keys.go
Normal file
|
|
@ -0,0 +1,97 @@
|
||||||
|
package yggdrasil
|
||||||
|
|
||||||
|
import (
|
||||||
|
"crypto/ed25519"
|
||||||
|
"encoding/hex"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/gologme/log"
|
||||||
|
"github.com/spf13/viper"
|
||||||
|
|
||||||
|
yggConfig "github.com/yggdrasil-network/yggdrasil-go/src/config"
|
||||||
|
"github.com/yggdrasil-network/yggdrasil-go/src/core"
|
||||||
|
)
|
||||||
|
|
||||||
|
func GeneratePrivateKey() yggConfig.KeyBytes {
|
||||||
|
return yggConfig.GenerateConfig().PrivateKey
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetPublicKey(keyPath string) (ed25519.PublicKey, error) {
|
||||||
|
data, err := os.ReadFile(keyPath)
|
||||||
|
if err != nil {
|
||||||
|
return ed25519.PublicKey{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
decoded, err := hex.DecodeString(strings.TrimSpace(string(data)))
|
||||||
|
if err != nil {
|
||||||
|
return ed25519.PublicKey{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(decoded) != ed25519.PrivateKeySize {
|
||||||
|
return ed25519.PublicKey{}, fmt.Errorf("invalid private key size: %d", len(decoded))
|
||||||
|
}
|
||||||
|
|
||||||
|
privateKey := ed25519.PrivateKey(decoded)
|
||||||
|
return privateKey.Public().(ed25519.PublicKey), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetYggdrasilAddress(config *viper.Viper) string {
|
||||||
|
//var remoteTcp types.TCPRemoteMappings
|
||||||
|
ygg := config.Sub("yggdrasil")
|
||||||
|
if ygg == nil {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
//laddr := config.Sub("p2p").GetString("laddr")
|
||||||
|
//remoteTcp.Set(laddr)
|
||||||
|
|
||||||
|
cfg := yggConfig.GenerateConfig()
|
||||||
|
|
||||||
|
cfg.AdminListen = ygg.GetString("admin_listen")
|
||||||
|
cfg.Listen = ygg.GetStringSlice("listen")
|
||||||
|
cfg.Peers = ygg.GetStringSlice("peers")
|
||||||
|
cfg.AllowedPublicKeys = ygg.GetStringSlice("allowed-public-keys")
|
||||||
|
cfg.PrivateKeyPath = ygg.GetString("private-key-file")
|
||||||
|
|
||||||
|
logger := log.Default()
|
||||||
|
|
||||||
|
n := &node{}
|
||||||
|
|
||||||
|
// Setup the Yggdrasil node itself.
|
||||||
|
{
|
||||||
|
options := []core.SetupOption{
|
||||||
|
core.NodeInfo(cfg.NodeInfo),
|
||||||
|
core.NodeInfoPrivacy(cfg.NodeInfoPrivacy),
|
||||||
|
}
|
||||||
|
for _, addr := range cfg.Listen {
|
||||||
|
options = append(options, core.ListenAddress(addr))
|
||||||
|
}
|
||||||
|
for _, peer := range cfg.Peers {
|
||||||
|
options = append(options, core.Peer{URI: peer})
|
||||||
|
}
|
||||||
|
for intf, peers := range cfg.InterfacePeers {
|
||||||
|
for _, peer := range peers {
|
||||||
|
options = append(options, core.Peer{URI: peer, SourceInterface: intf})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for _, allowed := range cfg.AllowedPublicKeys {
|
||||||
|
k, err := hex.DecodeString(allowed)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
options = append(options, core.AllowedPublicKey(k[:]))
|
||||||
|
}
|
||||||
|
|
||||||
|
var err error
|
||||||
|
if n.core, err = core.New(cfg.Certificate, logger, options...); err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
address := n.core.Address()
|
||||||
|
n.core.Stop()
|
||||||
|
return address.String()
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -2,17 +2,12 @@ package yggdrasil
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"crypto/ed25519"
|
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"errors"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"lbc/autopeering"
|
|
||||||
ppp "lbc/persistentpeersparser"
|
|
||||||
"net"
|
"net"
|
||||||
"os"
|
"os"
|
||||||
"os/signal"
|
"os/signal"
|
||||||
"regexp"
|
"regexp"
|
||||||
"runtime"
|
|
||||||
"strings"
|
"strings"
|
||||||
"syscall"
|
"syscall"
|
||||||
|
|
||||||
|
|
@ -45,88 +40,6 @@ type TCPRemoteListenerMapping struct {
|
||||||
Mapped *net.TCPListener
|
Mapped *net.TCPListener
|
||||||
}
|
}
|
||||||
|
|
||||||
func GeneratePrivateKey() yggConfig.KeyBytes {
|
|
||||||
return yggConfig.GenerateConfig().PrivateKey
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetPublicKey(keyPath string) (ed25519.PublicKey, error) {
|
|
||||||
data, err := os.ReadFile(keyPath)
|
|
||||||
if err != nil {
|
|
||||||
return ed25519.PublicKey{}, err
|
|
||||||
}
|
|
||||||
|
|
||||||
decoded, err := hex.DecodeString(strings.TrimSpace(string(data)))
|
|
||||||
if err != nil {
|
|
||||||
return ed25519.PublicKey{}, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(decoded) != ed25519.PrivateKeySize {
|
|
||||||
return ed25519.PublicKey{}, fmt.Errorf("invalid private key size: %d", len(decoded))
|
|
||||||
}
|
|
||||||
|
|
||||||
privateKey := ed25519.PrivateKey(decoded)
|
|
||||||
return privateKey.Public().(ed25519.PublicKey), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetYggdrasilAddress(config *viper.Viper) string {
|
|
||||||
//var remoteTcp types.TCPRemoteMappings
|
|
||||||
ygg := config.Sub("yggdrasil")
|
|
||||||
if ygg == nil {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
//laddr := config.Sub("p2p").GetString("laddr")
|
|
||||||
//remoteTcp.Set(laddr)
|
|
||||||
|
|
||||||
cfg := yggConfig.GenerateConfig()
|
|
||||||
|
|
||||||
cfg.AdminListen = ygg.GetString("admin_listen")
|
|
||||||
cfg.Listen = ygg.GetStringSlice("listen")
|
|
||||||
cfg.Peers = ygg.GetStringSlice("peers")
|
|
||||||
cfg.AllowedPublicKeys = ygg.GetStringSlice("allowed-public-keys")
|
|
||||||
cfg.PrivateKeyPath = ygg.GetString("private-key-file")
|
|
||||||
|
|
||||||
logger := log.Default()
|
|
||||||
|
|
||||||
n := &node{}
|
|
||||||
|
|
||||||
// Setup the Yggdrasil node itself.
|
|
||||||
{
|
|
||||||
options := []core.SetupOption{
|
|
||||||
core.NodeInfo(cfg.NodeInfo),
|
|
||||||
core.NodeInfoPrivacy(cfg.NodeInfoPrivacy),
|
|
||||||
}
|
|
||||||
for _, addr := range cfg.Listen {
|
|
||||||
options = append(options, core.ListenAddress(addr))
|
|
||||||
}
|
|
||||||
for _, peer := range cfg.Peers {
|
|
||||||
options = append(options, core.Peer{URI: peer})
|
|
||||||
}
|
|
||||||
for intf, peers := range cfg.InterfacePeers {
|
|
||||||
for _, peer := range peers {
|
|
||||||
options = append(options, core.Peer{URI: peer, SourceInterface: intf})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for _, allowed := range cfg.AllowedPublicKeys {
|
|
||||||
k, err := hex.DecodeString(allowed)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
options = append(options, core.AllowedPublicKey(k[:]))
|
|
||||||
}
|
|
||||||
|
|
||||||
var err error
|
|
||||||
if n.core, err = core.New(cfg.Certificate, logger, options...); err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
address := n.core.Address()
|
|
||||||
n.core.Stop()
|
|
||||||
return address.String()
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// The main function is responsible for configuring and starting Yggdrasil.
|
// The main function is responsible for configuring and starting Yggdrasil.
|
||||||
func Yggdrasil(config *viper.Viper, ch chan string) {
|
func Yggdrasil(config *viper.Viper, ch chan string) {
|
||||||
var remoteTcp types.TCPRemoteMappings
|
var remoteTcp types.TCPRemoteMappings
|
||||||
|
|
@ -157,9 +70,9 @@ func Yggdrasil(config *viper.Viper, ch chan string) {
|
||||||
ch <- remoteTcp[0].Mapped.String()
|
ch <- remoteTcp[0].Mapped.String()
|
||||||
|
|
||||||
peers := p2p.GetString("persistent_peers")
|
peers := p2p.GetString("persistent_peers")
|
||||||
parsed, err := ppp.ParseEntries(peers)
|
parsed, err := ParseEntries(peers)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
parsed = []ppp.ParsedEntry{}
|
parsed = []ParsedEntry{}
|
||||||
ch <- ""
|
ch <- ""
|
||||||
log.Warnln("Warning: persistent peers has an error")
|
log.Warnln("Warning: persistent peers has an error")
|
||||||
}
|
}
|
||||||
|
|
@ -169,7 +82,7 @@ func Yggdrasil(config *viper.Viper, ch chan string) {
|
||||||
cfg.AdminListen = ygg.GetString("admin_listen")
|
cfg.AdminListen = ygg.GetString("admin_listen")
|
||||||
cfg.Listen = ygg.GetStringSlice("listen")
|
cfg.Listen = ygg.GetStringSlice("listen")
|
||||||
if ygg.GetString("peers") == "auto" {
|
if ygg.GetString("peers") == "auto" {
|
||||||
publicPeers := autopeering.GetPublicPeers()
|
publicPeers := getPublicPeers()
|
||||||
var urlsAsStrings []string
|
var urlsAsStrings []string
|
||||||
for _, u := range publicPeers {
|
for _, u := range publicPeers {
|
||||||
urlsAsStrings = append(urlsAsStrings, u.String())
|
urlsAsStrings = append(urlsAsStrings, u.String())
|
||||||
|
|
@ -372,23 +285,4 @@ func Yggdrasil(config *viper.Viper, ch chan string) {
|
||||||
n.core.Stop()
|
n.core.Stop()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Helper to detect if socket address is in use
|
|
||||||
// https://stackoverflow.com/a/52152912
|
|
||||||
func isErrorAddressAlreadyInUse(err error) bool {
|
|
||||||
var eOsSyscall *os.SyscallError
|
|
||||||
if !errors.As(err, &eOsSyscall) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
var errErrno syscall.Errno // doesn't need a "*" (ptr) because it's already a ptr (uintptr)
|
|
||||||
if !errors.As(eOsSyscall, &errErrno) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if errors.Is(errErrno, syscall.EADDRINUSE) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
const WSAEADDRINUSE = 10048
|
|
||||||
if runtime.GOOS == "windows" && errErrno == WSAEADDRINUSE {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
package persistentpeersparser
|
package yggdrasil
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
Loading…
Add table
Add a link
Reference in a new issue