Merge pull request #3624 from filecoin-project/blocksync-refactor
blocksync: introduce interfaces; rename to ChainExchange (abbrev. chainxchg)
This commit is contained in:
commit
42b9d42a8f
@ -1,6 +1,6 @@
|
|||||||
// Code generated by github.com/whyrusleeping/cbor-gen. DO NOT EDIT.
|
// Code generated by github.com/whyrusleeping/cbor-gen. DO NOT EDIT.
|
||||||
|
|
||||||
package blocksync
|
package exchange
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
@ -1,4 +1,4 @@
|
|||||||
package blocksync
|
package exchange
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
@ -7,14 +7,16 @@ import (
|
|||||||
"math/rand"
|
"math/rand"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
host "github.com/libp2p/go-libp2p-core/host"
|
"github.com/libp2p/go-libp2p-core/host"
|
||||||
inet "github.com/libp2p/go-libp2p-core/network"
|
"github.com/libp2p/go-libp2p-core/network"
|
||||||
"github.com/libp2p/go-libp2p-core/peer"
|
"github.com/libp2p/go-libp2p-core/peer"
|
||||||
|
|
||||||
"go.opencensus.io/trace"
|
"go.opencensus.io/trace"
|
||||||
"go.uber.org/fx"
|
"go.uber.org/fx"
|
||||||
"golang.org/x/xerrors"
|
"golang.org/x/xerrors"
|
||||||
|
|
||||||
cborutil "github.com/filecoin-project/go-cbor-util"
|
cborutil "github.com/filecoin-project/go-cbor-util"
|
||||||
|
|
||||||
"github.com/filecoin-project/lotus/build"
|
"github.com/filecoin-project/lotus/build"
|
||||||
"github.com/filecoin-project/lotus/chain/store"
|
"github.com/filecoin-project/lotus/chain/store"
|
||||||
"github.com/filecoin-project/lotus/chain/types"
|
"github.com/filecoin-project/lotus/chain/types"
|
||||||
@ -22,11 +24,9 @@ import (
|
|||||||
"github.com/filecoin-project/lotus/lib/peermgr"
|
"github.com/filecoin-project/lotus/lib/peermgr"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Protocol client.
|
// client implements exchange.Client, using the libp2p ChainExchange protocol
|
||||||
// FIXME: Rename to just `Client`. Not done at the moment to avoid
|
// as the fetching mechanism.
|
||||||
// disrupting too much of the consumer code, should be done along
|
type client struct {
|
||||||
// https://github.com/filecoin-project/lotus/issues/2612.
|
|
||||||
type BlockSync struct {
|
|
||||||
// Connection manager used to contact the server.
|
// Connection manager used to contact the server.
|
||||||
// FIXME: We should have a reduced interface here, initialized
|
// FIXME: We should have a reduced interface here, initialized
|
||||||
// just with our protocol ID, we shouldn't be able to open *any*
|
// just with our protocol ID, we shouldn't be able to open *any*
|
||||||
@ -36,12 +36,12 @@ type BlockSync struct {
|
|||||||
peerTracker *bsPeerTracker
|
peerTracker *bsPeerTracker
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewClient(
|
var _ Client = (*client)(nil)
|
||||||
lc fx.Lifecycle,
|
|
||||||
host host.Host,
|
// NewClient creates a new libp2p-based exchange.Client that uses the libp2p
|
||||||
pmgr peermgr.MaybePeerMgr,
|
// ChainExhange protocol as the fetching mechanism.
|
||||||
) *BlockSync {
|
func NewClient(lc fx.Lifecycle, host host.Host, pmgr peermgr.MaybePeerMgr) Client {
|
||||||
return &BlockSync{
|
return &client{
|
||||||
host: host,
|
host: host,
|
||||||
peerTracker: newPeerTracker(lc, host, pmgr.Mgr),
|
peerTracker: newPeerTracker(lc, host, pmgr.Mgr),
|
||||||
}
|
}
|
||||||
@ -64,11 +64,7 @@ func NewClient(
|
|||||||
// request options without disrupting external calls. In the future the
|
// request options without disrupting external calls. In the future the
|
||||||
// consumers should be forced to use a more standardized service and
|
// consumers should be forced to use a more standardized service and
|
||||||
// adhere to a single API derived from this function.
|
// adhere to a single API derived from this function.
|
||||||
func (client *BlockSync) doRequest(
|
func (c *client) doRequest(ctx context.Context, req *Request, singlePeer *peer.ID) (*validatedResponse, error) {
|
||||||
ctx context.Context,
|
|
||||||
req *Request,
|
|
||||||
singlePeer *peer.ID,
|
|
||||||
) (*validatedResponse, error) {
|
|
||||||
// Validate request.
|
// Validate request.
|
||||||
if req.Length == 0 {
|
if req.Length == 0 {
|
||||||
return nil, xerrors.Errorf("invalid request of length 0")
|
return nil, xerrors.Errorf("invalid request of length 0")
|
||||||
@ -88,7 +84,7 @@ func (client *BlockSync) doRequest(
|
|||||||
if singlePeer != nil {
|
if singlePeer != nil {
|
||||||
peers = []peer.ID{*singlePeer}
|
peers = []peer.ID{*singlePeer}
|
||||||
} else {
|
} else {
|
||||||
peers = client.getShuffledPeers()
|
peers = c.getShuffledPeers()
|
||||||
if len(peers) == 0 {
|
if len(peers) == 0 {
|
||||||
return nil, xerrors.Errorf("no peers available")
|
return nil, xerrors.Errorf("no peers available")
|
||||||
}
|
}
|
||||||
@ -109,9 +105,9 @@ func (client *BlockSync) doRequest(
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Send request, read response.
|
// Send request, read response.
|
||||||
res, err := client.sendRequestToPeer(ctx, peer, req)
|
res, err := c.sendRequestToPeer(ctx, peer, req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if !xerrors.Is(err, inet.ErrNoConn) {
|
if !xerrors.Is(err, network.ErrNoConn) {
|
||||||
log.Warnf("could not connect to peer %s: %s",
|
log.Warnf("could not connect to peer %s: %s",
|
||||||
peer.String(), err)
|
peer.String(), err)
|
||||||
}
|
}
|
||||||
@ -119,15 +115,15 @@ func (client *BlockSync) doRequest(
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Process and validate response.
|
// Process and validate response.
|
||||||
validRes, err := client.processResponse(req, res)
|
validRes, err := c.processResponse(req, res)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warnf("processing peer %s response failed: %s",
|
log.Warnf("processing peer %s response failed: %s",
|
||||||
peer.String(), err)
|
peer.String(), err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
client.peerTracker.logGlobalSuccess(build.Clock.Since(globalTime))
|
c.peerTracker.logGlobalSuccess(build.Clock.Since(globalTime))
|
||||||
client.host.ConnManager().TagPeer(peer, "bsync", SUCCESS_PEER_TAG_VALUE)
|
c.host.ConnManager().TagPeer(peer, "bsync", SuccessPeerTagValue)
|
||||||
return validRes, nil
|
return validRes, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -146,11 +142,8 @@ func (client *BlockSync) doRequest(
|
|||||||
// We are conflating in the single error returned both status and validation
|
// We are conflating in the single error returned both status and validation
|
||||||
// errors. Peer penalization should happen here then, before returning, so
|
// errors. Peer penalization should happen here then, before returning, so
|
||||||
// we can apply the correct penalties depending on the cause of the error.
|
// we can apply the correct penalties depending on the cause of the error.
|
||||||
func (client *BlockSync) processResponse(
|
// FIXME: Add the `peer` as argument once we implement penalties.
|
||||||
req *Request,
|
func (c *client) processResponse(req *Request, res *Response) (*validatedResponse, error) {
|
||||||
res *Response,
|
|
||||||
// FIXME: Add the `peer` as argument once we implement penalties.
|
|
||||||
) (*validatedResponse, error) {
|
|
||||||
err := res.statusToError()
|
err := res.statusToError()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, xerrors.Errorf("status error: %s", err)
|
return nil, xerrors.Errorf("status error: %s", err)
|
||||||
@ -248,16 +241,8 @@ func (client *BlockSync) processResponse(
|
|||||||
return validRes, nil
|
return validRes, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetBlocks fetches count blocks from the network, from the provided tipset
|
// GetBlocks implements Client.GetBlocks(). Refer to the godocs there.
|
||||||
// *backwards*, returning as many tipsets as count.
|
func (c *client) GetBlocks(ctx context.Context, tsk types.TipSetKey, count int) ([]*types.TipSet, error) {
|
||||||
//
|
|
||||||
// {hint/usage}: This is used by the Syncer during normal chain syncing and when
|
|
||||||
// resolving forks.
|
|
||||||
func (client *BlockSync) GetBlocks(
|
|
||||||
ctx context.Context,
|
|
||||||
tsk types.TipSetKey,
|
|
||||||
count int,
|
|
||||||
) ([]*types.TipSet, error) {
|
|
||||||
ctx, span := trace.StartSpan(ctx, "bsync.GetBlocks")
|
ctx, span := trace.StartSpan(ctx, "bsync.GetBlocks")
|
||||||
defer span.End()
|
defer span.End()
|
||||||
if span.IsRecordingEvents() {
|
if span.IsRecordingEvents() {
|
||||||
@ -273,7 +258,7 @@ func (client *BlockSync) GetBlocks(
|
|||||||
Options: Headers,
|
Options: Headers,
|
||||||
}
|
}
|
||||||
|
|
||||||
validRes, err := client.doRequest(ctx, req, nil)
|
validRes, err := c.doRequest(ctx, req, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -281,11 +266,8 @@ func (client *BlockSync) GetBlocks(
|
|||||||
return validRes.tipsets, nil
|
return validRes.tipsets, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (client *BlockSync) GetFullTipSet(
|
// GetFullTipSet implements Client.GetFullTipSet(). Refer to the godocs there.
|
||||||
ctx context.Context,
|
func (c *client) GetFullTipSet(ctx context.Context, peer peer.ID, tsk types.TipSetKey) (*store.FullTipSet, error) {
|
||||||
peer peer.ID,
|
|
||||||
tsk types.TipSetKey,
|
|
||||||
) (*store.FullTipSet, error) {
|
|
||||||
// TODO: round robin through these peers on error
|
// TODO: round robin through these peers on error
|
||||||
|
|
||||||
req := &Request{
|
req := &Request{
|
||||||
@ -294,7 +276,7 @@ func (client *BlockSync) GetFullTipSet(
|
|||||||
Options: Headers | Messages,
|
Options: Headers | Messages,
|
||||||
}
|
}
|
||||||
|
|
||||||
validRes, err := client.doRequest(ctx, req, &peer)
|
validRes, err := c.doRequest(ctx, req, &peer)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -304,11 +286,8 @@ func (client *BlockSync) GetFullTipSet(
|
|||||||
// *one* tipset here, so it's safe to index directly.
|
// *one* tipset here, so it's safe to index directly.
|
||||||
}
|
}
|
||||||
|
|
||||||
func (client *BlockSync) GetChainMessages(
|
// GetChainMessages implements Client.GetChainMessages(). Refer to the godocs there.
|
||||||
ctx context.Context,
|
func (c *client) GetChainMessages(ctx context.Context, head *types.TipSet, length uint64) ([]*CompactedMessages, error) {
|
||||||
head *types.TipSet,
|
|
||||||
length uint64,
|
|
||||||
) ([]*CompactedMessages, error) {
|
|
||||||
ctx, span := trace.StartSpan(ctx, "GetChainMessages")
|
ctx, span := trace.StartSpan(ctx, "GetChainMessages")
|
||||||
if span.IsRecordingEvents() {
|
if span.IsRecordingEvents() {
|
||||||
span.AddAttributes(
|
span.AddAttributes(
|
||||||
@ -324,7 +303,7 @@ func (client *BlockSync) GetChainMessages(
|
|||||||
Options: Messages,
|
Options: Messages,
|
||||||
}
|
}
|
||||||
|
|
||||||
validRes, err := client.doRequest(ctx, req, nil)
|
validRes, err := c.doRequest(ctx, req, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -335,11 +314,7 @@ func (client *BlockSync) GetChainMessages(
|
|||||||
// Send a request to a peer. Write request in the stream and read the
|
// Send a request to a peer. Write request in the stream and read the
|
||||||
// response back. We do not do any processing of the request/response
|
// response back. We do not do any processing of the request/response
|
||||||
// here.
|
// here.
|
||||||
func (client *BlockSync) sendRequestToPeer(
|
func (c *client) sendRequestToPeer(ctx context.Context, peer peer.ID, req *Request) (_ *Response, err error) {
|
||||||
ctx context.Context,
|
|
||||||
peer peer.ID,
|
|
||||||
req *Request,
|
|
||||||
) (_ *Response, err error) {
|
|
||||||
// Trace code.
|
// Trace code.
|
||||||
ctx, span := trace.StartSpan(ctx, "sendRequestToPeer")
|
ctx, span := trace.StartSpan(ctx, "sendRequestToPeer")
|
||||||
defer span.End()
|
defer span.End()
|
||||||
@ -360,35 +335,33 @@ func (client *BlockSync) sendRequestToPeer(
|
|||||||
}()
|
}()
|
||||||
// -- TRACE --
|
// -- TRACE --
|
||||||
|
|
||||||
supported, err := client.host.Peerstore().SupportsProtocols(peer, BlockSyncProtocolID)
|
supported, err := c.host.Peerstore().SupportsProtocols(peer, BlockSyncProtocolID, ChainExchangeProtocolID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
client.RemovePeer(peer)
|
c.RemovePeer(peer)
|
||||||
return nil, xerrors.Errorf("failed to get protocols for peer: %w", err)
|
return nil, xerrors.Errorf("failed to get protocols for peer: %w", err)
|
||||||
}
|
}
|
||||||
if len(supported) == 0 || supported[0] != BlockSyncProtocolID {
|
if len(supported) == 0 || (supported[0] != BlockSyncProtocolID && supported[0] != ChainExchangeProtocolID) {
|
||||||
return nil, xerrors.Errorf("peer %s does not support protocol %s",
|
return nil, xerrors.Errorf("peer %s does not support protocols %s",
|
||||||
peer, BlockSyncProtocolID)
|
peer, []string{BlockSyncProtocolID, ChainExchangeProtocolID})
|
||||||
// FIXME: `ProtoBook` should support a *single* protocol check that returns
|
|
||||||
// a bool instead of a list.
|
|
||||||
}
|
}
|
||||||
|
|
||||||
connectionStart := build.Clock.Now()
|
connectionStart := build.Clock.Now()
|
||||||
|
|
||||||
// Open stream to peer.
|
// Open stream to peer.
|
||||||
stream, err := client.host.NewStream(
|
stream, err := c.host.NewStream(
|
||||||
inet.WithNoDial(ctx, "should already have connection"),
|
network.WithNoDial(ctx, "should already have connection"),
|
||||||
peer,
|
peer,
|
||||||
BlockSyncProtocolID)
|
ChainExchangeProtocolID, BlockSyncProtocolID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
client.RemovePeer(peer)
|
c.RemovePeer(peer)
|
||||||
return nil, xerrors.Errorf("failed to open stream to peer: %w", err)
|
return nil, xerrors.Errorf("failed to open stream to peer: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Write request.
|
// Write request.
|
||||||
_ = stream.SetWriteDeadline(time.Now().Add(WRITE_REQ_DEADLINE))
|
_ = stream.SetWriteDeadline(time.Now().Add(WriteReqDeadline))
|
||||||
if err := cborutil.WriteCborRPC(stream, req); err != nil {
|
if err := cborutil.WriteCborRPC(stream, req); err != nil {
|
||||||
_ = stream.SetWriteDeadline(time.Time{})
|
_ = stream.SetWriteDeadline(time.Time{})
|
||||||
client.peerTracker.logFailure(peer, build.Clock.Since(connectionStart), req.Length)
|
c.peerTracker.logFailure(peer, build.Clock.Since(connectionStart), req.Length)
|
||||||
// FIXME: Should we also remove peer here?
|
// FIXME: Should we also remove peer here?
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -398,11 +371,11 @@ func (client *BlockSync) sendRequestToPeer(
|
|||||||
// Read response.
|
// Read response.
|
||||||
var res Response
|
var res Response
|
||||||
err = cborutil.ReadCborRPC(
|
err = cborutil.ReadCborRPC(
|
||||||
bufio.NewReader(incrt.New(stream, READ_RES_MIN_SPEED, READ_RES_DEADLINE)),
|
bufio.NewReader(incrt.New(stream, ReadResMinSpeed, ReadResDeadline)),
|
||||||
&res)
|
&res)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
client.peerTracker.logFailure(peer, build.Clock.Since(connectionStart), req.Length)
|
c.peerTracker.logFailure(peer, build.Clock.Since(connectionStart), req.Length)
|
||||||
return nil, xerrors.Errorf("failed to read blocksync response: %w", err)
|
return nil, xerrors.Errorf("failed to read chainxchg response: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: Move all this together at the top using a defer as done elsewhere.
|
// FIXME: Move all this together at the top using a defer as done elsewhere.
|
||||||
@ -415,32 +388,34 @@ func (client *BlockSync) sendRequestToPeer(
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
client.peerTracker.logSuccess(peer, build.Clock.Since(connectionStart), uint64(len(res.Chain)))
|
c.peerTracker.logSuccess(peer, build.Clock.Since(connectionStart), uint64(len(res.Chain)))
|
||||||
// FIXME: We should really log a success only after we validate the response.
|
// FIXME: We should really log a success only after we validate the response.
|
||||||
// It might be a bit hard to do.
|
// It might be a bit hard to do.
|
||||||
return &res, nil
|
return &res, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (client *BlockSync) AddPeer(p peer.ID) {
|
// AddPeer implements Client.AddPeer(). Refer to the godocs there.
|
||||||
client.peerTracker.addPeer(p)
|
func (c *client) AddPeer(p peer.ID) {
|
||||||
|
c.peerTracker.addPeer(p)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (client *BlockSync) RemovePeer(p peer.ID) {
|
// RemovePeer implements Client.RemovePeer(). Refer to the godocs there.
|
||||||
client.peerTracker.removePeer(p)
|
func (c *client) RemovePeer(p peer.ID) {
|
||||||
|
c.peerTracker.removePeer(p)
|
||||||
}
|
}
|
||||||
|
|
||||||
// getShuffledPeers returns a preference-sorted set of peers (by latency
|
// getShuffledPeers returns a preference-sorted set of peers (by latency
|
||||||
// and failure counting), shuffling the first few peers so we don't always
|
// and failure counting), shuffling the first few peers so we don't always
|
||||||
// pick the same peer.
|
// pick the same peer.
|
||||||
// FIXME: Consider merging with `shufflePrefix()s`.
|
// FIXME: Consider merging with `shufflePrefix()s`.
|
||||||
func (client *BlockSync) getShuffledPeers() []peer.ID {
|
func (c *client) getShuffledPeers() []peer.ID {
|
||||||
peers := client.peerTracker.prefSortedPeers()
|
peers := c.peerTracker.prefSortedPeers()
|
||||||
shufflePrefix(peers)
|
shufflePrefix(peers)
|
||||||
return peers
|
return peers
|
||||||
}
|
}
|
||||||
|
|
||||||
func shufflePrefix(peers []peer.ID) {
|
func shufflePrefix(peers []peer.ID) {
|
||||||
prefix := SHUFFLE_PEERS_PREFIX
|
prefix := ShufflePeersPrefix
|
||||||
if len(peers) < prefix {
|
if len(peers) < prefix {
|
||||||
prefix = len(peers)
|
prefix = len(peers)
|
||||||
}
|
}
|
19
chain/exchange/doc.go
Normal file
19
chain/exchange/doc.go
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
// Package exchange contains the ChainExchange server and client components.
|
||||||
|
//
|
||||||
|
// ChainExchange is the basic chain synchronization protocol of Filecoin.
|
||||||
|
// ChainExchange is an RPC-oriented protocol, with a single operation to
|
||||||
|
// request blocks for now.
|
||||||
|
//
|
||||||
|
// A request contains a start anchor block (referred to with a CID), and a
|
||||||
|
// amount of blocks requested beyond the anchor (including the anchor itself).
|
||||||
|
//
|
||||||
|
// A client can also pass options, encoded as a 64-bit bitfield. Lotus supports
|
||||||
|
// two options at the moment:
|
||||||
|
//
|
||||||
|
// - include block contents
|
||||||
|
// - include block messages
|
||||||
|
//
|
||||||
|
// The response will include a status code, an optional message, and the
|
||||||
|
// response payload in case of success. The payload is a slice of serialized
|
||||||
|
// tipsets.
|
||||||
|
package exchange
|
51
chain/exchange/interfaces.go
Normal file
51
chain/exchange/interfaces.go
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
package exchange
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"github.com/libp2p/go-libp2p-core/network"
|
||||||
|
"github.com/libp2p/go-libp2p-core/peer"
|
||||||
|
|
||||||
|
"github.com/filecoin-project/lotus/chain/store"
|
||||||
|
"github.com/filecoin-project/lotus/chain/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Server is the responder side of the ChainExchange protocol. It accepts
|
||||||
|
// requests from clients and services them by returning the requested
|
||||||
|
// chain data.
|
||||||
|
type Server interface {
|
||||||
|
// HandleStream is the protocol handler to be registered on a libp2p
|
||||||
|
// protocol router.
|
||||||
|
//
|
||||||
|
// In the current version of the protocol, streams are single-use. The
|
||||||
|
// server will read a single Request, and will respond with a single
|
||||||
|
// Response. It will dispose of the stream straight after.
|
||||||
|
HandleStream(stream network.Stream)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Client is the requesting side of the ChainExchange protocol. It acts as
|
||||||
|
// a proxy for other components to request chain data from peers. It is chiefly
|
||||||
|
// used by the Syncer.
|
||||||
|
type Client interface {
|
||||||
|
// GetBlocks fetches block headers from the network, from the provided
|
||||||
|
// tipset *backwards*, returning as many tipsets as the count parameter,
|
||||||
|
// or less.
|
||||||
|
GetBlocks(ctx context.Context, tsk types.TipSetKey, count int) ([]*types.TipSet, error)
|
||||||
|
|
||||||
|
// GetChainMessages fetches messages from the network, from the provided
|
||||||
|
// tipset *backwards*, returning the messages from as many tipsets as the
|
||||||
|
// count parameter, or less.
|
||||||
|
GetChainMessages(ctx context.Context, head *types.TipSet, length uint64) ([]*CompactedMessages, error)
|
||||||
|
|
||||||
|
// GetFullTipSet fetches a full tipset from a given peer. If successful,
|
||||||
|
// the fetched object contains block headers and all messages in full form.
|
||||||
|
GetFullTipSet(ctx context.Context, peer peer.ID, tsk types.TipSetKey) (*store.FullTipSet, error)
|
||||||
|
|
||||||
|
// AddPeer adds a peer to the pool of peers that the Client requests
|
||||||
|
// data from.
|
||||||
|
AddPeer(peer peer.ID)
|
||||||
|
|
||||||
|
// RemovePeer removes a peer from the pool of peers that the Client
|
||||||
|
// requests data from.
|
||||||
|
RemovePeer(peer peer.ID)
|
||||||
|
}
|
@ -1,4 +1,4 @@
|
|||||||
package blocksync
|
package exchange
|
||||||
|
|
||||||
// FIXME: This needs to be reviewed.
|
// FIXME: This needs to be reviewed.
|
||||||
|
|
@ -1,4 +1,4 @@
|
|||||||
package blocksync
|
package exchange
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"time"
|
"time"
|
||||||
@ -13,9 +13,17 @@ import (
|
|||||||
"github.com/filecoin-project/lotus/chain/types"
|
"github.com/filecoin-project/lotus/chain/types"
|
||||||
)
|
)
|
||||||
|
|
||||||
var log = logging.Logger("blocksync")
|
var log = logging.Logger("chainxchg")
|
||||||
|
|
||||||
const BlockSyncProtocolID = "/fil/sync/blk/0.0.1"
|
const (
|
||||||
|
// BlockSyncProtocolID is the protocol ID of the former blocksync protocol.
|
||||||
|
// Deprecated.
|
||||||
|
BlockSyncProtocolID = "/fil/sync/blk/0.0.1"
|
||||||
|
|
||||||
|
// ChainExchangeProtocolID is the protocol ID of the chain exchange
|
||||||
|
// protocol.
|
||||||
|
ChainExchangeProtocolID = "/fil/chain/xchg/0.0.1"
|
||||||
|
)
|
||||||
|
|
||||||
// FIXME: Bumped from original 800 to this to accommodate `syncFork()`
|
// FIXME: Bumped from original 800 to this to accommodate `syncFork()`
|
||||||
// use of `GetBlocks()`. It seems the expectation of that API is to
|
// use of `GetBlocks()`. It seems the expectation of that API is to
|
||||||
@ -25,14 +33,16 @@ const BlockSyncProtocolID = "/fil/sync/blk/0.0.1"
|
|||||||
// qualifier to avoid "const initializer [...] is not a constant" error.)
|
// qualifier to avoid "const initializer [...] is not a constant" error.)
|
||||||
var MaxRequestLength = uint64(build.ForkLengthThreshold)
|
var MaxRequestLength = uint64(build.ForkLengthThreshold)
|
||||||
|
|
||||||
// Extracted constants from the code.
|
const (
|
||||||
// FIXME: Should be reviewed and confirmed.
|
// Extracted constants from the code.
|
||||||
const SUCCESS_PEER_TAG_VALUE = 25
|
// FIXME: Should be reviewed and confirmed.
|
||||||
const WRITE_REQ_DEADLINE = 5 * time.Second
|
SuccessPeerTagValue = 25
|
||||||
const READ_RES_DEADLINE = WRITE_REQ_DEADLINE
|
WriteReqDeadline = 5 * time.Second
|
||||||
const READ_RES_MIN_SPEED = 50 << 10
|
ReadResDeadline = WriteReqDeadline
|
||||||
const SHUFFLE_PEERS_PREFIX = 5
|
ReadResMinSpeed = 50 << 10
|
||||||
const WRITE_RES_DEADLINE = 60 * time.Second
|
ShufflePeersPrefix = 5
|
||||||
|
WriteResDeadline = 60 * time.Second
|
||||||
|
)
|
||||||
|
|
||||||
// FIXME: Rename. Make private.
|
// FIXME: Rename. Make private.
|
||||||
type Request struct {
|
type Request struct {
|
||||||
@ -117,7 +127,7 @@ func (res *Response) statusToError() error {
|
|||||||
case NotFound:
|
case NotFound:
|
||||||
return xerrors.Errorf("not found")
|
return xerrors.Errorf("not found")
|
||||||
case GoAway:
|
case GoAway:
|
||||||
return xerrors.Errorf("not handling 'go away' blocksync responses yet")
|
return xerrors.Errorf("not handling 'go away' chainxchg responses yet")
|
||||||
case InternalError:
|
case InternalError:
|
||||||
return xerrors.Errorf("block sync peer errored: %s", res.ErrorMessage)
|
return xerrors.Errorf("block sync peer errored: %s", res.ErrorMessage)
|
||||||
case BadRequest:
|
case BadRequest:
|
@ -1,4 +1,4 @@
|
|||||||
package blocksync
|
package exchange
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
@ -18,38 +18,25 @@ import (
|
|||||||
inet "github.com/libp2p/go-libp2p-core/network"
|
inet "github.com/libp2p/go-libp2p-core/network"
|
||||||
)
|
)
|
||||||
|
|
||||||
// BlockSyncService is the component that services BlockSync requests from
|
// server implements exchange.Server. It services requests for the
|
||||||
// peers.
|
// libp2p ChainExchange protocol.
|
||||||
//
|
type server struct {
|
||||||
// BlockSync is the basic chain synchronization protocol of Filecoin. BlockSync
|
|
||||||
// is an RPC-oriented protocol, with a single operation to request blocks.
|
|
||||||
//
|
|
||||||
// A request contains a start anchor block (referred to with a CID), and a
|
|
||||||
// amount of blocks requested beyond the anchor (including the anchor itself).
|
|
||||||
//
|
|
||||||
// A client can also pass options, encoded as a 64-bit bitfield. Lotus supports
|
|
||||||
// two options at the moment:
|
|
||||||
//
|
|
||||||
// - include block contents
|
|
||||||
// - include block messages
|
|
||||||
//
|
|
||||||
// The response will include a status code, an optional message, and the
|
|
||||||
// response payload in case of success. The payload is a slice of serialized
|
|
||||||
// tipsets.
|
|
||||||
// FIXME: Rename to just `Server` (will be done later, see note on `BlockSync`).
|
|
||||||
type BlockSyncService struct {
|
|
||||||
cs *store.ChainStore
|
cs *store.ChainStore
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewBlockSyncService(cs *store.ChainStore) *BlockSyncService {
|
var _ Server = (*server)(nil)
|
||||||
return &BlockSyncService{
|
|
||||||
|
// NewServer creates a new libp2p-based exchange.Server. It services requests
|
||||||
|
// for the libp2p ChainExchange protocol.
|
||||||
|
func NewServer(cs *store.ChainStore) Server {
|
||||||
|
return &server{
|
||||||
cs: cs,
|
cs: cs,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Entry point of the service, handles `Request`s.
|
// HandleStream implements Server.HandleStream. Refer to the godocs there.
|
||||||
func (server *BlockSyncService) HandleStream(stream inet.Stream) {
|
func (s *server) HandleStream(stream inet.Stream) {
|
||||||
ctx, span := trace.StartSpan(context.Background(), "blocksync.HandleStream")
|
ctx, span := trace.StartSpan(context.Background(), "chainxchg.HandleStream")
|
||||||
defer span.End()
|
defer span.End()
|
||||||
|
|
||||||
defer stream.Close() //nolint:errcheck
|
defer stream.Close() //nolint:errcheck
|
||||||
@ -62,13 +49,13 @@ func (server *BlockSyncService) HandleStream(stream inet.Stream) {
|
|||||||
log.Infow("block sync request",
|
log.Infow("block sync request",
|
||||||
"start", req.Head, "len", req.Length)
|
"start", req.Head, "len", req.Length)
|
||||||
|
|
||||||
resp, err := server.processRequest(ctx, &req)
|
resp, err := s.processRequest(ctx, &req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warn("failed to process request: ", err)
|
log.Warn("failed to process request: ", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
_ = stream.SetDeadline(time.Now().Add(WRITE_RES_DEADLINE))
|
_ = stream.SetDeadline(time.Now().Add(WriteResDeadline))
|
||||||
if err := cborutil.WriteCborRPC(stream, resp); err != nil {
|
if err := cborutil.WriteCborRPC(stream, resp); err != nil {
|
||||||
_ = stream.SetDeadline(time.Time{})
|
_ = stream.SetDeadline(time.Time{})
|
||||||
log.Warnw("failed to write back response for handle stream",
|
log.Warnw("failed to write back response for handle stream",
|
||||||
@ -80,10 +67,7 @@ func (server *BlockSyncService) HandleStream(stream inet.Stream) {
|
|||||||
|
|
||||||
// Validate and service the request. We return either a protocol
|
// Validate and service the request. We return either a protocol
|
||||||
// response or an internal error.
|
// response or an internal error.
|
||||||
func (server *BlockSyncService) processRequest(
|
func (s *server) processRequest(ctx context.Context, req *Request) (*Response, error) {
|
||||||
ctx context.Context,
|
|
||||||
req *Request,
|
|
||||||
) (*Response, error) {
|
|
||||||
validReq, errResponse := validateRequest(ctx, req)
|
validReq, errResponse := validateRequest(ctx, req)
|
||||||
if errResponse != nil {
|
if errResponse != nil {
|
||||||
// The request did not pass validation, return the response
|
// The request did not pass validation, return the response
|
||||||
@ -91,17 +75,14 @@ func (server *BlockSyncService) processRequest(
|
|||||||
return errResponse, nil
|
return errResponse, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
return server.serviceRequest(ctx, validReq)
|
return s.serviceRequest(ctx, validReq)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Validate request. We either return a `validatedRequest`, or an error
|
// Validate request. We either return a `validatedRequest`, or an error
|
||||||
// `Response` indicating why we can't process it. We do not return any
|
// `Response` indicating why we can't process it. We do not return any
|
||||||
// internal errors here, we just signal protocol ones.
|
// internal errors here, we just signal protocol ones.
|
||||||
func validateRequest(
|
func validateRequest(ctx context.Context, req *Request) (*validatedRequest, *Response) {
|
||||||
ctx context.Context,
|
_, span := trace.StartSpan(ctx, "chainxchg.ValidateRequest")
|
||||||
req *Request,
|
|
||||||
) (*validatedRequest, *Response) {
|
|
||||||
_, span := trace.StartSpan(ctx, "blocksync.ValidateRequest")
|
|
||||||
defer span.End()
|
defer span.End()
|
||||||
|
|
||||||
validReq := validatedRequest{}
|
validReq := validatedRequest{}
|
||||||
@ -147,14 +128,11 @@ func validateRequest(
|
|||||||
return &validReq, nil
|
return &validReq, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (server *BlockSyncService) serviceRequest(
|
func (s *server) serviceRequest(ctx context.Context, req *validatedRequest) (*Response, error) {
|
||||||
ctx context.Context,
|
_, span := trace.StartSpan(ctx, "chainxchg.ServiceRequest")
|
||||||
req *validatedRequest,
|
|
||||||
) (*Response, error) {
|
|
||||||
_, span := trace.StartSpan(ctx, "blocksync.ServiceRequest")
|
|
||||||
defer span.End()
|
defer span.End()
|
||||||
|
|
||||||
chain, err := collectChainSegment(server.cs, req)
|
chain, err := collectChainSegment(s.cs, req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warn("block sync request: collectChainSegment failed: ", err)
|
log.Warn("block sync request: collectChainSegment failed: ", err)
|
||||||
return &Response{
|
return &Response{
|
||||||
@ -174,10 +152,7 @@ func (server *BlockSyncService) serviceRequest(
|
|||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func collectChainSegment(
|
func collectChainSegment(cs *store.ChainStore, req *validatedRequest) ([]*BSTipSet, error) {
|
||||||
cs *store.ChainStore,
|
|
||||||
req *validatedRequest,
|
|
||||||
) ([]*BSTipSet, error) {
|
|
||||||
var bstips []*BSTipSet
|
var bstips []*BSTipSet
|
||||||
|
|
||||||
cur := req.head
|
cur := req.head
|
@ -38,7 +38,7 @@ import (
|
|||||||
"github.com/filecoin-project/lotus/api"
|
"github.com/filecoin-project/lotus/api"
|
||||||
"github.com/filecoin-project/lotus/build"
|
"github.com/filecoin-project/lotus/build"
|
||||||
"github.com/filecoin-project/lotus/chain/beacon"
|
"github.com/filecoin-project/lotus/chain/beacon"
|
||||||
"github.com/filecoin-project/lotus/chain/blocksync"
|
"github.com/filecoin-project/lotus/chain/exchange"
|
||||||
"github.com/filecoin-project/lotus/chain/gen"
|
"github.com/filecoin-project/lotus/chain/gen"
|
||||||
"github.com/filecoin-project/lotus/chain/state"
|
"github.com/filecoin-project/lotus/chain/state"
|
||||||
"github.com/filecoin-project/lotus/chain/stmgr"
|
"github.com/filecoin-project/lotus/chain/stmgr"
|
||||||
@ -52,7 +52,7 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
// Blocks that are more than MaxHeightDrift epochs above
|
// Blocks that are more than MaxHeightDrift epochs above
|
||||||
//the theoretical max height based on systime are quickly rejected
|
// the theoretical max height based on systime are quickly rejected
|
||||||
const MaxHeightDrift = 5
|
const MaxHeightDrift = 5
|
||||||
|
|
||||||
var defaultMessageFetchWindowSize = 200
|
var defaultMessageFetchWindowSize = 200
|
||||||
@ -89,7 +89,7 @@ var LocalIncoming = "incoming"
|
|||||||
// The Syncer does not run workers itself. It's mainly concerned with
|
// The Syncer does not run workers itself. It's mainly concerned with
|
||||||
// ensuring a consistent state of chain consensus. The reactive and network-
|
// ensuring a consistent state of chain consensus. The reactive and network-
|
||||||
// interfacing processes are part of other components, such as the SyncManager
|
// interfacing processes are part of other components, such as the SyncManager
|
||||||
// (which owns the sync scheduler and sync workers), BlockSync, the HELLO
|
// (which owns the sync scheduler and sync workers), ChainExchange, the HELLO
|
||||||
// protocol, and the gossipsub block propagation layer.
|
// protocol, and the gossipsub block propagation layer.
|
||||||
//
|
//
|
||||||
// {hint/concept} The fork-choice rule as it currently stands is: "pick the
|
// {hint/concept} The fork-choice rule as it currently stands is: "pick the
|
||||||
@ -112,7 +112,7 @@ type Syncer struct {
|
|||||||
bad *BadBlockCache
|
bad *BadBlockCache
|
||||||
|
|
||||||
// handle to the block sync service
|
// handle to the block sync service
|
||||||
Bsync *blocksync.BlockSync
|
Exchange exchange.Client
|
||||||
|
|
||||||
self peer.ID
|
self peer.ID
|
||||||
|
|
||||||
@ -132,7 +132,7 @@ type Syncer struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewSyncer creates a new Syncer object.
|
// NewSyncer creates a new Syncer object.
|
||||||
func NewSyncer(sm *stmgr.StateManager, bsync *blocksync.BlockSync, connmgr connmgr.ConnManager, self peer.ID, beacon beacon.RandomBeacon, verifier ffiwrapper.Verifier) (*Syncer, error) {
|
func NewSyncer(sm *stmgr.StateManager, exchange exchange.Client, connmgr connmgr.ConnManager, self peer.ID, beacon beacon.RandomBeacon, verifier ffiwrapper.Verifier) (*Syncer, error) {
|
||||||
gen, err := sm.ChainStore().GetGenesis()
|
gen, err := sm.ChainStore().GetGenesis()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, xerrors.Errorf("getting genesis block: %w", err)
|
return nil, xerrors.Errorf("getting genesis block: %w", err)
|
||||||
@ -147,7 +147,7 @@ func NewSyncer(sm *stmgr.StateManager, bsync *blocksync.BlockSync, connmgr connm
|
|||||||
beacon: beacon,
|
beacon: beacon,
|
||||||
bad: NewBadBlockCache(),
|
bad: NewBadBlockCache(),
|
||||||
Genesis: gent,
|
Genesis: gent,
|
||||||
Bsync: bsync,
|
Exchange: exchange,
|
||||||
store: sm.ChainStore(),
|
store: sm.ChainStore(),
|
||||||
sm: sm,
|
sm: sm,
|
||||||
self: self,
|
self: self,
|
||||||
@ -248,7 +248,7 @@ func (syncer *Syncer) InformNewHead(from peer.ID, fts *store.FullTipSet) bool {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
syncer.Bsync.AddPeer(from)
|
syncer.Exchange.AddPeer(from)
|
||||||
|
|
||||||
bestPweight := syncer.store.GetHeaviestTipSet().ParentWeight()
|
bestPweight := syncer.store.GetHeaviestTipSet().ParentWeight()
|
||||||
targetWeight := fts.TipSet().ParentWeight()
|
targetWeight := fts.TipSet().ParentWeight()
|
||||||
@ -479,7 +479,7 @@ func computeMsgMeta(bs cbor.IpldStore, bmsgCids, smsgCids []cid.Cid) (cid.Cid, e
|
|||||||
}
|
}
|
||||||
|
|
||||||
// FetchTipSet tries to load the provided tipset from the store, and falls back
|
// FetchTipSet tries to load the provided tipset from the store, and falls back
|
||||||
// to the network (BlockSync) by querying the supplied peer if not found
|
// to the network (client) by querying the supplied peer if not found
|
||||||
// locally.
|
// locally.
|
||||||
//
|
//
|
||||||
// {hint/usage} This is used from the HELLO protocol, to fetch the greeting
|
// {hint/usage} This is used from the HELLO protocol, to fetch the greeting
|
||||||
@ -490,7 +490,7 @@ func (syncer *Syncer) FetchTipSet(ctx context.Context, p peer.ID, tsk types.TipS
|
|||||||
}
|
}
|
||||||
|
|
||||||
// fall back to the network.
|
// fall back to the network.
|
||||||
return syncer.Bsync.GetFullTipSet(ctx, p, tsk)
|
return syncer.Exchange.GetFullTipSet(ctx, p, tsk)
|
||||||
}
|
}
|
||||||
|
|
||||||
// tryLoadFullTipSet queries the tipset in the ChainStore, and returns a full
|
// tryLoadFullTipSet queries the tipset in the ChainStore, and returns a full
|
||||||
@ -1192,7 +1192,7 @@ func extractSyncState(ctx context.Context) *SyncerState {
|
|||||||
// total equality of the BeaconEntries in each block.
|
// total equality of the BeaconEntries in each block.
|
||||||
// 3. Traverse the chain backwards, for each tipset:
|
// 3. Traverse the chain backwards, for each tipset:
|
||||||
// 3a. Load it from the chainstore; if found, it move on to its parent.
|
// 3a. Load it from the chainstore; if found, it move on to its parent.
|
||||||
// 3b. Query our peers via BlockSync in batches, requesting up to a
|
// 3b. Query our peers via client in batches, requesting up to a
|
||||||
// maximum of 500 tipsets every time.
|
// maximum of 500 tipsets every time.
|
||||||
//
|
//
|
||||||
// Once we've concluded, if we find a mismatching tipset at the height where the
|
// Once we've concluded, if we find a mismatching tipset at the height where the
|
||||||
@ -1293,7 +1293,7 @@ loop:
|
|||||||
if gap := int(blockSet[len(blockSet)-1].Height() - untilHeight); gap < window {
|
if gap := int(blockSet[len(blockSet)-1].Height() - untilHeight); gap < window {
|
||||||
window = gap
|
window = gap
|
||||||
}
|
}
|
||||||
blks, err := syncer.Bsync.GetBlocks(ctx, at, window)
|
blks, err := syncer.Exchange.GetBlocks(ctx, at, window)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// Most likely our peers aren't fully synced yet, but forwarded
|
// Most likely our peers aren't fully synced yet, but forwarded
|
||||||
// new block message (ideally we'd find better peers)
|
// new block message (ideally we'd find better peers)
|
||||||
@ -1311,7 +1311,7 @@ loop:
|
|||||||
// have. Since we fetch from the head backwards our reassembled chain
|
// have. Since we fetch from the head backwards our reassembled chain
|
||||||
// is sorted in reverse here: we have a child -> parent order, our last
|
// is sorted in reverse here: we have a child -> parent order, our last
|
||||||
// tipset then should be child of the first tipset retrieved.
|
// tipset then should be child of the first tipset retrieved.
|
||||||
// FIXME: The reassembly logic should be part of the `BlockSync`
|
// FIXME: The reassembly logic should be part of the `client`
|
||||||
// service, the consumer should not be concerned with the
|
// service, the consumer should not be concerned with the
|
||||||
// `MaxRequestLength` limitation, it should just be able to request
|
// `MaxRequestLength` limitation, it should just be able to request
|
||||||
// an segment of arbitrary length. The same burden is put on
|
// an segment of arbitrary length. The same burden is put on
|
||||||
@ -1385,7 +1385,7 @@ var ErrForkTooLong = fmt.Errorf("fork longer than threshold")
|
|||||||
// denylist. Else, we find the common ancestor, and add the missing chain
|
// denylist. Else, we find the common ancestor, and add the missing chain
|
||||||
// fragment until the fork point to the returned []TipSet.
|
// fragment until the fork point to the returned []TipSet.
|
||||||
func (syncer *Syncer) syncFork(ctx context.Context, incoming *types.TipSet, known *types.TipSet) ([]*types.TipSet, error) {
|
func (syncer *Syncer) syncFork(ctx context.Context, incoming *types.TipSet, known *types.TipSet) ([]*types.TipSet, error) {
|
||||||
tips, err := syncer.Bsync.GetBlocks(ctx, incoming.Parents(), int(build.ForkLengthThreshold))
|
tips, err := syncer.Exchange.GetBlocks(ctx, incoming.Parents(), int(build.ForkLengthThreshold))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@ -1468,12 +1468,12 @@ mainLoop:
|
|||||||
nextI := (i + 1) - batchSize // want to fetch batchSize values, 'i' points to last one we want to fetch, so its 'inclusive' of our request, thus we need to add one to our request start index
|
nextI := (i + 1) - batchSize // want to fetch batchSize values, 'i' points to last one we want to fetch, so its 'inclusive' of our request, thus we need to add one to our request start index
|
||||||
|
|
||||||
ss.SetStage(api.StageFetchingMessages)
|
ss.SetStage(api.StageFetchingMessages)
|
||||||
var bstout []*blocksync.CompactedMessages
|
var bstout []*exchange.CompactedMessages
|
||||||
for len(bstout) < batchSize {
|
for len(bstout) < batchSize {
|
||||||
next := headers[nextI]
|
next := headers[nextI]
|
||||||
|
|
||||||
nreq := batchSize - len(bstout)
|
nreq := batchSize - len(bstout)
|
||||||
bstips, err := syncer.Bsync.GetChainMessages(ctx, next, uint64(nreq))
|
bstips, err := syncer.Exchange.GetChainMessages(ctx, next, uint64(nreq))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// TODO check errors for temporary nature
|
// TODO check errors for temporary nature
|
||||||
if windowSize > 1 {
|
if windowSize > 1 {
|
||||||
@ -1519,8 +1519,8 @@ mainLoop:
|
|||||||
|
|
||||||
if i >= windowSize {
|
if i >= windowSize {
|
||||||
newWindowSize := windowSize + 10
|
newWindowSize := windowSize + 10
|
||||||
if newWindowSize > int(blocksync.MaxRequestLength) {
|
if newWindowSize > int(exchange.MaxRequestLength) {
|
||||||
newWindowSize = int(blocksync.MaxRequestLength)
|
newWindowSize = int(exchange.MaxRequestLength)
|
||||||
}
|
}
|
||||||
if newWindowSize > windowSize {
|
if newWindowSize > windowSize {
|
||||||
windowSize = newWindowSize
|
windowSize = newWindowSize
|
||||||
@ -1537,7 +1537,7 @@ mainLoop:
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func persistMessages(bs bstore.Blockstore, bst *blocksync.CompactedMessages) error {
|
func persistMessages(bs bstore.Blockstore, bst *exchange.CompactedMessages) error {
|
||||||
for _, m := range bst.Bls {
|
for _, m := range bst.Bls {
|
||||||
//log.Infof("putting BLS message: %s", m.Cid())
|
//log.Infof("putting BLS message: %s", m.Cid())
|
||||||
if _, err := store.PutMessage(bs, m); err != nil {
|
if _, err := store.PutMessage(bs, m); err != nil {
|
||||||
|
@ -49,7 +49,7 @@ var logSetLevel = &cli.Command{
|
|||||||
|
|
||||||
The system flag can be specified multiple times.
|
The system flag can be specified multiple times.
|
||||||
|
|
||||||
eg) log set-level --system chain --system blocksync debug
|
eg) log set-level --system chain --system chainxchg debug
|
||||||
|
|
||||||
Available Levels:
|
Available Levels:
|
||||||
debug
|
debug
|
||||||
|
12
gen/main.go
12
gen/main.go
@ -7,7 +7,7 @@ import (
|
|||||||
gen "github.com/whyrusleeping/cbor-gen"
|
gen "github.com/whyrusleeping/cbor-gen"
|
||||||
|
|
||||||
"github.com/filecoin-project/lotus/api"
|
"github.com/filecoin-project/lotus/api"
|
||||||
"github.com/filecoin-project/lotus/chain/blocksync"
|
"github.com/filecoin-project/lotus/chain/exchange"
|
||||||
"github.com/filecoin-project/lotus/chain/types"
|
"github.com/filecoin-project/lotus/chain/types"
|
||||||
"github.com/filecoin-project/lotus/node/hello"
|
"github.com/filecoin-project/lotus/node/hello"
|
||||||
"github.com/filecoin-project/lotus/paychmgr"
|
"github.com/filecoin-project/lotus/paychmgr"
|
||||||
@ -63,11 +63,11 @@ func main() {
|
|||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
|
|
||||||
err = gen.WriteTupleEncodersToFile("./chain/blocksync/cbor_gen.go", "blocksync",
|
err = gen.WriteTupleEncodersToFile("./chain/exchange/cbor_gen.go", "exchange",
|
||||||
blocksync.Request{},
|
exchange.Request{},
|
||||||
blocksync.Response{},
|
exchange.Response{},
|
||||||
blocksync.CompactedMessages{},
|
exchange.CompactedMessages{},
|
||||||
blocksync.BSTipSet{},
|
exchange.BSTipSet{},
|
||||||
)
|
)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
|
1
go.sum
1
go.sum
@ -1510,6 +1510,7 @@ go.uber.org/atomic v1.5.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ=
|
|||||||
go.uber.org/atomic v1.5.1/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ=
|
go.uber.org/atomic v1.5.1/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ=
|
||||||
go.uber.org/atomic v1.6.0 h1:Ezj3JGmsOnG1MoRWQkPBsKLe9DwWD9QeXzTRzzldNVk=
|
go.uber.org/atomic v1.6.0 h1:Ezj3JGmsOnG1MoRWQkPBsKLe9DwWD9QeXzTRzzldNVk=
|
||||||
go.uber.org/atomic v1.6.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ=
|
go.uber.org/atomic v1.6.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ=
|
||||||
|
go.uber.org/dig v1.8.0 h1:1rR6hnL/bu1EVcjnRDN5kx1vbIjEJDTGhSQ2B3ddpcI=
|
||||||
go.uber.org/dig v1.8.0/go.mod h1:X34SnWGr8Fyla9zQNO2GSO2D+TIuqB14OS8JhYocIyw=
|
go.uber.org/dig v1.8.0/go.mod h1:X34SnWGr8Fyla9zQNO2GSO2D+TIuqB14OS8JhYocIyw=
|
||||||
go.uber.org/dig v1.10.0 h1:yLmDDj9/zuDjv3gz8GQGviXMs9TfysIUMUilCpgzUJY=
|
go.uber.org/dig v1.10.0 h1:yLmDDj9/zuDjv3gz8GQGviXMs9TfysIUMUilCpgzUJY=
|
||||||
go.uber.org/dig v1.10.0/go.mod h1:X34SnWGr8Fyla9zQNO2GSO2D+TIuqB14OS8JhYocIyw=
|
go.uber.org/dig v1.10.0/go.mod h1:X34SnWGr8Fyla9zQNO2GSO2D+TIuqB14OS8JhYocIyw=
|
||||||
|
@ -29,7 +29,7 @@ import (
|
|||||||
"github.com/filecoin-project/lotus/api"
|
"github.com/filecoin-project/lotus/api"
|
||||||
"github.com/filecoin-project/lotus/chain"
|
"github.com/filecoin-project/lotus/chain"
|
||||||
"github.com/filecoin-project/lotus/chain/beacon"
|
"github.com/filecoin-project/lotus/chain/beacon"
|
||||||
"github.com/filecoin-project/lotus/chain/blocksync"
|
"github.com/filecoin-project/lotus/chain/exchange"
|
||||||
"github.com/filecoin-project/lotus/chain/gen"
|
"github.com/filecoin-project/lotus/chain/gen"
|
||||||
"github.com/filecoin-project/lotus/chain/gen/slashfilter"
|
"github.com/filecoin-project/lotus/chain/gen/slashfilter"
|
||||||
"github.com/filecoin-project/lotus/chain/market"
|
"github.com/filecoin-project/lotus/chain/market"
|
||||||
@ -103,7 +103,7 @@ const (
|
|||||||
SetGenesisKey
|
SetGenesisKey
|
||||||
|
|
||||||
RunHelloKey
|
RunHelloKey
|
||||||
RunBlockSyncKey
|
RunChainExchangeKey
|
||||||
RunChainGraphsync
|
RunChainGraphsync
|
||||||
RunPeerMgrKey
|
RunPeerMgrKey
|
||||||
|
|
||||||
@ -238,12 +238,12 @@ func Online() Option {
|
|||||||
|
|
||||||
Override(new(dtypes.ChainGCLocker), blockstore.NewGCLocker),
|
Override(new(dtypes.ChainGCLocker), blockstore.NewGCLocker),
|
||||||
Override(new(dtypes.ChainGCBlockstore), modules.ChainGCBlockstore),
|
Override(new(dtypes.ChainGCBlockstore), modules.ChainGCBlockstore),
|
||||||
Override(new(dtypes.ChainExchange), modules.ChainExchange),
|
Override(new(dtypes.ChainBitswap), modules.ChainBitswap),
|
||||||
Override(new(dtypes.ChainBlockService), modules.ChainBlockservice),
|
Override(new(dtypes.ChainBlockService), modules.ChainBlockService),
|
||||||
|
|
||||||
// Filecoin services
|
// Filecoin services
|
||||||
Override(new(*chain.Syncer), modules.NewSyncer),
|
Override(new(*chain.Syncer), modules.NewSyncer),
|
||||||
Override(new(*blocksync.BlockSync), blocksync.NewClient),
|
Override(new(exchange.Client), exchange.NewClient),
|
||||||
Override(new(*messagepool.MessagePool), modules.MessagePool),
|
Override(new(*messagepool.MessagePool), modules.MessagePool),
|
||||||
|
|
||||||
Override(new(modules.Genesis), modules.ErrorGenesis),
|
Override(new(modules.Genesis), modules.ErrorGenesis),
|
||||||
@ -252,14 +252,14 @@ func Online() Option {
|
|||||||
|
|
||||||
Override(new(dtypes.NetworkName), modules.NetworkName),
|
Override(new(dtypes.NetworkName), modules.NetworkName),
|
||||||
Override(new(*hello.Service), hello.NewHelloService),
|
Override(new(*hello.Service), hello.NewHelloService),
|
||||||
Override(new(*blocksync.BlockSyncService), blocksync.NewBlockSyncService),
|
Override(new(exchange.Server), exchange.NewServer),
|
||||||
Override(new(*peermgr.PeerMgr), peermgr.NewPeerMgr),
|
Override(new(*peermgr.PeerMgr), peermgr.NewPeerMgr),
|
||||||
|
|
||||||
Override(new(dtypes.Graphsync), modules.Graphsync),
|
Override(new(dtypes.Graphsync), modules.Graphsync),
|
||||||
Override(new(*dtypes.MpoolLocker), new(dtypes.MpoolLocker)),
|
Override(new(*dtypes.MpoolLocker), new(dtypes.MpoolLocker)),
|
||||||
|
|
||||||
Override(RunHelloKey, modules.RunHello),
|
Override(RunHelloKey, modules.RunHello),
|
||||||
Override(RunBlockSyncKey, modules.RunBlockSync),
|
Override(RunChainExchangeKey, modules.RunChainExchange),
|
||||||
Override(RunPeerMgrKey, modules.RunPeerMgr),
|
Override(RunPeerMgrKey, modules.RunPeerMgr),
|
||||||
Override(HandleIncomingBlocksKey, modules.HandleIncomingBlocks),
|
Override(HandleIncomingBlocksKey, modules.HandleIncomingBlocks),
|
||||||
|
|
||||||
|
@ -20,7 +20,7 @@ import (
|
|||||||
|
|
||||||
"github.com/filecoin-project/lotus/chain"
|
"github.com/filecoin-project/lotus/chain"
|
||||||
"github.com/filecoin-project/lotus/chain/beacon"
|
"github.com/filecoin-project/lotus/chain/beacon"
|
||||||
"github.com/filecoin-project/lotus/chain/blocksync"
|
"github.com/filecoin-project/lotus/chain/exchange"
|
||||||
"github.com/filecoin-project/lotus/chain/gen/slashfilter"
|
"github.com/filecoin-project/lotus/chain/gen/slashfilter"
|
||||||
"github.com/filecoin-project/lotus/chain/messagepool"
|
"github.com/filecoin-project/lotus/chain/messagepool"
|
||||||
"github.com/filecoin-project/lotus/chain/stmgr"
|
"github.com/filecoin-project/lotus/chain/stmgr"
|
||||||
@ -33,7 +33,7 @@ import (
|
|||||||
"github.com/filecoin-project/lotus/node/repo"
|
"github.com/filecoin-project/lotus/node/repo"
|
||||||
)
|
)
|
||||||
|
|
||||||
func ChainExchange(mctx helpers.MetricsCtx, lc fx.Lifecycle, host host.Host, rt routing.Routing, bs dtypes.ChainGCBlockstore) dtypes.ChainExchange {
|
func ChainBitswap(mctx helpers.MetricsCtx, lc fx.Lifecycle, host host.Host, rt routing.Routing, bs dtypes.ChainGCBlockstore) dtypes.ChainBitswap {
|
||||||
// prefix protocol for chain bitswap
|
// prefix protocol for chain bitswap
|
||||||
// (so bitswap uses /chain/ipfs/bitswap/1.0.0 internally for chain sync stuff)
|
// (so bitswap uses /chain/ipfs/bitswap/1.0.0 internally for chain sync stuff)
|
||||||
bitswapNetwork := network.NewFromIpfsHost(host, rt, network.Prefix("/chain"))
|
bitswapNetwork := network.NewFromIpfsHost(host, rt, network.Prefix("/chain"))
|
||||||
@ -83,7 +83,7 @@ func ChainGCBlockstore(bs dtypes.ChainBlockstore, gcl dtypes.ChainGCLocker) dtyp
|
|||||||
return blockstore.NewGCBlockstore(bs, gcl)
|
return blockstore.NewGCBlockstore(bs, gcl)
|
||||||
}
|
}
|
||||||
|
|
||||||
func ChainBlockservice(bs dtypes.ChainBlockstore, rem dtypes.ChainExchange) dtypes.ChainBlockService {
|
func ChainBlockService(bs dtypes.ChainBlockstore, rem dtypes.ChainBitswap) dtypes.ChainBlockService {
|
||||||
return blockservice.New(bs, rem)
|
return blockservice.New(bs, rem)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -163,8 +163,8 @@ func NetworkName(mctx helpers.MetricsCtx, lc fx.Lifecycle, cs *store.ChainStore,
|
|||||||
return netName, err
|
return netName, err
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewSyncer(lc fx.Lifecycle, sm *stmgr.StateManager, bsync *blocksync.BlockSync, h host.Host, beacon beacon.RandomBeacon, verifier ffiwrapper.Verifier) (*chain.Syncer, error) {
|
func NewSyncer(lc fx.Lifecycle, sm *stmgr.StateManager, exchange exchange.Client, h host.Host, beacon beacon.RandomBeacon, verifier ffiwrapper.Verifier) (*chain.Syncer, error) {
|
||||||
syncer, err := chain.NewSyncer(sm, bsync, h.ConnManager(), h.ID(), beacon, verifier)
|
syncer, err := chain.NewSyncer(sm, exchange, h.ConnManager(), h.ID(), beacon, verifier)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -27,7 +27,7 @@ type ChainBlockstore blockstore.Blockstore
|
|||||||
|
|
||||||
type ChainGCLocker blockstore.GCLocker
|
type ChainGCLocker blockstore.GCLocker
|
||||||
type ChainGCBlockstore blockstore.GCBlockstore
|
type ChainGCBlockstore blockstore.GCBlockstore
|
||||||
type ChainExchange exchange.Interface
|
type ChainBitswap exchange.Interface
|
||||||
type ChainBlockService bserv.BlockService
|
type ChainBlockService bserv.BlockService
|
||||||
|
|
||||||
type ClientMultiDstore *multistore.MultiStore
|
type ClientMultiDstore *multistore.MultiStore
|
||||||
|
@ -17,7 +17,7 @@ import (
|
|||||||
"github.com/filecoin-project/lotus/chain"
|
"github.com/filecoin-project/lotus/chain"
|
||||||
"github.com/filecoin-project/lotus/chain/beacon"
|
"github.com/filecoin-project/lotus/chain/beacon"
|
||||||
"github.com/filecoin-project/lotus/chain/beacon/drand"
|
"github.com/filecoin-project/lotus/chain/beacon/drand"
|
||||||
"github.com/filecoin-project/lotus/chain/blocksync"
|
"github.com/filecoin-project/lotus/chain/exchange"
|
||||||
"github.com/filecoin-project/lotus/chain/messagepool"
|
"github.com/filecoin-project/lotus/chain/messagepool"
|
||||||
"github.com/filecoin-project/lotus/chain/stmgr"
|
"github.com/filecoin-project/lotus/chain/stmgr"
|
||||||
"github.com/filecoin-project/lotus/chain/store"
|
"github.com/filecoin-project/lotus/chain/store"
|
||||||
@ -69,8 +69,9 @@ func RunPeerMgr(mctx helpers.MetricsCtx, lc fx.Lifecycle, pmgr *peermgr.PeerMgr)
|
|||||||
go pmgr.Run(helpers.LifecycleCtx(mctx, lc))
|
go pmgr.Run(helpers.LifecycleCtx(mctx, lc))
|
||||||
}
|
}
|
||||||
|
|
||||||
func RunBlockSync(h host.Host, svc *blocksync.BlockSyncService) {
|
func RunChainExchange(h host.Host, svc exchange.Server) {
|
||||||
h.SetStreamHandler(blocksync.BlockSyncProtocolID, svc.HandleStream)
|
h.SetStreamHandler(exchange.BlockSyncProtocolID, svc.HandleStream) // old
|
||||||
|
h.SetStreamHandler(exchange.ChainExchangeProtocolID, svc.HandleStream) // new
|
||||||
}
|
}
|
||||||
|
|
||||||
func HandleIncomingBlocks(mctx helpers.MetricsCtx, lc fx.Lifecycle, ps *pubsub.PubSub, s *chain.Syncer, bserv dtypes.ChainBlockService, chain *store.ChainStore, stmgr *stmgr.StateManager, h host.Host, nn dtypes.NetworkName) {
|
func HandleIncomingBlocks(mctx helpers.MetricsCtx, lc fx.Lifecycle, ps *pubsub.PubSub, s *chain.Syncer, bserv dtypes.ChainBlockService, chain *store.ChainStore, stmgr *stmgr.StateManager, h host.Host, nn dtypes.NetworkName) {
|
||||||
|
Loading…
Reference in New Issue
Block a user