203 lines
5.4 KiB
Go
203 lines
5.4 KiB
Go
package full
|
|
|
|
import (
|
|
"context"
|
|
"github.com/filecoin-project/specs-actors/actors/abi"
|
|
"github.com/filecoin-project/specs-actors/actors/abi/big"
|
|
|
|
"github.com/ipfs/go-cid"
|
|
"go.uber.org/fx"
|
|
"golang.org/x/xerrors"
|
|
|
|
"github.com/filecoin-project/go-address"
|
|
"github.com/filecoin-project/lotus/api"
|
|
"github.com/filecoin-project/lotus/chain/messagepool"
|
|
"github.com/filecoin-project/lotus/chain/store"
|
|
"github.com/filecoin-project/lotus/chain/types"
|
|
"github.com/filecoin-project/lotus/node/modules/dtypes"
|
|
)
|
|
|
|
type MpoolAPI struct {
|
|
fx.In
|
|
|
|
WalletAPI
|
|
GasAPI
|
|
|
|
Chain *store.ChainStore
|
|
|
|
Mpool *messagepool.MessagePool
|
|
|
|
PushLocks *dtypes.MpoolLocker
|
|
}
|
|
|
|
func (a *MpoolAPI) MpoolGetConfig(context.Context) (*types.MpoolConfig, error) {
|
|
return a.Mpool.GetConfig(), nil
|
|
}
|
|
|
|
func (a *MpoolAPI) MpoolSetConfig(ctx context.Context, cfg *types.MpoolConfig) error {
|
|
a.Mpool.SetConfig(cfg)
|
|
return nil
|
|
}
|
|
|
|
func (a *MpoolAPI) MpoolSelect(ctx context.Context, tsk types.TipSetKey, ticketQuality float64) ([]*types.SignedMessage, error) {
|
|
ts, err := a.Chain.GetTipSetFromKey(tsk)
|
|
if err != nil {
|
|
return nil, xerrors.Errorf("loading tipset %s: %w", tsk, err)
|
|
}
|
|
|
|
return a.Mpool.SelectMessages(ts, ticketQuality)
|
|
}
|
|
|
|
func (a *MpoolAPI) MpoolPending(ctx context.Context, tsk types.TipSetKey) ([]*types.SignedMessage, error) {
|
|
ts, err := a.Chain.GetTipSetFromKey(tsk)
|
|
if err != nil {
|
|
return nil, xerrors.Errorf("loading tipset %s: %w", tsk, err)
|
|
}
|
|
pending, mpts := a.Mpool.Pending()
|
|
|
|
haveCids := map[cid.Cid]struct{}{}
|
|
for _, m := range pending {
|
|
haveCids[m.Cid()] = struct{}{}
|
|
}
|
|
|
|
if ts == nil || mpts.Height() > ts.Height() {
|
|
return pending, nil
|
|
}
|
|
|
|
for {
|
|
if mpts.Height() == ts.Height() {
|
|
if mpts.Equals(ts) {
|
|
return pending, nil
|
|
}
|
|
// different blocks in tipsets
|
|
|
|
have, err := a.Mpool.MessagesForBlocks(ts.Blocks())
|
|
if err != nil {
|
|
return nil, xerrors.Errorf("getting messages for base ts: %w", err)
|
|
}
|
|
|
|
for _, m := range have {
|
|
haveCids[m.Cid()] = struct{}{}
|
|
}
|
|
}
|
|
|
|
msgs, err := a.Mpool.MessagesForBlocks(ts.Blocks())
|
|
if err != nil {
|
|
return nil, xerrors.Errorf(": %w", err)
|
|
}
|
|
|
|
for _, m := range msgs {
|
|
if _, ok := haveCids[m.Cid()]; ok {
|
|
continue
|
|
}
|
|
|
|
haveCids[m.Cid()] = struct{}{}
|
|
pending = append(pending, m)
|
|
}
|
|
|
|
if mpts.Height() >= ts.Height() {
|
|
return pending, nil
|
|
}
|
|
|
|
ts, err = a.Chain.LoadTipSet(ts.Parents())
|
|
if err != nil {
|
|
return nil, xerrors.Errorf("loading parent tipset: %w", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (a *MpoolAPI) MpoolPush(ctx context.Context, smsg *types.SignedMessage) (cid.Cid, error) {
|
|
return a.Mpool.Push(smsg)
|
|
}
|
|
|
|
func capGasFee(msg *types.Message, maxFee abi.TokenAmount) {
|
|
if maxFee.Equals(big.Zero()) {
|
|
return
|
|
}
|
|
|
|
chainFee := types.BigMul(msg.GasFeeCap, types.NewInt(uint64(msg.GasLimit)))
|
|
minerFee := types.BigMul(msg.GasPremium, types.NewInt(uint64(msg.GasLimit)))
|
|
|
|
if big.Add(chainFee, minerFee).LessThanEqual(maxFee) {
|
|
return
|
|
}
|
|
|
|
// scale chain/miner fee down proportionally to fit in our budget
|
|
// TODO: there are probably smarter things we can do here to optimize
|
|
// message inclusion latency
|
|
|
|
totalFee := big.Add(chainFee, minerFee)
|
|
msg.GasFeeCap = big.Div(big.Mul(chainFee, maxFee), totalFee)
|
|
msg.GasPremium = big.Div(big.Mul(minerFee, maxFee), totalFee)
|
|
}
|
|
|
|
func (a *MpoolAPI) MpoolPushMessage(ctx context.Context, msg *types.Message, maxFee abi.TokenAmount) (*types.SignedMessage, error) {
|
|
{
|
|
fromA, err := a.Stmgr.ResolveToKeyAddress(ctx, msg.From, nil)
|
|
if err != nil {
|
|
return nil, xerrors.Errorf("getting key address: %w", err)
|
|
}
|
|
done, err := a.PushLocks.TakeLock(ctx, fromA)
|
|
if err != nil {
|
|
return nil, xerrors.Errorf("taking lock: %w", err)
|
|
}
|
|
defer done()
|
|
}
|
|
|
|
if msg.Nonce != 0 {
|
|
return nil, xerrors.Errorf("MpoolPushMessage expects message nonce to be 0, was %d", msg.Nonce)
|
|
}
|
|
if msg.GasLimit == 0 {
|
|
gasLimit, err := a.GasEstimateGasLimit(ctx, msg, types.TipSetKey{})
|
|
if err != nil {
|
|
return nil, xerrors.Errorf("estimating gas used: %w", err)
|
|
}
|
|
msg.GasLimit = int64(float64(gasLimit) * a.Mpool.GetConfig().GasLimitOverestimation)
|
|
}
|
|
|
|
if msg.GasPremium == types.EmptyInt || types.BigCmp(msg.GasPremium, types.NewInt(0)) == 0 {
|
|
gasPremium, err := a.GasEstimateGasPremium(ctx, 2, msg.From, msg.GasLimit, types.TipSetKey{})
|
|
if err != nil {
|
|
return nil, xerrors.Errorf("estimating gas price: %w", err)
|
|
}
|
|
msg.GasPremium = gasPremium
|
|
}
|
|
|
|
if msg.GasFeeCap == types.EmptyInt || types.BigCmp(msg.GasFeeCap, types.NewInt(0)) == 0 {
|
|
feeCap, err := a.GasEstimateFeeCap(ctx, msg, 10, types.EmptyTSK)
|
|
if err != nil {
|
|
return nil, xerrors.Errorf("estimating fee cap: %w", err)
|
|
}
|
|
msg.GasFeeCap = feeCap
|
|
}
|
|
|
|
capGasFee(msg, maxFee)
|
|
|
|
return a.Mpool.PushWithNonce(ctx, msg.From, func(from address.Address, nonce uint64) (*types.SignedMessage, error) {
|
|
msg.Nonce = nonce
|
|
if msg.From.Protocol() == address.ID {
|
|
log.Warnf("Push from ID address (%s), adjusting to %s", msg.From, from)
|
|
msg.From = from
|
|
}
|
|
|
|
b, err := a.WalletBalance(ctx, msg.From)
|
|
if err != nil {
|
|
return nil, xerrors.Errorf("mpool push: getting origin balance: %w", err)
|
|
}
|
|
|
|
if b.LessThan(msg.Value) {
|
|
return nil, xerrors.Errorf("mpool push: not enough funds: %s < %s", b, msg.Value)
|
|
}
|
|
|
|
return a.WalletSignMessage(ctx, from, msg)
|
|
})
|
|
}
|
|
|
|
func (a *MpoolAPI) MpoolGetNonce(ctx context.Context, addr address.Address) (uint64, error) {
|
|
return a.Mpool.GetNonce(addr)
|
|
}
|
|
|
|
func (a *MpoolAPI) MpoolSub(ctx context.Context) (<-chan api.MpoolUpdate, error) {
|
|
return a.Mpool.Updates(ctx)
|
|
}
|