lotus/tvx/drivers/state_driver.go
Anton Evangelatov 5353210814
generate test-vectors based on tests from chain-validation project (#181)
* 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>
2020-08-05 19:40:09 +02:00

207 lines
6.3 KiB
Go

package drivers
import (
"context"
"github.com/filecoin-project/go-address"
"github.com/filecoin-project/oni/tvx/chain"
abi_spec "github.com/filecoin-project/specs-actors/actors/abi"
big_spec "github.com/filecoin-project/specs-actors/actors/abi/big"
miner_spec "github.com/filecoin-project/specs-actors/actors/builtin/miner"
power_spec "github.com/filecoin-project/specs-actors/actors/builtin/power"
"github.com/filecoin-project/specs-actors/actors/runtime"
adt_spec "github.com/filecoin-project/specs-actors/actors/util/adt"
"github.com/ipfs/go-cid"
"github.com/stretchr/testify/require"
cbg "github.com/whyrusleeping/cbor-gen"
builtin_spec "github.com/filecoin-project/specs-actors/actors/builtin"
account_spec "github.com/filecoin-project/specs-actors/actors/builtin/account"
)
var (
SECP = address.SECP256K1
BLS = address.BLS
)
// StateDriver mutates and inspects a state.
type StateDriver struct {
st *StateWrapper
w *KeyManager
rs RandomnessSource
minerInfo *MinerInfo
// Mapping for IDAddresses to their pubkey/actor addresses. Used for lookup when signing messages.
actorIDMap map[address.Address]address.Address
}
// info about the state drivers builtin miner
type MinerInfo struct {
Owner address.Address
OwnerID address.Address
Worker address.Address
WorkerID address.Address
}
// NewStateDriver creates a new state driver for a state.
func NewStateDriver(st *StateWrapper, w *KeyManager) *StateDriver {
return &StateDriver{st, w, NewRandomnessSource(), nil, make(map[address.Address]address.Address)}
}
// State returns the state.
func (d *StateDriver) State() *StateWrapper {
return d.st
}
func (d *StateDriver) Wallet() *KeyManager {
return d.w
}
func (d *StateDriver) Randomness() RandomnessSource {
return d.rs
}
func (d *StateDriver) GetState(c cid.Cid, out cbg.CBORUnmarshaler) {
err := d.st.StoreGet(c, out)
require.NoError(t, err)
}
func (d *StateDriver) PutState(in cbg.CBORMarshaler) cid.Cid {
c, err := d.st.StorePut(in)
require.NoError(t, err)
return c
}
func (d *StateDriver) GetActorState(actorAddr address.Address, out cbg.CBORUnmarshaler) {
actor, err := d.State().Actor(actorAddr)
require.NoError(t, err)
require.NotNil(t, actor)
d.GetState(actor.Head(), out)
}
// NewAccountActor installs a new account actor, returning the address.
func (d *StateDriver) NewAccountActor(addrType address.Protocol, balanceAttoFil abi_spec.TokenAmount) (pubkey address.Address, id address.Address) {
var addr address.Address
switch addrType {
case address.SECP256K1:
addr = d.w.NewSECP256k1AccountAddress()
case address.BLS:
addr = d.w.NewBLSAccountAddress()
default:
require.FailNowf(t, "unsupported address", "protocol for account actor: %v", addrType)
}
_, idAddr, err := d.st.CreateActor(builtin_spec.AccountActorCodeID, addr, balanceAttoFil, &account_spec.State{Address: addr})
require.NoError(t, err)
d.actorIDMap[idAddr] = addr
return addr, idAddr
}
func (d *StateDriver) ActorPubKey(idAddress address.Address) address.Address {
if idAddress.Protocol() != address.ID {
t.Fatalf("ActorPubKey methods expects ID protocol address. actual: %v", idAddress.Protocol())
}
pubkeyAddr, found := d.actorIDMap[idAddress]
if !found {
t.Fatalf("Failed to find pubkey address for: %s", idAddress)
}
return pubkeyAddr
}
func (d *StateDriver) BuiltinMinerInfo() *MinerInfo {
return d.minerInfo
}
// create miner without sending a message. modify the init and power actor manually
func (d *StateDriver) newMinerAccountActor(sealProofType abi_spec.RegisteredSealProof, periodBoundary abi_spec.ChainEpoch) address.Address {
// creat a miner, owner, and its worker
minerOwnerPk, minerOwnerID := d.NewAccountActor(address.SECP256K1, big_spec.NewInt(1_000_000_000))
minerWorkerPk, minerWorkerID := d.NewAccountActor(address.BLS, big_spec.Zero())
expectedMinerActorIDAddress := chain.MustNewIDAddr(chain.MustIdFromAddress(minerWorkerID) + 1)
minerActorAddrs := computeInitActorExecReturn(minerWorkerPk, 0, 1, expectedMinerActorIDAddress)
d.minerInfo = &MinerInfo{
Owner: minerOwnerPk,
OwnerID: minerOwnerID,
Worker: minerWorkerPk,
WorkerID: minerWorkerID,
}
ss, err := sealProofType.SectorSize()
require.NoError(t, err)
ps, err := sealProofType.WindowPoStPartitionSectors()
require.NoError(t, err)
mi := &miner_spec.MinerInfo{
Owner: minerOwnerID,
Worker: minerWorkerID,
PendingWorkerKey: nil,
PeerId: abi_spec.PeerID("chain-validation"),
Multiaddrs: nil,
SealProofType: sealProofType,
SectorSize: ss,
WindowPoStPartitionSectors: ps,
}
mc, err := d.st.StorePut(mi)
require.NoError(t, err)
// create the miner actor s.t. it exists in the init actors map
minerState, err := miner_spec.ConstructState(mc, periodBoundary, EmptyBitfieldCid, EmptyArrayCid, EmptyMapCid, EmptyDeadlinesCid)
require.NoError(t, err)
_, minerActorIDAddr, err := d.State().CreateActor(builtin_spec.StorageMinerActorCodeID, minerActorAddrs.RobustAddress, big_spec.Zero(), minerState)
require.NoError(t, err)
require.Equal(t, expectedMinerActorIDAddress, minerActorIDAddr)
// a miner actor has been created, exists in the state tree, and has an entry in the init actor.
// next update the storage power actor to track the miner
var spa power_spec.State
d.GetActorState(builtin_spec.StoragePowerActorAddr, &spa)
// set the miners claim
hm, err := adt_spec.AsMap(AsStore(d.State()), spa.Claims)
require.NoError(t, err)
// add claim for the miner
err = hm.Put(adt_spec.AddrKey(minerActorIDAddr), &power_spec.Claim{
RawBytePower: abi_spec.NewStoragePower(0),
QualityAdjPower: abi_spec.NewTokenAmount(0),
})
require.NoError(t, err)
// save the claim
spa.Claims, err = hm.Root()
require.NoError(t, err)
// update miner count
spa.MinerCount += 1
// update storage power actor's state in the tree
d.PutState(&spa)
return minerActorIDAddr
}
func AsStore(vmw *StateWrapper) adt_spec.Store {
return &storeWrapper{vmw: vmw}
}
type storeWrapper struct {
vmw *StateWrapper
}
func (s storeWrapper) Context() context.Context {
return context.TODO()
}
func (s storeWrapper) Get(ctx context.Context, c cid.Cid, out interface{}) error {
return s.vmw.StoreGet(c, out.(runtime.CBORUnmarshaler))
}
func (s storeWrapper) Put(ctx context.Context, v interface{}) (cid.Cid, error) {
return s.vmw.StorePut(v.(runtime.CBORMarshaler))
}