136 lines
4.4 KiB
Go
136 lines
4.4 KiB
Go
package sealing
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
commcid "github.com/filecoin-project/go-fil-commcid"
|
|
"github.com/filecoin-project/lotus/lib/zerocomm"
|
|
"github.com/ipfs/go-cid"
|
|
cbg "github.com/whyrusleeping/cbor-gen"
|
|
|
|
"github.com/filecoin-project/specs-actors/actors/builtin"
|
|
"github.com/multiformats/go-multihash"
|
|
"golang.org/x/xerrors"
|
|
|
|
"github.com/filecoin-project/go-address"
|
|
|
|
"github.com/filecoin-project/lotus/build"
|
|
"github.com/filecoin-project/lotus/chain/actors"
|
|
"github.com/filecoin-project/lotus/chain/types"
|
|
)
|
|
|
|
// TODO: For now we handle this by halting state execution, when we get jsonrpc reconnecting
|
|
// We should implement some wait-for-api logic
|
|
type ErrApi struct{ error }
|
|
|
|
type ErrInvalidDeals struct{ error }
|
|
type ErrInvalidPiece struct{ error }
|
|
type ErrExpiredDeals struct{ error }
|
|
|
|
type ErrBadCommD struct{ error }
|
|
type ErrExpiredTicket struct{ error }
|
|
|
|
// checkPieces validates that:
|
|
// - Each piece han a corresponding on chain deal
|
|
// - Piece commitments match with on chain deals
|
|
// - Piece sizes match
|
|
// - Deals aren't expired
|
|
func checkPieces(ctx context.Context, si SectorInfo, api sealingApi) error {
|
|
head, err := api.ChainHead(ctx)
|
|
if err != nil {
|
|
return &ErrApi{xerrors.Errorf("getting chain head: %w", err)}
|
|
}
|
|
|
|
for i, piece := range si.Pieces {
|
|
if piece.DealID == nil {
|
|
exp := zerocomm.ForSize(piece.Size)
|
|
if string(piece.CommP) != string(exp[:]) {
|
|
return &ErrInvalidPiece{xerrors.Errorf("deal %d piece %d had non-zero CommP %+v", piece.DealID, i, piece.CommP)}
|
|
}
|
|
continue
|
|
}
|
|
deal, err := api.StateMarketStorageDeal(ctx, *piece.DealID, types.EmptyTSK)
|
|
if err != nil {
|
|
return &ErrApi{xerrors.Errorf("getting deal %d for piece %d: %w", piece.DealID, i, err)}
|
|
}
|
|
|
|
h, err := multihash.Decode(deal.Proposal.PieceCID.Hash())
|
|
if err != nil {
|
|
return &ErrInvalidDeals{xerrors.Errorf("decoding piece CID: %w", err)}
|
|
}
|
|
|
|
if string(h.Digest) != string(piece.CommP) {
|
|
return &ErrInvalidDeals{xerrors.Errorf("piece %d (or %d) of sector %d refers deal %d with wrong CommP: %x != %x", i, len(si.Pieces), si.SectorID, piece.DealID, piece.CommP, h.Digest)}
|
|
}
|
|
|
|
if piece.Size != deal.Proposal.PieceSize.Unpadded() {
|
|
return &ErrInvalidDeals{xerrors.Errorf("piece %d (or %d) of sector %d refers deal %d with different size: %d != %d", i, len(si.Pieces), si.SectorID, piece.DealID, piece.Size, deal.Proposal.PieceSize)}
|
|
}
|
|
|
|
if head.Height() >= deal.Proposal.StartEpoch {
|
|
return &ErrExpiredDeals{xerrors.Errorf("piece %d (or %d) of sector %d refers expired deal %d - should start at %d, head %d", i, len(si.Pieces), si.SectorID, piece.DealID, deal.Proposal.StartEpoch, head.Height())}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// checkSeal checks that data commitment generated in the sealing process
|
|
// matches pieces, and that the seal ticket isn't expired
|
|
func checkSeal(ctx context.Context, maddr address.Address, si SectorInfo, api sealingApi) (err error) {
|
|
head, err := api.ChainHead(ctx)
|
|
if err != nil {
|
|
return &ErrApi{xerrors.Errorf("getting chain head: %w", err)}
|
|
}
|
|
|
|
ssize, err := api.StateMinerSectorSize(ctx, maddr, head.Key())
|
|
if err != nil {
|
|
return &ErrApi{err}
|
|
}
|
|
|
|
ccparams, err := actors.SerializeParams(&actors.ComputeDataCommitmentParams{
|
|
DealIDs: si.deals(),
|
|
SectorSize: ssize,
|
|
})
|
|
if err != nil {
|
|
return xerrors.Errorf("computing params for ComputeDataCommitment: %w", err)
|
|
}
|
|
|
|
ccmt := &types.Message{
|
|
To: actors.StorageMarketAddress,
|
|
From: maddr,
|
|
Value: types.NewInt(0),
|
|
GasPrice: types.NewInt(0),
|
|
GasLimit: types.NewInt(9999999999),
|
|
Method: builtin.MethodsMarket.ComputeDataCommitment,
|
|
Params: ccparams,
|
|
}
|
|
r, err := api.StateCall(ctx, ccmt, types.EmptyTSK)
|
|
if err != nil {
|
|
return &ErrApi{xerrors.Errorf("calling ComputeDataCommitment: %w", err)}
|
|
}
|
|
if r.ExitCode != 0 {
|
|
return &ErrBadCommD{xerrors.Errorf("receipt for ComputeDataCommitment had exit code %d", r.ExitCode)}
|
|
}
|
|
|
|
var c cbg.CborCid
|
|
if err := c.UnmarshalCBOR(bytes.NewReader(r.Return)); err != nil {
|
|
return err
|
|
}
|
|
cd, err := commcid.CIDToDataCommitmentV1(cid.Cid(c))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if string(cd) != string(si.CommD) {
|
|
return &ErrBadCommD{xerrors.Errorf("on chain CommD differs from sector: %x != %x", r.Return, si.CommD)}
|
|
}
|
|
|
|
if int64(head.Height())-int64(si.Ticket.BlockHeight+build.SealRandomnessLookback) > build.SealRandomnessLookbackLimit {
|
|
return &ErrExpiredTicket{xerrors.Errorf("ticket expired: seal height: %d, head: %d", si.Ticket.BlockHeight+build.SealRandomnessLookback, head.Height())}
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|