prysm-pulse/sharding/simulator/service.go
Raul Jordan f3f5b8e5a6 Simplify Service Registry, Pass in References as Dependencies to Prevent Nil Pointer Errors (#217)
sharding: simplify service registry to prevent nil pointer errors
Former-commit-id: ba4833c385e5212723932491810baad62e3ff0f9 [formerly c550c6d0837999f46a6de55a36fb1ae92d2ecd6f]
Former-commit-id: 80e9e13bc811444b461dad6bdf9eec633b911bec
2018-06-28 20:56:51 -04:00

101 lines
3.6 KiB
Go

package simulator
import (
"context"
"fmt"
"math/big"
"time"
"github.com/ethereum/go-ethereum/event"
"github.com/ethereum/go-ethereum/log"
"github.com/ethereum/go-ethereum/sharding/mainchain"
"github.com/ethereum/go-ethereum/sharding/p2p"
"github.com/ethereum/go-ethereum/sharding/p2p/messages"
"github.com/ethereum/go-ethereum/sharding/params"
"github.com/ethereum/go-ethereum/sharding/syncer"
"github.com/ethereum/go-ethereum/sharding/utils"
)
// Simulator is a service in a shard node that simulates requests from
// remote notes coming over the shardp2p network. For example, if
// we are running a proposer service, we would want to simulate notary requests
// requests coming to us via a p2p feed. This service will be removed
// once p2p internals and end-to-end testing across remote
// nodes have been implemented.
type Simulator struct {
config *params.Config
client *mainchain.SMCClient
p2p *p2p.Server
shardID int
ctx context.Context
cancel context.CancelFunc
errChan chan error // Useful channel for handling errors at the service layer.
delay time.Duration
requestFeed *event.Feed
}
// NewSimulator creates a struct instance of a simulator service.
// It will have access to config, a mainchain client, a p2p server,
// and a shardID.
func NewSimulator(config *params.Config, client *mainchain.SMCClient, p2p *p2p.Server, shardID int, delay time.Duration) (*Simulator, error) {
ctx, cancel := context.WithCancel(context.Background())
errChan := make(chan error)
return &Simulator{config, client, p2p, shardID, ctx, cancel, errChan, delay, nil}, nil
}
// Start the main loop for simulating p2p requests.
func (s *Simulator) Start() {
log.Info("Starting simulator service")
s.requestFeed = s.p2p.Feed(messages.CollationBodyRequest{})
go utils.HandleServiceErrors(s.ctx.Done(), s.errChan)
go s.simulateNotaryRequests(s.client.SMCCaller(), s.client.ChainReader(), time.Tick(time.Second*s.delay))
}
// Stop the main loop for simulator requests.
func (s *Simulator) Stop() error {
// Triggers a cancel call in the service's context which shuts down every goroutine
// in this service.
defer s.cancel()
defer close(s.errChan)
log.Warn("Stopping simulator service")
return nil
}
// simulateNotaryRequests simulates p2p message sent out by notaries
// once the system is in production. Notaries will be performing
// this action within their own service when they are selected on a shard, period
// pair to perform their responsibilities. This function in particular simulates
// requests for collation bodies that will be relayed to the appropriate proposer
// by the p2p feed layer.
func (s *Simulator) simulateNotaryRequests(fetcher mainchain.RecordFetcher, reader mainchain.Reader, delayChan <-chan time.Time) {
for {
select {
// Makes sure to close this goroutine when the service stops.
case <-s.ctx.Done():
log.Debug("Simulator context closed, exiting goroutine")
return
case <-delayChan:
blockNumber, err := reader.BlockByNumber(s.ctx, nil)
if err != nil {
s.errChan <- fmt.Errorf("could not fetch current block number: %v", err)
continue
}
period := new(big.Int).Div(blockNumber.Number(), big.NewInt(s.config.PeriodLength))
req, err := syncer.RequestCollationBody(fetcher, big.NewInt(int64(s.shardID)), period)
if err != nil {
s.errChan <- fmt.Errorf("error constructing collation body request: %v", err)
continue
}
if req != nil {
msg := p2p.Message{
Peer: p2p.Peer{},
Data: *req,
}
s.requestFeed.Send(msg)
log.Info("Sent request for collation body via a shardp2p feed")
}
}
}
}