package multisig import ( actorstypes "github.com/filecoin-project/go-state-types/actors" "github.com/ipfs/go-cid" "fmt" "github.com/minio/blake2b-simd" cbg "github.com/whyrusleeping/cbor-gen" "golang.org/x/xerrors" "github.com/filecoin-project/go-address" "github.com/filecoin-project/go-state-types/abi" "github.com/filecoin-project/go-state-types/cbor" msig{{.latestVersion}} "github.com/filecoin-project/go-state-types/builtin{{import .latestVersion}}multisig" {{range .versions}} {{if (le . 7)}} builtin{{.}} "github.com/filecoin-project/specs-actors{{import .}}actors/builtin" {{end}} {{end}} builtintypes "github.com/filecoin-project/go-state-types/builtin" "github.com/filecoin-project/go-state-types/manifest" "github.com/filecoin-project/lotus/chain/actors" "github.com/filecoin-project/lotus/chain/actors/adt" "github.com/filecoin-project/lotus/chain/types" ) func Load(store adt.Store, act *types.Actor) (State, error) { if name, av, ok := actors.GetActorMetaByCode(act.Code); ok { if name != manifest.MultisigKey { return nil, xerrors.Errorf("actor code is not multisig: %s", name) } switch av { {{range .versions}} {{if (ge . 8)}} case actorstypes.Version{{.}}: return load{{.}}(store, act.Head) {{end}} {{end}} } } switch act.Code { {{range .versions}} {{if (le . 7)}} case builtin{{.}}.MultisigActorCodeID: return load{{.}}(store, act.Head) {{end}} {{end}} } return nil, xerrors.Errorf("unknown actor code %s", act.Code) } func MakeState(store adt.Store, av actorstypes.Version, signers []address.Address, threshold uint64, startEpoch abi.ChainEpoch, unlockDuration abi.ChainEpoch, initialBalance abi.TokenAmount) (State, error) { switch av { {{range .versions}} case actorstypes.Version{{.}}: return make{{.}}(store, signers, threshold, startEpoch, unlockDuration, initialBalance) {{end}} } return nil, xerrors.Errorf("unknown actor version %d", av) } type State interface { cbor.Marshaler Code() cid.Cid ActorKey() string ActorVersion() actorstypes.Version LockedBalance(epoch abi.ChainEpoch) (abi.TokenAmount, error) StartEpoch() (abi.ChainEpoch, error) UnlockDuration() (abi.ChainEpoch, error) InitialBalance() (abi.TokenAmount, error) Threshold() (uint64, error) Signers() ([]address.Address, error) ForEachPendingTxn(func(id int64, txn Transaction) error) error PendingTxnChanged(State) (bool, error) transactions() (adt.Map, error) decodeTransaction(val *cbg.Deferred) (Transaction, error) GetState() interface{} } type Transaction = msig{{.latestVersion}}.Transaction var Methods = builtintypes.MethodsMultisig func Message(version actorstypes.Version, from address.Address) MessageBuilder { switch version { {{range .versions}} case actorstypes.Version{{.}}: return message{{.}}{{"{"}}{{if (ge . 2)}}message0{from}{{else}}from{{end}}} {{end}} default: panic(fmt.Sprintf("unsupported actors version: %d", version)) } } type MessageBuilder interface { // Create a new multisig with the specified parameters. Create(signers []address.Address, threshold uint64, vestingStart, vestingDuration abi.ChainEpoch, initialAmount abi.TokenAmount) (*types.Message, error) // Propose a transaction to the given multisig. Propose(msig, target address.Address, amt abi.TokenAmount, method abi.MethodNum, params []byte) (*types.Message, error) // Approve a multisig transaction. The "hash" is optional. Approve(msig address.Address, txID uint64, hash *ProposalHashData) (*types.Message, error) // Cancel a multisig transaction. The "hash" is optional. Cancel(msig address.Address, txID uint64, hash *ProposalHashData) (*types.Message, error) } // this type is the same between v0 and v2 type ProposalHashData = msig{{.latestVersion}}.ProposalHashData type ProposeReturn = msig{{.latestVersion}}.ProposeReturn type ProposeParams = msig{{.latestVersion}}.ProposeParams type ApproveReturn = msig{{.latestVersion}}.ApproveReturn func txnParams(id uint64, data *ProposalHashData) ([]byte, error) { params := msig{{.latestVersion}}.TxnIDParams{ID: msig{{.latestVersion}}.TxnID(id)} if data != nil { if data.Requester.Protocol() != address.ID { return nil, xerrors.Errorf("proposer address must be an ID address, was %s", data.Requester) } if data.Value.Sign() == -1 { return nil, xerrors.Errorf("proposal value must be non-negative, was %s", data.Value) } if data.To == address.Undef { return nil, xerrors.Errorf("proposed destination address must be set") } pser, err := data.Serialize() if err != nil { return nil, err } hash := blake2b.Sum256(pser) params.ProposalHash = hash[:] } return actors.SerializeParams(¶ms) } func AllCodes() []cid.Cid { return []cid.Cid{ {{range .versions}} (&state{{.}}{}).Code(), {{- end}} } }