[Server] Implements Orchestration server (#252)
* [pubsub] - remove pubsub package * [chain] - Add height to chain * [peer] - remove unnecesary println * [server] - Implement server package * Add main.go to run node
This commit is contained in:
parent
cb21c66316
commit
1a6bdd4099
15 changed files with 366 additions and 51 deletions
20
main.go
Normal file
20
main.go
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/server"
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/wire/protocol"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
s, err := server.New(protocol.MainNet, 10332)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
err = s.Run()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("Server has stopped from the following error: ", err.Error())
|
||||||
|
}
|
||||||
|
}
|
|
@ -24,6 +24,7 @@ var (
|
||||||
// Chain represents a blockchain instance
|
// Chain represents a blockchain instance
|
||||||
type Chain struct {
|
type Chain struct {
|
||||||
Db *Chaindb
|
Db *Chaindb
|
||||||
|
height uint32
|
||||||
}
|
}
|
||||||
|
|
||||||
// New returns a new chain instance
|
// New returns a new chain instance
|
||||||
|
@ -128,3 +129,9 @@ func (c *Chain) ProcessHeaders(hdrs []*payload.BlockBase) error {
|
||||||
func (c *Chain) verifyHeaders(hdrs []*payload.BlockBase) error {
|
func (c *Chain) verifyHeaders(hdrs []*payload.BlockBase) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// CurrentHeight returns the index of the block
|
||||||
|
// at the tip of the chain
|
||||||
|
func (c Chain) CurrentHeight() uint32 {
|
||||||
|
return c.height
|
||||||
|
}
|
||||||
|
|
|
@ -176,7 +176,6 @@ func (p *Peer) IsVerackReceived() bool {
|
||||||
//NotifyDisconnect returns once the peer has disconnected
|
//NotifyDisconnect returns once the peer has disconnected
|
||||||
// Blocking
|
// Blocking
|
||||||
func (p *Peer) NotifyDisconnect() {
|
func (p *Peer) NotifyDisconnect() {
|
||||||
fmt.Println("Peer has not disconnected yet")
|
|
||||||
<-p.quitch
|
<-p.quitch
|
||||||
fmt.Println("Peer has just disconnected")
|
fmt.Println("Peer has just disconnected")
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,20 +0,0 @@
|
||||||
package pubsub
|
|
||||||
|
|
||||||
// EventType is an enum
|
|
||||||
// representing the types of messages we can subscribe to
|
|
||||||
type EventType int
|
|
||||||
|
|
||||||
const (
|
|
||||||
// NewBlock is called When blockchain connects a new block, it will emit an NewBlock Event
|
|
||||||
NewBlock EventType = iota
|
|
||||||
// BadBlock is called When blockchain declines a block, it will emit a new block event
|
|
||||||
BadBlock
|
|
||||||
// BadHeader is called When blockchain rejects a Header, it will emit this event
|
|
||||||
BadHeader
|
|
||||||
)
|
|
||||||
|
|
||||||
// Event represents a new Event that a subscriber can listen to
|
|
||||||
type Event struct {
|
|
||||||
Type EventType // E.g. event.NewBlock
|
|
||||||
data []byte // Raw information
|
|
||||||
}
|
|
|
@ -1,21 +0,0 @@
|
||||||
package pubsub
|
|
||||||
|
|
||||||
// Publisher sends events to subscribers
|
|
||||||
type Publisher struct {
|
|
||||||
subs []Subscriber
|
|
||||||
}
|
|
||||||
|
|
||||||
// Send iterates over each subscriber and checks
|
|
||||||
// if they are interested in the Event
|
|
||||||
// By looking at their topics, if they are then
|
|
||||||
// the event is emitted to them
|
|
||||||
func (p *Publisher) Send(e Event) error {
|
|
||||||
for _, sub := range p.subs {
|
|
||||||
for _, topic := range sub.Topics() {
|
|
||||||
if e.Type == topic {
|
|
||||||
sub.Emit(e)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
|
@ -1,7 +0,0 @@
|
||||||
package pubsub
|
|
||||||
|
|
||||||
// Subscriber will listen for Events from publishers
|
|
||||||
type Subscriber interface {
|
|
||||||
Topics() []EventType
|
|
||||||
Emit(Event)
|
|
||||||
}
|
|
7
pkg/server/addrmgr.go
Normal file
7
pkg/server/addrmgr.go
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
package server
|
||||||
|
|
||||||
|
// etAddress will return a viable address to connect to
|
||||||
|
// Currently it is hardcoded to be one neo node until address manager is implemented
|
||||||
|
func (s *Server) getAddress() (string, error) {
|
||||||
|
return "seed1.ngd.network:10333", nil
|
||||||
|
}
|
15
pkg/server/chain.go
Normal file
15
pkg/server/chain.go
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/chain"
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/database"
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/wire/protocol"
|
||||||
|
)
|
||||||
|
|
||||||
|
func setupChain(db database.Database, net protocol.Magic) (*chain.Chain, error) {
|
||||||
|
chain, err := chain.New(db, net)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return chain, nil
|
||||||
|
}
|
62
pkg/server/connmgr.go
Normal file
62
pkg/server/connmgr.go
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/hex"
|
||||||
|
"fmt"
|
||||||
|
"net"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/connmgr"
|
||||||
|
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/peer"
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/wire/util"
|
||||||
|
iputils "github.com/CityOfZion/neo-go/pkg/wire/util/ip"
|
||||||
|
)
|
||||||
|
|
||||||
|
func setupConnManager(s *Server, port uint16) *connmgr.Connmgr {
|
||||||
|
cfg := connmgr.Config{
|
||||||
|
GetAddress: s.getAddress,
|
||||||
|
OnAccept: s.onAccept,
|
||||||
|
OnConnection: s.onConnection,
|
||||||
|
AddressPort: iputils.GetLocalIP().String() + ":" + strconv.FormatUint(uint64(port), 10),
|
||||||
|
}
|
||||||
|
return connmgr.New(cfg)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) onConnection(conn net.Conn, addr string) {
|
||||||
|
fmt.Println("We have connected successfully to: ", addr)
|
||||||
|
|
||||||
|
p := peer.NewPeer(conn, false, *s.peerCfg)
|
||||||
|
err := p.Run()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("Error running peer" + err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
s.pmg.AddPeer(p)
|
||||||
|
|
||||||
|
byt, err := hex.DecodeString("d42561e3d30e15be6400b6df2f328e02d2bf6354c41dce433bc57687c82144bf")
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("Error getting hash " + err.Error())
|
||||||
|
}
|
||||||
|
lh, err := util.Uint256DecodeBytes(byt)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("Error getting hash " + err.Error())
|
||||||
|
}
|
||||||
|
err = p.RequestHeaders(lh.Reverse())
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) onAccept(conn net.Conn) {
|
||||||
|
fmt.Println("A peer with address: ", conn.RemoteAddr().String(), "has connect to us")
|
||||||
|
|
||||||
|
p := peer.NewPeer(conn, true, *s.peerCfg)
|
||||||
|
err := p.Run()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("Error running peer" + err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
s.pmg.AddPeer(p)
|
||||||
|
}
|
14
pkg/server/database.go
Normal file
14
pkg/server/database.go
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/database"
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/wire/protocol"
|
||||||
|
)
|
||||||
|
|
||||||
|
func setupDatabase(net protocol.Magic) (database.Database, error) {
|
||||||
|
db, err := database.New(net.String())
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return db, nil
|
||||||
|
}
|
23
pkg/server/peerconfig.go
Normal file
23
pkg/server/peerconfig.go
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"math/rand"
|
||||||
|
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/peer"
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/wire/protocol"
|
||||||
|
)
|
||||||
|
|
||||||
|
func setupPeerConfig(s *Server, port uint16, net protocol.Magic) *peer.LocalConfig {
|
||||||
|
return &peer.LocalConfig{
|
||||||
|
Net: net,
|
||||||
|
UserAgent: "NEO-GO",
|
||||||
|
Services: protocol.NodePeerService,
|
||||||
|
Nonce: rand.Uint32(),
|
||||||
|
ProtocolVer: 0,
|
||||||
|
Relay: false,
|
||||||
|
Port: port,
|
||||||
|
StartHeight: s.chain.CurrentHeight,
|
||||||
|
OnHeader: s.onHeader,
|
||||||
|
OnBlock: s.onBlock,
|
||||||
|
}
|
||||||
|
}
|
9
pkg/server/peermgr.go
Normal file
9
pkg/server/peermgr.go
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/peermgr"
|
||||||
|
)
|
||||||
|
|
||||||
|
func setupPeerManager() *peermgr.PeerMgr {
|
||||||
|
return peermgr.New()
|
||||||
|
}
|
113
pkg/server/server.go
Normal file
113
pkg/server/server.go
Normal file
|
@ -0,0 +1,113 @@
|
||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/peermgr"
|
||||||
|
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/chain"
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/connmgr"
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/peer"
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/syncmgr"
|
||||||
|
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/database"
|
||||||
|
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/wire/protocol"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Server orchestrates all of the modules
|
||||||
|
type Server struct {
|
||||||
|
net protocol.Magic
|
||||||
|
stopCh chan error
|
||||||
|
|
||||||
|
// Modules
|
||||||
|
db database.Database
|
||||||
|
smg *syncmgr.Syncmgr
|
||||||
|
cmg *connmgr.Connmgr
|
||||||
|
pmg *peermgr.PeerMgr
|
||||||
|
chain *chain.Chain
|
||||||
|
|
||||||
|
peerCfg *peer.LocalConfig
|
||||||
|
}
|
||||||
|
|
||||||
|
//New creates a new server object for a particular network and sets up each module
|
||||||
|
func New(net protocol.Magic, port uint16) (*Server, error) {
|
||||||
|
s := &Server{
|
||||||
|
net: net,
|
||||||
|
stopCh: make(chan error, 0),
|
||||||
|
}
|
||||||
|
|
||||||
|
// Setup database
|
||||||
|
db, err := setupDatabase(net)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
s.db = db
|
||||||
|
|
||||||
|
// setup peermgr
|
||||||
|
peermgr := setupPeerManager()
|
||||||
|
s.pmg = peermgr
|
||||||
|
|
||||||
|
// Setup chain
|
||||||
|
chain, err := setupChain(db, net)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
s.chain = chain
|
||||||
|
|
||||||
|
// Setup sync manager
|
||||||
|
syncmgr := setupSyncManager(s)
|
||||||
|
s.smg = syncmgr
|
||||||
|
|
||||||
|
// Setup connection manager
|
||||||
|
connmgr := setupConnManager(s, port)
|
||||||
|
s.cmg = connmgr
|
||||||
|
|
||||||
|
// Setup peer config
|
||||||
|
peerCfg := setupPeerConfig(s, port, net)
|
||||||
|
s.peerCfg = peerCfg
|
||||||
|
|
||||||
|
return s, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Run starts the daemon by connecting to previously nodes or connectng to seed nodes.
|
||||||
|
// This should be called once all modules have been setup
|
||||||
|
func (s *Server) Run() error {
|
||||||
|
fmt.Println("Server is starting up")
|
||||||
|
|
||||||
|
// start the connmgr
|
||||||
|
err := s.cmg.Run()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Attempt to connect to a peer
|
||||||
|
err = s.cmg.NewRequest()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Request header to start synchronisation
|
||||||
|
bestHeader, err := s.chain.Db.GetLastHeader()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
err = s.pmg.RequestHeaders(bestHeader.Hash.Reverse())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
fmt.Println("Server Successfully started")
|
||||||
|
return s.wait()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) wait() error {
|
||||||
|
err := <-s.stopCh
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Stop stops the server
|
||||||
|
func (s *Server) Stop(err error) error {
|
||||||
|
fmt.Println("Server is shutting down")
|
||||||
|
s.stopCh <- err
|
||||||
|
return nil
|
||||||
|
}
|
94
pkg/server/syncmgr.go
Normal file
94
pkg/server/syncmgr.go
Normal file
|
@ -0,0 +1,94 @@
|
||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/binary"
|
||||||
|
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/peer"
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/syncmgr"
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/wire/payload"
|
||||||
|
"github.com/CityOfZion/neo-go/pkg/wire/util"
|
||||||
|
)
|
||||||
|
|
||||||
|
func setupSyncManager(s *Server) *syncmgr.Syncmgr {
|
||||||
|
|
||||||
|
cfg := &syncmgr.Config{
|
||||||
|
ProcessBlock: s.processBlock,
|
||||||
|
ProcessHeaders: s.processHeaders,
|
||||||
|
|
||||||
|
RequestBlock: s.requestBlock,
|
||||||
|
RequestHeaders: s.requestHeaders,
|
||||||
|
|
||||||
|
GetNextBlockHash: s.getNextBlockHash,
|
||||||
|
AskForNewBlocks: s.askForNewBlocks,
|
||||||
|
|
||||||
|
FetchHeadersAgain: s.fetchHeadersAgain,
|
||||||
|
FetchBlockAgain: s.fetchBlockAgain,
|
||||||
|
}
|
||||||
|
|
||||||
|
return syncmgr.New(cfg)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) onHeader(peer *peer.Peer, hdrsMessage *payload.HeadersMessage) {
|
||||||
|
s.pmg.MsgReceived(peer, hdrsMessage.Command())
|
||||||
|
s.smg.OnHeader(peer, hdrsMessage)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) onBlock(peer *peer.Peer, blockMsg *payload.BlockMessage) {
|
||||||
|
s.pmg.MsgReceived(peer, blockMsg.Command())
|
||||||
|
s.smg.OnBlock(peer, blockMsg)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) processBlock(block payload.Block) error {
|
||||||
|
return s.chain.ProcessBlock(block)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) processHeaders(hdrs []*payload.BlockBase) error {
|
||||||
|
return s.chain.ProcessHeaders(hdrs)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) requestHeaders(hash util.Uint256) error {
|
||||||
|
return s.pmg.RequestHeaders(hash)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) requestBlock(hash util.Uint256) error {
|
||||||
|
return s.pmg.RequestBlock(hash)
|
||||||
|
}
|
||||||
|
|
||||||
|
// getNextBlockHash searches the database for the blockHash
|
||||||
|
// that is the height above our best block. The hash will be taken from a header.
|
||||||
|
func (s *Server) getNextBlockHash() (util.Uint256, error) {
|
||||||
|
bestBlock, err := s.chain.Db.GetLastBlock()
|
||||||
|
if err != nil {
|
||||||
|
// Panic!
|
||||||
|
// XXX: One alternative, is to get the network, erase the database and then start again from scratch.
|
||||||
|
// This should never happen. The latest block will always be atleast the genesis block
|
||||||
|
panic("could not get best block from database" + err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
index := make([]byte, 4)
|
||||||
|
binary.BigEndian.PutUint32(index, bestBlock.Index+1)
|
||||||
|
|
||||||
|
hdr, err := s.chain.Db.GetHeaderFromHeight(index)
|
||||||
|
if err != nil {
|
||||||
|
return util.Uint256{}, err
|
||||||
|
}
|
||||||
|
return hdr.Hash, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) getBestBlockHash() (util.Uint256, error) {
|
||||||
|
return util.Uint256{}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) askForNewBlocks() {
|
||||||
|
// send a getblocks message with the latest block saved
|
||||||
|
|
||||||
|
// when we receive something then send get data
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) fetchHeadersAgain(util.Uint256) error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Server) fetchBlockAgain(util.Uint256) error {
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -9,7 +9,7 @@ import (
|
||||||
type Config struct {
|
type Config struct {
|
||||||
|
|
||||||
// Chain functions
|
// Chain functions
|
||||||
ProcessBlock func(msg payload.Block) error
|
ProcessBlock func(block payload.Block) error
|
||||||
ProcessHeaders func(hdrs []*payload.BlockBase) error
|
ProcessHeaders func(hdrs []*payload.BlockBase) error
|
||||||
|
|
||||||
// RequestHeaders will send a getHeaders request
|
// RequestHeaders will send a getHeaders request
|
||||||
|
|
Loading…
Reference in a new issue