172 lines
4.5 KiB
Go
172 lines
4.5 KiB
Go
package genesis
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
|
|
"github.com/filecoin-project/go-address"
|
|
"github.com/filecoin-project/go-state-types/abi"
|
|
|
|
"github.com/filecoin-project/specs-actors/actors/builtin"
|
|
"github.com/filecoin-project/specs-actors/actors/util/adt"
|
|
|
|
init_ "github.com/filecoin-project/specs-actors/actors/builtin/init"
|
|
cbor "github.com/ipfs/go-ipld-cbor"
|
|
cbg "github.com/whyrusleeping/cbor-gen"
|
|
"golang.org/x/xerrors"
|
|
|
|
bstore "github.com/filecoin-project/lotus/blockstore"
|
|
"github.com/filecoin-project/lotus/chain/types"
|
|
"github.com/filecoin-project/lotus/genesis"
|
|
)
|
|
|
|
func SetupInitActor(bs bstore.Blockstore, netname string, initialActors []genesis.Actor, rootVerifier genesis.Actor, remainder genesis.Actor) (int64, *types.Actor, map[address.Address]address.Address, error) {
|
|
if len(initialActors) > MaxAccounts {
|
|
return 0, nil, nil, xerrors.New("too many initial actors")
|
|
}
|
|
|
|
var ias init_.State
|
|
ias.NextID = MinerStart
|
|
ias.NetworkName = netname
|
|
|
|
store := adt.WrapStore(context.TODO(), cbor.NewCborStore(bs))
|
|
amap := adt.MakeEmptyMap(store)
|
|
|
|
keyToId := map[address.Address]address.Address{}
|
|
counter := int64(AccountStart)
|
|
|
|
for _, a := range initialActors {
|
|
if a.Type == genesis.TMultisig {
|
|
var ainfo genesis.MultisigMeta
|
|
if err := json.Unmarshal(a.Meta, &ainfo); err != nil {
|
|
return 0, nil, nil, xerrors.Errorf("unmarshaling account meta: %w", err)
|
|
}
|
|
for _, e := range ainfo.Signers {
|
|
|
|
if _, ok := keyToId[e]; ok {
|
|
continue
|
|
}
|
|
|
|
fmt.Printf("init set %s t0%d\n", e, counter)
|
|
|
|
value := cbg.CborInt(counter)
|
|
if err := amap.Put(abi.AddrKey(e), &value); err != nil {
|
|
return 0, nil, nil, err
|
|
}
|
|
counter = counter + 1
|
|
var err error
|
|
keyToId[e], err = address.NewIDAddress(uint64(value))
|
|
if err != nil {
|
|
return 0, nil, nil, err
|
|
}
|
|
|
|
}
|
|
// Need to add actors for all multisigs too
|
|
continue
|
|
}
|
|
|
|
if a.Type != genesis.TAccount {
|
|
return 0, nil, nil, xerrors.Errorf("unsupported account type: %s", a.Type)
|
|
}
|
|
|
|
var ainfo genesis.AccountMeta
|
|
if err := json.Unmarshal(a.Meta, &ainfo); err != nil {
|
|
return 0, nil, nil, xerrors.Errorf("unmarshaling account meta: %w", err)
|
|
}
|
|
|
|
fmt.Printf("init set %s t0%d\n", ainfo.Owner, counter)
|
|
|
|
value := cbg.CborInt(counter)
|
|
if err := amap.Put(abi.AddrKey(ainfo.Owner), &value); err != nil {
|
|
return 0, nil, nil, err
|
|
}
|
|
counter = counter + 1
|
|
|
|
var err error
|
|
keyToId[ainfo.Owner], err = address.NewIDAddress(uint64(value))
|
|
if err != nil {
|
|
return 0, nil, nil, err
|
|
}
|
|
}
|
|
|
|
setupMsig := func(meta json.RawMessage) error {
|
|
var ainfo genesis.MultisigMeta
|
|
if err := json.Unmarshal(meta, &ainfo); err != nil {
|
|
return xerrors.Errorf("unmarshaling account meta: %w", err)
|
|
}
|
|
for _, e := range ainfo.Signers {
|
|
if _, ok := keyToId[e]; ok {
|
|
continue
|
|
}
|
|
fmt.Printf("init set %s t0%d\n", e, counter)
|
|
|
|
value := cbg.CborInt(counter)
|
|
if err := amap.Put(abi.AddrKey(e), &value); err != nil {
|
|
return err
|
|
}
|
|
counter = counter + 1
|
|
var err error
|
|
keyToId[e], err = address.NewIDAddress(uint64(value))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
if rootVerifier.Type == genesis.TAccount {
|
|
var ainfo genesis.AccountMeta
|
|
if err := json.Unmarshal(rootVerifier.Meta, &ainfo); err != nil {
|
|
return 0, nil, nil, xerrors.Errorf("unmarshaling account meta: %w", err)
|
|
}
|
|
value := cbg.CborInt(80)
|
|
if err := amap.Put(abi.AddrKey(ainfo.Owner), &value); err != nil {
|
|
return 0, nil, nil, err
|
|
}
|
|
} else if rootVerifier.Type == genesis.TMultisig {
|
|
err := setupMsig(rootVerifier.Meta)
|
|
if err != nil {
|
|
return 0, nil, nil, xerrors.Errorf("setting up root verifier msig: %w", err)
|
|
}
|
|
}
|
|
|
|
if remainder.Type == genesis.TAccount {
|
|
var ainfo genesis.AccountMeta
|
|
if err := json.Unmarshal(remainder.Meta, &ainfo); err != nil {
|
|
return 0, nil, nil, xerrors.Errorf("unmarshaling account meta: %w", err)
|
|
}
|
|
|
|
// TODO: Use builtin.ReserveAddress...
|
|
value := cbg.CborInt(90)
|
|
if err := amap.Put(abi.AddrKey(ainfo.Owner), &value); err != nil {
|
|
return 0, nil, nil, err
|
|
}
|
|
} else if remainder.Type == genesis.TMultisig {
|
|
err := setupMsig(remainder.Meta)
|
|
if err != nil {
|
|
return 0, nil, nil, xerrors.Errorf("setting up remainder msig: %w", err)
|
|
}
|
|
}
|
|
|
|
amapaddr, err := amap.Root()
|
|
if err != nil {
|
|
return 0, nil, nil, err
|
|
}
|
|
ias.AddressMap = amapaddr
|
|
|
|
statecid, err := store.Put(store.Context(), &ias)
|
|
if err != nil {
|
|
return 0, nil, nil, err
|
|
}
|
|
|
|
act := &types.Actor{
|
|
Code: builtin.InitActorCodeID,
|
|
Head: statecid,
|
|
}
|
|
|
|
return counter, act, keyToId, nil
|
|
}
|