lotus/extern/sector-storage/mock/mock.go

547 lines
14 KiB
Go
Raw Normal View History

2020-03-23 11:40:02 +00:00
package mock
import (
"bytes"
"context"
"crypto/sha256"
2020-03-23 11:40:02 +00:00
"fmt"
"io"
2021-05-18 11:35:25 +00:00
"io/ioutil"
2020-03-23 11:40:02 +00:00
"math/rand"
"sync"
2020-10-08 01:09:33 +00:00
proof2 "github.com/filecoin-project/specs-actors/v2/actors/runtime/proof"
2020-09-07 06:08:53 +00:00
ffiwrapper2 "github.com/filecoin-project/go-commp-utils/ffiwrapper"
2020-03-23 11:40:02 +00:00
commcid "github.com/filecoin-project/go-fil-commcid"
2020-09-07 03:49:10 +00:00
"github.com/filecoin-project/go-state-types/abi"
2020-03-23 11:40:02 +00:00
"github.com/filecoin-project/specs-storage/storage"
"github.com/ipfs/go-cid"
logging "github.com/ipfs/go-log/v2"
2020-03-23 11:40:02 +00:00
"golang.org/x/xerrors"
"github.com/filecoin-project/lotus/extern/sector-storage/ffiwrapper"
"github.com/filecoin-project/lotus/extern/sector-storage/storiface"
2020-03-23 11:40:02 +00:00
)
var log = logging.Logger("sbmock")
type SectorMgr struct {
sectors map[abi.SectorID]*sectorState
failPoSt bool
2020-05-26 14:39:25 +00:00
pieces map[cid.Cid][]byte
2020-03-23 11:40:02 +00:00
nextSectorID abi.SectorNumber
lk sync.Mutex
}
type mockVerif struct{}
2020-11-05 06:34:24 +00:00
func NewMockSectorMgr(genesisSectors []abi.SectorID) *SectorMgr {
2020-07-23 17:00:20 +00:00
sectors := make(map[abi.SectorID]*sectorState)
for _, sid := range genesisSectors {
sectors[sid] = &sectorState{
failed: false,
state: stateCommit,
}
}
2020-03-23 11:40:02 +00:00
return &SectorMgr{
2020-07-23 17:00:20 +00:00
sectors: sectors,
2020-05-26 14:39:25 +00:00
pieces: map[cid.Cid][]byte{},
2020-03-23 11:40:02 +00:00
nextSectorID: 5,
}
}
const (
statePacking = iota
statePreCommit
2020-04-01 23:18:20 +00:00
stateCommit // nolint
2020-03-23 11:40:02 +00:00
)
type sectorState struct {
2020-09-10 20:07:20 +00:00
pieces []cid.Cid
failed bool
corrupted bool
2020-03-23 11:40:02 +00:00
state int
lk sync.Mutex
}
2020-11-05 06:34:24 +00:00
func (mgr *SectorMgr) NewSector(ctx context.Context, sector storage.SectorRef) error {
2020-03-23 11:40:02 +00:00
return nil
}
2020-11-05 06:34:24 +00:00
func (mgr *SectorMgr) AddPiece(ctx context.Context, sectorID storage.SectorRef, existingPieces []abi.UnpaddedPieceSize, size abi.UnpaddedPieceSize, r io.Reader) (abi.PieceInfo, error) {
log.Warn("Add piece: ", sectorID, size, sectorID.ProofType)
2020-03-23 11:40:02 +00:00
2020-05-26 14:39:25 +00:00
var b bytes.Buffer
tr := io.TeeReader(r, &b)
c, err := ffiwrapper2.GeneratePieceCIDFromFile(sectorID.ProofType, tr, size)
2020-03-23 11:40:02 +00:00
if err != nil {
return abi.PieceInfo{}, xerrors.Errorf("failed to generate piece cid: %w", err)
}
log.Warn("Generated Piece CID: ", c)
mgr.lk.Lock()
2020-05-26 14:39:25 +00:00
mgr.pieces[c] = b.Bytes()
2020-11-05 06:34:24 +00:00
ss, ok := mgr.sectors[sectorID.ID]
if !ok {
ss = &sectorState{
state: statePacking,
}
2020-11-05 06:34:24 +00:00
mgr.sectors[sectorID.ID] = ss
}
mgr.lk.Unlock()
ss.lk.Lock()
2020-03-23 11:40:02 +00:00
ss.pieces = append(ss.pieces, c)
ss.lk.Unlock()
2020-03-23 11:40:02 +00:00
return abi.PieceInfo{
2020-03-23 11:40:02 +00:00
Size: size.Padded(),
PieceCID: c,
}, nil
}
2020-04-10 19:12:23 +00:00
func (mgr *SectorMgr) AcquireSectorNumber() (abi.SectorNumber, error) {
mgr.lk.Lock()
defer mgr.lk.Unlock()
id := mgr.nextSectorID
mgr.nextSectorID++
2020-03-23 11:40:02 +00:00
return id, nil
}
2020-11-05 06:34:24 +00:00
func (mgr *SectorMgr) ForceState(sid storage.SectorRef, st int) error {
2020-09-30 17:18:38 +00:00
mgr.lk.Lock()
2020-11-05 06:34:24 +00:00
ss, ok := mgr.sectors[sid.ID]
2020-09-30 17:18:38 +00:00
mgr.lk.Unlock()
if !ok {
return xerrors.Errorf("no sector with id %d in storage", sid)
}
ss.state = st
return nil
}
2020-11-05 06:34:24 +00:00
func (mgr *SectorMgr) SealPreCommit1(ctx context.Context, sid storage.SectorRef, ticket abi.SealRandomness, pieces []abi.PieceInfo) (out storage.PreCommit1Out, err error) {
2020-04-10 19:12:23 +00:00
mgr.lk.Lock()
2020-11-05 06:34:24 +00:00
ss, ok := mgr.sectors[sid.ID]
2020-04-10 19:12:23 +00:00
mgr.lk.Unlock()
2020-03-23 11:40:02 +00:00
if !ok {
2020-03-26 19:34:38 +00:00
return nil, xerrors.Errorf("no sector with id %d in storage", sid)
2020-03-23 11:40:02 +00:00
}
2020-11-05 06:34:24 +00:00
ssize, err := sid.ProofType.SectorSize()
if err != nil {
return nil, xerrors.Errorf("failed to get proof sector size: %w", err)
}
2020-03-23 11:40:02 +00:00
ss.lk.Lock()
defer ss.lk.Unlock()
2020-11-05 06:34:24 +00:00
ussize := abi.PaddedPieceSize(ssize).Unpadded()
2020-03-23 11:40:02 +00:00
// TODO: verify pieces in sinfo.pieces match passed in pieces
var sum abi.UnpaddedPieceSize
for _, p := range pieces {
sum += p.Size.Unpadded()
}
if sum != ussize {
return nil, xerrors.Errorf("aggregated piece sizes don't match up: %d != %d", sum, ussize)
}
if ss.state != statePacking {
return nil, xerrors.Errorf("cannot call pre-seal on sector not in 'packing' state")
}
opFinishWait(ctx)
ss.state = statePreCommit
pis := make([]abi.PieceInfo, len(ss.pieces))
for i, piece := range ss.pieces {
pis[i] = abi.PieceInfo{
Size: pieces[i].Size,
PieceCID: piece,
}
}
2020-11-05 06:34:24 +00:00
commd, err := MockVerifier.GenerateDataCommitment(sid.ProofType, pis)
2020-03-23 11:40:02 +00:00
if err != nil {
return nil, err
}
_, _, cc, err := commcid.CIDToCommitment(commd)
2020-03-23 11:40:02 +00:00
if err != nil {
panic(err)
}
cc[0] ^= 'd'
return cc, nil
}
2020-11-05 06:34:24 +00:00
func (mgr *SectorMgr) SealPreCommit2(ctx context.Context, sid storage.SectorRef, phase1Out storage.PreCommit1Out) (cids storage.SectorCids, err error) {
2020-03-23 11:40:02 +00:00
db := []byte(string(phase1Out))
db[0] ^= 'd'
d, _ := commcid.DataCommitmentV1ToCID(db)
2020-03-23 11:40:02 +00:00
commr := make([]byte, 32)
for i := range db {
commr[32-(i+1)] = db[i]
}
commR, _ := commcid.ReplicaCommitmentV1ToCID(commr)
2020-03-23 11:40:02 +00:00
return storage.SectorCids{
Unsealed: d,
Sealed: commR,
}, nil
}
2020-11-05 06:34:24 +00:00
func (mgr *SectorMgr) SealCommit1(ctx context.Context, sid storage.SectorRef, ticket abi.SealRandomness, seed abi.InteractiveSealRandomness, pieces []abi.PieceInfo, cids storage.SectorCids) (output storage.Commit1Out, err error) {
2020-04-10 19:12:23 +00:00
mgr.lk.Lock()
2020-11-05 06:34:24 +00:00
ss, ok := mgr.sectors[sid.ID]
2020-04-10 19:12:23 +00:00
mgr.lk.Unlock()
2020-03-23 11:40:02 +00:00
if !ok {
return nil, xerrors.Errorf("no such sector %d", sid)
}
ss.lk.Lock()
defer ss.lk.Unlock()
if ss.failed {
return nil, xerrors.Errorf("[mock] cannot commit failed sector %d", sid)
}
if ss.state != statePreCommit {
return nil, xerrors.Errorf("cannot commit sector that has not been precommitted")
}
opFinishWait(ctx)
var out [32]byte
for i := range out {
2020-11-05 06:34:24 +00:00
out[i] = cids.Unsealed.Bytes()[i] + cids.Sealed.Bytes()[31-i] - ticket[i]*seed[i] ^ byte(sid.ID.Number&0xff)
2020-03-23 11:40:02 +00:00
}
return out[:], nil
}
2020-11-05 06:34:24 +00:00
func (mgr *SectorMgr) SealCommit2(ctx context.Context, sid storage.SectorRef, phase1Out storage.Commit1Out) (proof storage.Proof, err error) {
plen, err := sid.ProofType.ProofSize()
if err != nil {
return nil, err
}
out := make([]byte, plen)
2020-10-12 06:31:23 +00:00
for i := range out[:len(phase1Out)] {
2020-11-05 06:34:24 +00:00
out[i] = phase1Out[i] ^ byte(sid.ID.Number&0xff)
2020-03-23 11:40:02 +00:00
}
return out[:], nil
}
// Test Instrumentation Methods
2020-11-05 06:34:24 +00:00
func (mgr *SectorMgr) MarkFailed(sid storage.SectorRef, failed bool) error {
2020-04-10 19:12:23 +00:00
mgr.lk.Lock()
defer mgr.lk.Unlock()
2020-11-05 06:34:24 +00:00
ss, ok := mgr.sectors[sid.ID]
2020-03-23 11:40:02 +00:00
if !ok {
2020-03-26 19:34:38 +00:00
return fmt.Errorf("no such sector in storage")
2020-03-23 11:40:02 +00:00
}
ss.failed = failed
2020-03-23 11:40:02 +00:00
return nil
}
func (mgr *SectorMgr) Fail() {
mgr.lk.Lock()
defer mgr.lk.Unlock()
mgr.failPoSt = true
return
}
2020-11-05 06:34:24 +00:00
func (mgr *SectorMgr) MarkCorrupted(sid storage.SectorRef, corrupted bool) error {
mgr.lk.Lock()
defer mgr.lk.Unlock()
2020-11-05 06:34:24 +00:00
ss, ok := mgr.sectors[sid.ID]
if !ok {
return fmt.Errorf("no such sector in storage")
}
ss.corrupted = corrupted
return nil
}
2020-03-23 11:40:02 +00:00
func opFinishWait(ctx context.Context) {
val, ok := ctx.Value("opfinish").(chan struct{})
if !ok {
return
}
<-val
}
func AddOpFinish(ctx context.Context) (context.Context, func()) {
done := make(chan struct{})
2020-08-16 10:40:35 +00:00
return context.WithValue(ctx, "opfinish", done), func() { // nolint
2020-03-23 11:40:02 +00:00
close(done)
}
}
2020-10-08 01:09:33 +00:00
func (mgr *SectorMgr) GenerateWinningPoSt(ctx context.Context, minerID abi.ActorID, sectorInfo []proof2.SectorInfo, randomness abi.PoStRandomness) ([]proof2.PoStProof, error) {
mgr.lk.Lock()
defer mgr.lk.Unlock()
2020-07-23 17:00:20 +00:00
return generateFakePoSt(sectorInfo, abi.RegisteredSealProof.RegisteredWinningPoStProof, randomness), nil
2020-03-23 11:40:02 +00:00
}
2020-10-08 01:09:33 +00:00
func (mgr *SectorMgr) GenerateWindowPoSt(ctx context.Context, minerID abi.ActorID, sectorInfo []proof2.SectorInfo, randomness abi.PoStRandomness) ([]proof2.PoStProof, []abi.SectorID, error) {
mgr.lk.Lock()
defer mgr.lk.Unlock()
if mgr.failPoSt {
return nil, nil, xerrors.Errorf("failed to post (mock)")
}
2020-10-08 01:09:33 +00:00
si := make([]proof2.SectorInfo, 0, len(sectorInfo))
2020-07-23 17:00:20 +00:00
var skipped []abi.SectorID
var err error
2020-07-23 17:00:20 +00:00
for _, info := range sectorInfo {
sid := abi.SectorID{
Miner: minerID,
Number: info.SectorNumber,
}
_, found := mgr.sectors[sid]
if found && !mgr.sectors[sid].failed && !mgr.sectors[sid].corrupted {
2020-07-23 17:00:20 +00:00
si = append(si, info)
} else {
skipped = append(skipped, sid)
err = xerrors.Errorf("skipped some sectors")
2020-07-23 17:00:20 +00:00
}
}
if err != nil {
return nil, skipped, err
}
2020-07-23 17:00:20 +00:00
return generateFakePoSt(si, abi.RegisteredSealProof.RegisteredWindowPoStProof, randomness), skipped, nil
}
2020-10-08 01:09:33 +00:00
func generateFakePoStProof(sectorInfo []proof2.SectorInfo, randomness abi.PoStRandomness) []byte {
randomness[31] &= 0x3f
hasher := sha256.New()
_, _ = hasher.Write(randomness)
2020-07-23 17:00:20 +00:00
for _, info := range sectorInfo {
err := info.MarshalCBOR(hasher)
if err != nil {
panic(err)
}
}
return hasher.Sum(nil)
2020-07-23 17:00:20 +00:00
}
2020-07-23 17:00:20 +00:00
2020-10-08 01:09:33 +00:00
func generateFakePoSt(sectorInfo []proof2.SectorInfo, rpt func(abi.RegisteredSealProof) (abi.RegisteredPoStProof, error), randomness abi.PoStRandomness) []proof2.PoStProof {
wp, err := rpt(sectorInfo[0].SealProof)
2020-06-15 12:32:17 +00:00
if err != nil {
panic(err)
}
2020-10-08 01:09:33 +00:00
return []proof2.PoStProof{
{
2020-06-15 12:32:17 +00:00
PoStProof: wp,
ProofBytes: generateFakePoStProof(sectorInfo, randomness),
},
}
2020-03-23 11:40:02 +00:00
}
2021-05-18 11:35:25 +00:00
func (mgr *SectorMgr) ReadPiece(ctx context.Context, sector storage.SectorRef, offset storiface.UnpaddedByteIndex, size abi.UnpaddedPieceSize, ticket abi.SealRandomness, unsealed cid.Cid) (io.ReadCloser, bool, error) {
if offset != 0 {
2020-03-23 11:40:02 +00:00
panic("implme")
}
2020-05-26 08:19:42 +00:00
2021-05-18 11:35:25 +00:00
return ioutil.NopCloser(bytes.NewReader(mgr.pieces[mgr.sectors[sector.ID].pieces[0]][:size])), false, nil
2020-03-23 11:40:02 +00:00
}
2020-11-05 06:34:24 +00:00
func (mgr *SectorMgr) StageFakeData(mid abi.ActorID, spt abi.RegisteredSealProof) (storage.SectorRef, []abi.PieceInfo, error) {
psize, err := spt.SectorSize()
if err != nil {
return storage.SectorRef{}, nil, err
}
usize := abi.PaddedPieceSize(psize).Unpadded()
2020-04-10 19:12:23 +00:00
sid, err := mgr.AcquireSectorNumber()
2020-03-23 11:40:02 +00:00
if err != nil {
2020-11-05 06:34:24 +00:00
return storage.SectorRef{}, nil, err
2020-03-23 11:40:02 +00:00
}
buf := make([]byte, usize)
2020-08-16 10:40:35 +00:00
_, _ = rand.Read(buf) // nolint:gosec
2020-03-23 11:40:02 +00:00
2020-11-05 06:34:24 +00:00
id := storage.SectorRef{
ID: abi.SectorID{
Miner: mid,
Number: sid,
},
ProofType: spt,
2020-03-23 11:40:02 +00:00
}
2020-04-10 19:12:23 +00:00
pi, err := mgr.AddPiece(context.TODO(), id, nil, usize, bytes.NewReader(buf))
2020-03-23 11:40:02 +00:00
if err != nil {
2020-11-05 06:34:24 +00:00
return storage.SectorRef{}, nil, err
2020-03-23 11:40:02 +00:00
}
return id, []abi.PieceInfo{pi}, nil
}
2020-11-05 06:34:24 +00:00
func (mgr *SectorMgr) FinalizeSector(context.Context, storage.SectorRef, []storage.Range) error {
return nil
}
2020-11-05 06:34:24 +00:00
func (mgr *SectorMgr) ReleaseUnsealed(ctx context.Context, sector storage.SectorRef, safeToFree []storage.Range) error {
return nil
}
2020-11-05 06:34:24 +00:00
func (mgr *SectorMgr) Remove(ctx context.Context, sector storage.SectorRef) error {
mgr.lk.Lock()
defer mgr.lk.Unlock()
2020-11-05 06:34:24 +00:00
if _, has := mgr.sectors[sector.ID]; !has {
return xerrors.Errorf("sector not found")
}
2020-11-05 06:34:24 +00:00
delete(mgr.sectors, sector.ID)
2020-03-23 11:40:02 +00:00
return nil
}
2020-12-01 23:35:55 +00:00
func (mgr *SectorMgr) CheckProvable(ctx context.Context, pp abi.RegisteredPoStProof, ids []storage.SectorRef, rg storiface.RGetter) (map[abi.SectorID]string, error) {
2020-11-27 15:34:41 +00:00
bad := map[abi.SectorID]string{}
2020-07-23 17:00:20 +00:00
for _, sid := range ids {
2020-11-05 12:43:05 +00:00
_, found := mgr.sectors[sid.ID]
2020-07-23 17:00:20 +00:00
2020-11-05 12:43:05 +00:00
if !found || mgr.sectors[sid.ID].failed {
2020-11-27 15:34:41 +00:00
bad[sid.ID] = "mock fail"
2020-07-23 17:00:20 +00:00
}
}
return bad, nil
2020-05-16 21:03:29 +00:00
}
2020-11-17 15:17:45 +00:00
func (mgr *SectorMgr) ReturnAddPiece(ctx context.Context, callID storiface.CallID, pi abi.PieceInfo, err *storiface.CallError) error {
2020-09-07 14:35:54 +00:00
panic("not supported")
}
2020-11-17 15:17:45 +00:00
func (mgr *SectorMgr) ReturnSealPreCommit1(ctx context.Context, callID storiface.CallID, p1o storage.PreCommit1Out, err *storiface.CallError) error {
2020-09-07 14:35:54 +00:00
panic("not supported")
}
2020-11-17 15:17:45 +00:00
func (mgr *SectorMgr) ReturnSealPreCommit2(ctx context.Context, callID storiface.CallID, sealed storage.SectorCids, err *storiface.CallError) error {
2020-09-07 14:35:54 +00:00
panic("not supported")
}
2020-11-17 15:17:45 +00:00
func (mgr *SectorMgr) ReturnSealCommit1(ctx context.Context, callID storiface.CallID, out storage.Commit1Out, err *storiface.CallError) error {
2020-09-07 14:35:54 +00:00
panic("not supported")
}
2020-11-17 15:17:45 +00:00
func (mgr *SectorMgr) ReturnSealCommit2(ctx context.Context, callID storiface.CallID, proof storage.Proof, err *storiface.CallError) error {
2020-09-07 14:35:54 +00:00
panic("not supported")
}
2020-11-17 15:17:45 +00:00
func (mgr *SectorMgr) ReturnFinalizeSector(ctx context.Context, callID storiface.CallID, err *storiface.CallError) error {
2020-09-07 14:35:54 +00:00
panic("not supported")
}
2020-11-17 15:17:45 +00:00
func (mgr *SectorMgr) ReturnReleaseUnsealed(ctx context.Context, callID storiface.CallID, err *storiface.CallError) error {
2020-09-07 14:35:54 +00:00
panic("not supported")
}
2020-11-17 15:17:45 +00:00
func (mgr *SectorMgr) ReturnMoveStorage(ctx context.Context, callID storiface.CallID, err *storiface.CallError) error {
2020-09-07 14:35:54 +00:00
panic("not supported")
}
2020-11-17 15:17:45 +00:00
func (mgr *SectorMgr) ReturnUnsealPiece(ctx context.Context, callID storiface.CallID, err *storiface.CallError) error {
2020-09-07 14:35:54 +00:00
panic("not supported")
}
2020-11-17 15:17:45 +00:00
func (mgr *SectorMgr) ReturnReadPiece(ctx context.Context, callID storiface.CallID, ok bool, err *storiface.CallError) error {
2020-09-07 14:35:54 +00:00
panic("not supported")
}
2020-11-17 15:17:45 +00:00
func (mgr *SectorMgr) ReturnFetch(ctx context.Context, callID storiface.CallID, err *storiface.CallError) error {
2020-09-07 14:35:54 +00:00
panic("not supported")
}
2021-05-18 11:35:25 +00:00
func (mgr *SectorMgr) SectorsUnsealPiece(ctx context.Context, sector storage.SectorRef, offset storiface.UnpaddedByteIndex, size abi.UnpaddedPieceSize, randomness abi.SealRandomness, commd *cid.Cid) error {
return nil
}
2020-10-08 01:09:33 +00:00
func (m mockVerif) VerifySeal(svi proof2.SealVerifyInfo) (bool, error) {
plen, err := svi.SealProof.ProofSize()
if err != nil {
return false, err
}
if len(svi.Proof) != int(plen) {
2020-03-23 11:40:02 +00:00
return false, nil
}
2020-10-12 06:31:23 +00:00
// only the first 32 bytes, the rest are 0.
for i, b := range svi.Proof[:32] {
2020-05-22 01:19:46 +00:00
if b != svi.UnsealedCID.Bytes()[i]+svi.SealedCID.Bytes()[31-i]-svi.InteractiveRandomness[i]*svi.Randomness[i] {
2020-03-23 11:40:02 +00:00
return false, nil
}
}
return true, nil
}
2020-10-08 01:09:33 +00:00
func (m mockVerif) VerifyWinningPoSt(ctx context.Context, info proof2.WinningPoStVerifyInfo) (bool, error) {
info.Randomness[31] &= 0x3f
return true, nil
2020-04-10 19:12:23 +00:00
}
2020-10-08 01:09:33 +00:00
func (m mockVerif) VerifyWindowPoSt(ctx context.Context, info proof2.WindowPoStVerifyInfo) (bool, error) {
2020-07-23 17:00:20 +00:00
if len(info.Proofs) != 1 {
return false, xerrors.Errorf("expected 1 proof entry")
}
proof := info.Proofs[0]
expected := generateFakePoStProof(info.ChallengedSectors, info.Randomness)
if !bytes.Equal(proof.ProofBytes, expected) {
return false, xerrors.Errorf("bad proof")
2020-07-23 17:00:20 +00:00
}
return true, nil
2020-04-10 19:12:23 +00:00
}
2020-06-15 12:32:17 +00:00
func (m mockVerif) GenerateDataCommitment(pt abi.RegisteredSealProof, pieces []abi.PieceInfo) (cid.Cid, error) {
2020-03-26 02:50:56 +00:00
return ffiwrapper.GenerateUnsealedCID(pt, pieces)
2020-03-23 11:40:02 +00:00
}
2020-06-15 12:32:17 +00:00
func (m mockVerif) GenerateWinningPoStSectorChallenge(ctx context.Context, proofType abi.RegisteredPoStProof, minerID abi.ActorID, randomness abi.PoStRandomness, eligibleSectorCount uint64) ([]uint64, error) {
2020-04-17 22:52:42 +00:00
return []uint64{0}, nil
}
2020-03-23 11:40:02 +00:00
var MockVerifier = mockVerif{}
var _ storage.Sealer = &SectorMgr{}
2020-03-26 02:50:56 +00:00
var _ ffiwrapper.Verifier = MockVerifier