282 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			282 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package full
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"context"
 | |
| 	"github.com/filecoin-project/go-amt-ipld"
 | |
| 	"strconv"
 | |
| 
 | |
| 	cid "github.com/ipfs/go-cid"
 | |
| 	"github.com/ipfs/go-hamt-ipld"
 | |
| 	"github.com/libp2p/go-libp2p-core/peer"
 | |
| 	cbg "github.com/whyrusleeping/cbor-gen"
 | |
| 	"go.uber.org/fx"
 | |
| 	"golang.org/x/xerrors"
 | |
| 
 | |
| 	"github.com/filecoin-project/lotus/api"
 | |
| 	"github.com/filecoin-project/lotus/chain/actors"
 | |
| 	"github.com/filecoin-project/lotus/chain/address"
 | |
| 	"github.com/filecoin-project/lotus/chain/gen"
 | |
| 	"github.com/filecoin-project/lotus/chain/state"
 | |
| 	"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"
 | |
| 	"github.com/filecoin-project/lotus/chain/wallet"
 | |
| 	"github.com/filecoin-project/lotus/lib/bufbstore"
 | |
| )
 | |
| 
 | |
| type StateAPI struct {
 | |
| 	fx.In
 | |
| 
 | |
| 	// TODO: the wallet here is only needed because we have the MinerCreateBlock
 | |
| 	// API attached to the state API. It probably should live somewhere better
 | |
| 	Wallet *wallet.Wallet
 | |
| 
 | |
| 	StateManager *stmgr.StateManager
 | |
| 	Chain        *store.ChainStore
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateMinerSectors(ctx context.Context, addr address.Address, ts *types.TipSet) ([]*api.SectorInfo, error) {
 | |
| 	return stmgr.GetMinerSectorSet(ctx, a.StateManager, ts, addr)
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateMinerProvingSet(ctx context.Context, addr address.Address, ts *types.TipSet) ([]*api.SectorInfo, error) {
 | |
| 	return stmgr.GetMinerProvingSet(ctx, a.StateManager, ts, addr)
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateMinerPower(ctx context.Context, maddr address.Address, ts *types.TipSet) (api.MinerPower, error) {
 | |
| 	mpow, tpow, err := stmgr.GetPower(ctx, a.StateManager, ts, maddr)
 | |
| 	if err != nil {
 | |
| 		return api.MinerPower{}, err
 | |
| 	}
 | |
| 
 | |
| 	return api.MinerPower{
 | |
| 		MinerPower: mpow,
 | |
| 		TotalPower: tpow,
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateMinerWorker(ctx context.Context, m address.Address, ts *types.TipSet) (address.Address, error) {
 | |
| 	return stmgr.GetMinerWorker(ctx, a.StateManager, ts, m)
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateMinerPeerID(ctx context.Context, m address.Address, ts *types.TipSet) (peer.ID, error) {
 | |
| 	return stmgr.GetMinerPeerID(ctx, a.StateManager, ts, m)
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateMinerProvingPeriodEnd(ctx context.Context, actor address.Address, ts *types.TipSet) (uint64, error) {
 | |
| 	return stmgr.GetMinerProvingPeriodEnd(ctx, a.StateManager, ts, actor)
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateMinerSectorSize(ctx context.Context, actor address.Address, ts *types.TipSet) (uint64, error) {
 | |
| 	return stmgr.GetMinerSectorSize(ctx, a.StateManager, ts, actor)
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StatePledgeCollateral(ctx context.Context, ts *types.TipSet) (types.BigInt, error) {
 | |
| 	param, err := actors.SerializeParams(&actors.PledgeCollateralParams{Size: types.NewInt(0)})
 | |
| 	if err != nil {
 | |
| 		return types.NewInt(0), err
 | |
| 	}
 | |
| 
 | |
| 	ret, aerr := a.StateManager.Call(ctx, &types.Message{
 | |
| 		From:   actors.StoragePowerAddress,
 | |
| 		To:     actors.StoragePowerAddress,
 | |
| 		Method: actors.SPAMethods.PledgeCollateralForSize,
 | |
| 
 | |
| 		Params: param,
 | |
| 	}, ts)
 | |
| 	if aerr != nil {
 | |
| 		return types.NewInt(0), xerrors.Errorf("failed to get miner worker addr: %w", err)
 | |
| 	}
 | |
| 
 | |
| 	if ret.ExitCode != 0 {
 | |
| 		return types.NewInt(0), xerrors.Errorf("failed to get miner worker addr (exit code %d)", ret.ExitCode)
 | |
| 	}
 | |
| 
 | |
| 	return types.BigFromBytes(ret.Return), nil
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateCall(ctx context.Context, msg *types.Message, ts *types.TipSet) (*types.MessageReceipt, error) {
 | |
| 	return a.StateManager.Call(ctx, msg, ts)
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateReplay(ctx context.Context, ts *types.TipSet, mc cid.Cid) (*api.ReplayResults, error) {
 | |
| 	m, r, err := a.StateManager.Replay(ctx, ts, mc)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	var errstr string
 | |
| 	if r.ActorErr != nil {
 | |
| 		errstr = r.ActorErr.Error()
 | |
| 	}
 | |
| 
 | |
| 	return &api.ReplayResults{
 | |
| 		Msg:     m,
 | |
| 		Receipt: &r.MessageReceipt,
 | |
| 		Error:   errstr,
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) stateForTs(ctx context.Context, ts *types.TipSet) (*state.StateTree, error) {
 | |
| 	if ts == nil {
 | |
| 		ts = a.Chain.GetHeaviestTipSet()
 | |
| 	}
 | |
| 
 | |
| 	st, _, err := a.StateManager.TipSetState(ctx, ts)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	buf := bufbstore.NewBufferedBstore(a.Chain.Blockstore())
 | |
| 	cst := hamt.CSTFromBstore(buf)
 | |
| 	return state.LoadStateTree(cst, st)
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateGetActor(ctx context.Context, actor address.Address, ts *types.TipSet) (*types.Actor, error) {
 | |
| 	state, err := a.stateForTs(ctx, ts)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	return state.GetActor(actor)
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateReadState(ctx context.Context, act *types.Actor, ts *types.TipSet) (*api.ActorState, error) {
 | |
| 	state, err := a.stateForTs(ctx, ts)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	blk, err := state.Store.Blocks.GetBlock(ctx, act.Head)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	oif, err := vm.DumpActorState(act.Code, blk.RawData())
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	return &api.ActorState{
 | |
| 		Balance: act.Balance,
 | |
| 		State:   oif,
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| // This is on StateAPI because miner.Miner requires this, and MinerAPI requires miner.Miner
 | |
| func (a *StateAPI) MinerCreateBlock(ctx context.Context, addr address.Address, parents *types.TipSet, tickets []*types.Ticket, proof types.ElectionProof, msgs []*types.SignedMessage, ts uint64) (*types.BlockMsg, error) {
 | |
| 	fblk, err := gen.MinerCreateBlock(ctx, a.StateManager, a.Wallet, addr, parents, tickets, proof, msgs, ts)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	var out types.BlockMsg
 | |
| 	out.Header = fblk.Header
 | |
| 	for _, msg := range fblk.BlsMessages {
 | |
| 		out.BlsMessages = append(out.BlsMessages, msg.Cid())
 | |
| 	}
 | |
| 	for _, msg := range fblk.SecpkMessages {
 | |
| 		out.SecpkMessages = append(out.SecpkMessages, msg.Cid())
 | |
| 	}
 | |
| 
 | |
| 	return &out, nil
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateWaitMsg(ctx context.Context, msg cid.Cid) (*api.MsgWait, error) {
 | |
| 	// TODO: consider using event system for this, expose confidence
 | |
| 
 | |
| 	ts, recpt, err := a.StateManager.WaitForMessage(ctx, msg)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	return &api.MsgWait{
 | |
| 		Receipt: *recpt,
 | |
| 		TipSet:  ts,
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateListMiners(ctx context.Context, ts *types.TipSet) ([]address.Address, error) {
 | |
| 	var state actors.StoragePowerState
 | |
| 	if _, err := a.StateManager.LoadActorState(ctx, actors.StoragePowerAddress, &state, ts); err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	cst := hamt.CSTFromBstore(a.StateManager.ChainStore().Blockstore())
 | |
| 	miners, err := actors.MinerSetList(ctx, cst, state.Miners)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	return miners, nil
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateListActors(ctx context.Context, ts *types.TipSet) ([]address.Address, error) {
 | |
| 	return a.StateManager.ListAllActors(ctx, ts)
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateMarketBalance(ctx context.Context, addr address.Address, ts *types.TipSet) (actors.StorageParticipantBalance, error) {
 | |
| 	return a.StateManager.MarketBalance(ctx, addr, ts)
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateMarketParticipants(ctx context.Context, ts *types.TipSet) (map[string]actors.StorageParticipantBalance, error) {
 | |
| 	out := map[string]actors.StorageParticipantBalance{}
 | |
| 
 | |
| 	var state actors.StorageMarketState
 | |
| 	if _, err := a.StateManager.LoadActorState(ctx, actors.StorageMarketAddress, &state, ts); err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	cst := hamt.CSTFromBstore(a.StateManager.ChainStore().Blockstore())
 | |
| 	nd, err := hamt.LoadNode(ctx, cst, state.Balances)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	err = nd.ForEach(ctx, func(k string, val interface{}) error {
 | |
| 		cv := val.(*cbg.Deferred)
 | |
| 		a, err := address.NewFromBytes([]byte(k))
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		var b actors.StorageParticipantBalance
 | |
| 		if err := b.UnmarshalCBOR(bytes.NewReader(cv.Raw)); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		out[a.String()] = b
 | |
| 		return nil
 | |
| 	})
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return out, nil
 | |
| }
 | |
| 
 | |
| func (a *StateAPI) StateMarketDeals(ctx context.Context, ts *types.TipSet) (map[string]actors.OnChainDeal, error) {
 | |
| 	out := map[string]actors.OnChainDeal{}
 | |
| 
 | |
| 	var state actors.StorageMarketState
 | |
| 	if _, err := a.StateManager.LoadActorState(ctx, actors.StorageMarketAddress, &state, ts); err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	blks := amt.WrapBlockstore(a.StateManager.ChainStore().Blockstore())
 | |
| 	da, err := amt.LoadAMT(blks, state.Deals)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	if err := da.ForEach(func(i uint64, v *cbg.Deferred) error {
 | |
| 		var d actors.OnChainDeal
 | |
| 		if err := d.UnmarshalCBOR(bytes.NewReader(v.Raw)); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		out[strconv.FormatInt(int64(i), 10)] = d
 | |
| 		return nil
 | |
| 	}); err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return out, nil
 | |
| }
 |