lotus/storage/addresses.go

88 lines
2.4 KiB
Go
Raw Normal View History

package storage
import (
"context"
"github.com/filecoin-project/go-address"
2020-09-07 03:49:10 +00:00
"github.com/filecoin-project/go-state-types/abi"
"github.com/filecoin-project/go-state-types/big"
"github.com/filecoin-project/lotus/chain/actors/builtin/miner"
"github.com/filecoin-project/lotus/chain/types"
)
type AddrUse int
const (
PreCommitAddr AddrUse = iota
CommitAddr
PoStAddr
)
type addrSelectApi interface {
WalletBalance(context.Context, address.Address) (types.BigInt, error)
WalletHas(context.Context, address.Address) (bool, error)
StateAccountKey(context.Context, address.Address, types.TipSetKey) (address.Address, error)
}
func AddressFor(ctx context.Context, a addrSelectApi, mi miner.MinerInfo, use AddrUse, goodFunds, minFunds abi.TokenAmount) (address.Address, abi.TokenAmount, error) {
switch use {
case PreCommitAddr, CommitAddr:
// always use worker, at least for now
return mi.Worker, big.Zero(), nil
}
2020-10-26 15:28:39 +00:00
leastBad := mi.Worker
bestAvail := minFunds
2020-10-26 15:28:39 +00:00
for _, addr := range append(mi.ControlAddresses, mi.Owner, mi.Worker) {
if maybeUseAddress(ctx, a, addr, goodFunds, &leastBad, &bestAvail) {
return leastBad, bestAvail, nil
}
2020-10-26 15:28:39 +00:00
}
2020-11-19 19:37:00 +00:00
log.Warnw("No address had enough funds to for full PoSt message Fee, selecting least bad address", "address", leastBad, "balance", types.FIL(bestAvail), "optimalFunds", types.FIL(goodFunds), "minFunds", types.FIL(minFunds))
2020-11-19 19:37:00 +00:00
return leastBad, bestAvail, nil
2020-10-26 15:28:39 +00:00
}
func maybeUseAddress(ctx context.Context, a addrSelectApi, addr address.Address, goodFunds abi.TokenAmount, leastBad *address.Address, bestAvail *abi.TokenAmount) bool {
b, err := a.WalletBalance(ctx, addr)
if err != nil {
2020-10-26 15:28:39 +00:00
log.Errorw("checking control address balance", "addr", addr, "error", err)
return false
}
2020-10-26 15:28:39 +00:00
if b.GreaterThanEqual(goodFunds) {
k, err := a.StateAccountKey(ctx, addr, types.EmptyTSK)
if err != nil {
2020-10-26 15:28:39 +00:00
log.Errorw("getting account key", "error", err)
return false
}
2020-10-26 15:28:39 +00:00
have, err := a.WalletHas(ctx, k)
if err != nil {
log.Errorw("failed to check control address", "addr", addr, "error", err)
return false
}
2020-10-26 15:28:39 +00:00
if !have {
log.Errorw("don't have key", "key", k)
return false
}
2020-10-26 15:28:39 +00:00
*leastBad = addr
*bestAvail = b
return true
}
2020-10-26 15:28:39 +00:00
if b.GreaterThan(*bestAvail) {
*leastBad = addr
*bestAvail = b
}
2020-10-26 15:28:39 +00:00
log.Warnw("address didn't have enough funds for window post message", "address", addr, "required", types.FIL(goodFunds), "balance", types.FIL(b))
return false
2020-08-19 23:26:13 +00:00
}