5353210814
* Read a subset of filecoin state over the full node API * wip * import export wip * extract actors from message * generate car for any state * library for providing a pruned statetree * test vector schema draft + example 'message' class test vector. * message => messages test vector class. * fixup * wip * use lb.NewBlockstore with ID * fixup lotus-soup, and generate * fix deals * magic params * work on schema / export of test vector * fixup * wip deserialise state tree * pass at building a test case from a message * progress loading / serializing * recreation of ipld nodes * generation of vector creates json * kick off tvx tool. * wip * wip * retain init actor state. * initial test with printed out state * remove testing.T, but keep require and assert libraries * wip refactor state tree plucking. * simplified * removed factories * remove builder.Build ; remove interface - use concrete iface * comment out validateState * remove Validator * remove TestDriverBuilder * remove client * remove box * remove gen * remove factories * remove KeyManager interfafce * moved stuff around * remove ValidationConfig * extract randomness * extract config and key_manager * extract statewrapper * extract applier * rename factories to various * flatten chain-validation package * initial marshal of test vector * do not require schema package * fixup * run all messages tests * better names * run all messages tests * remove Indent setting from JSON encoder for now * refactor, and actually running successfully ;-) * remove irrelevant files; rename extract-msg command. * remove root CID from state_tree object in schema. * add tvx/lotus package; adjust .gitignore. * tidy up command flag management. * add comment. * remove validateState and trackState * remove xerrors * remove NewVM * remove commented out RootCID sets * enable more tests * add all `message_application` tests * delete all.json * update Message struct * fix message serialization * support multiple messages * gofmt * remove custom Message and SignedMessage types * update tests with gzip and adhere to new schema for compressed CAR * improved iface for Marshal * update Validation * remove test-suites and utils * better names for chain. methods * go mod tidy * remove top-level dummyT Co-authored-by: Will Scott <will@cypherpunk.email> Co-authored-by: Raúl Kripalani <raul@protocol.ai>
151 lines
4.2 KiB
Go
151 lines
4.2 KiB
Go
package drivers
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
|
|
acrypto "github.com/filecoin-project/specs-actors/actors/crypto"
|
|
|
|
"github.com/filecoin-project/specs-actors/actors/abi"
|
|
"github.com/filecoin-project/specs-actors/actors/abi/big"
|
|
"github.com/filecoin-project/specs-actors/actors/builtin"
|
|
"github.com/filecoin-project/specs-actors/actors/puppet"
|
|
"github.com/ipfs/go-cid"
|
|
|
|
"github.com/filecoin-project/lotus/chain/stmgr"
|
|
"github.com/filecoin-project/lotus/chain/store"
|
|
"github.com/filecoin-project/lotus/chain/types"
|
|
"github.com/filecoin-project/lotus/chain/vm"
|
|
vtypes "github.com/filecoin-project/oni/tvx/chain/types"
|
|
)
|
|
|
|
// Applier applies messages to state trees and storage.
|
|
type Applier struct {
|
|
stateWrapper *StateWrapper
|
|
syscalls vm.SyscallBuilder
|
|
}
|
|
|
|
func NewApplier(sw *StateWrapper, syscalls vm.SyscallBuilder) *Applier {
|
|
return &Applier{sw, syscalls}
|
|
}
|
|
|
|
func (a *Applier) ApplyMessage(epoch abi.ChainEpoch, lm *types.Message) (vtypes.ApplyMessageResult, error) {
|
|
receipt, penalty, reward, err := a.applyMessage(epoch, lm)
|
|
|
|
return vtypes.ApplyMessageResult{
|
|
Receipt: receipt,
|
|
Penalty: penalty,
|
|
Reward: reward,
|
|
Root: a.stateWrapper.Root().String(),
|
|
}, err
|
|
}
|
|
|
|
func (a *Applier) ApplySignedMessage(epoch abi.ChainEpoch, msg *types.SignedMessage) (vtypes.ApplyMessageResult, error) {
|
|
var lm types.ChainMsg
|
|
switch msg.Signature.Type {
|
|
case acrypto.SigTypeSecp256k1:
|
|
lm = msg
|
|
case acrypto.SigTypeBLS:
|
|
lm = &msg.Message
|
|
default:
|
|
return vtypes.ApplyMessageResult{}, errors.New("Unknown signature type")
|
|
}
|
|
// TODO: Validate the sig first
|
|
receipt, penalty, reward, err := a.applyMessage(epoch, lm)
|
|
return vtypes.ApplyMessageResult{
|
|
Receipt: receipt,
|
|
Penalty: penalty,
|
|
Reward: reward,
|
|
Root: a.stateWrapper.Root().String(),
|
|
}, err
|
|
|
|
}
|
|
|
|
func (a *Applier) ApplyTipSetMessages(epoch abi.ChainEpoch, blocks []vtypes.BlockMessagesInfo, rnd RandomnessSource) (vtypes.ApplyTipSetResult, error) {
|
|
cs := store.NewChainStore(a.stateWrapper.bs, a.stateWrapper.ds, a.syscalls)
|
|
sm := stmgr.NewStateManager(cs)
|
|
|
|
var bms []stmgr.BlockMessages
|
|
for _, b := range blocks {
|
|
bm := stmgr.BlockMessages{
|
|
Miner: b.Miner,
|
|
WinCount: 1,
|
|
}
|
|
|
|
for _, m := range b.BLSMessages {
|
|
bm.BlsMessages = append(bm.BlsMessages, m)
|
|
}
|
|
|
|
for _, m := range b.SECPMessages {
|
|
bm.SecpkMessages = append(bm.SecpkMessages, m)
|
|
}
|
|
|
|
bms = append(bms, bm)
|
|
}
|
|
|
|
var receipts []vtypes.MessageReceipt
|
|
sroot, _, err := sm.ApplyBlocks(context.TODO(), epoch-1, a.stateWrapper.Root(), bms, epoch, &randWrapper{rnd}, func(c cid.Cid, msg *types.Message, ret *vm.ApplyRet) error {
|
|
if msg.From == builtin.SystemActorAddr {
|
|
return nil // ignore reward and cron calls
|
|
}
|
|
rval := ret.Return
|
|
if rval == nil {
|
|
rval = []byte{} // chain validation tests expect empty arrays to not be nil...
|
|
}
|
|
receipts = append(receipts, vtypes.MessageReceipt{
|
|
ExitCode: ret.ExitCode,
|
|
ReturnValue: rval,
|
|
|
|
GasUsed: vtypes.GasUnits(ret.GasUsed),
|
|
})
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return vtypes.ApplyTipSetResult{}, err
|
|
}
|
|
|
|
a.stateWrapper.stateRoot = sroot
|
|
|
|
return vtypes.ApplyTipSetResult{
|
|
Receipts: receipts,
|
|
Root: a.stateWrapper.Root().String(),
|
|
}, nil
|
|
}
|
|
|
|
func (a *Applier) applyMessage(epoch abi.ChainEpoch, lm types.ChainMsg) (vtypes.MessageReceipt, abi.TokenAmount, abi.TokenAmount, error) {
|
|
ctx := context.TODO()
|
|
base := a.stateWrapper.Root()
|
|
|
|
lotusVM, err := vm.NewVM(base, epoch, &vmRand{}, a.stateWrapper.bs, a.syscalls, nil)
|
|
// need to modify the VM invoker to add the puppet actor
|
|
chainValInvoker := vm.NewInvoker()
|
|
chainValInvoker.Register(puppet.PuppetActorCodeID, puppet.Actor{}, puppet.State{})
|
|
lotusVM.SetInvoker(chainValInvoker)
|
|
if err != nil {
|
|
return vtypes.MessageReceipt{}, big.Zero(), big.Zero(), err
|
|
}
|
|
|
|
ret, err := lotusVM.ApplyMessage(ctx, lm)
|
|
if err != nil {
|
|
return vtypes.MessageReceipt{}, big.Zero(), big.Zero(), err
|
|
}
|
|
|
|
rval := ret.Return
|
|
if rval == nil {
|
|
rval = []byte{}
|
|
}
|
|
|
|
a.stateWrapper.stateRoot, err = lotusVM.Flush(ctx)
|
|
if err != nil {
|
|
return vtypes.MessageReceipt{}, big.Zero(), big.Zero(), err
|
|
}
|
|
|
|
mr := vtypes.MessageReceipt{
|
|
ExitCode: ret.ExitCode,
|
|
ReturnValue: rval,
|
|
GasUsed: vtypes.GasUnits(ret.GasUsed),
|
|
}
|
|
|
|
return mr, ret.Penalty, abi.NewTokenAmount(ret.GasUsed), nil
|
|
}
|