368 lines
10 KiB
Go
368 lines
10 KiB
Go
package fakelm
|
|
|
|
import (
|
|
"context"
|
|
"encoding/base64"
|
|
"net/http"
|
|
"net/url"
|
|
|
|
"github.com/BurntSushi/toml"
|
|
"github.com/gbrlsnchs/jwt/v3"
|
|
"github.com/google/uuid"
|
|
"golang.org/x/xerrors"
|
|
|
|
"github.com/filecoin-project/go-address"
|
|
"github.com/filecoin-project/go-jsonrpc/auth"
|
|
"github.com/filecoin-project/go-state-types/abi"
|
|
"github.com/filecoin-project/go-state-types/big"
|
|
"github.com/filecoin-project/go-state-types/network"
|
|
|
|
"github.com/filecoin-project/lotus/api"
|
|
"github.com/filecoin-project/lotus/chain/actors/builtin/miner"
|
|
"github.com/filecoin-project/lotus/chain/types"
|
|
"github.com/filecoin-project/lotus/lib/harmony/harmonydb"
|
|
"github.com/filecoin-project/lotus/node/config"
|
|
"github.com/filecoin-project/lotus/provider/lpmarket"
|
|
"github.com/filecoin-project/lotus/storage/paths"
|
|
sealing "github.com/filecoin-project/lotus/storage/pipeline"
|
|
"github.com/filecoin-project/lotus/storage/sealer/storiface"
|
|
)
|
|
|
|
type LMRPCProvider struct {
|
|
si paths.SectorIndex
|
|
full api.FullNode
|
|
|
|
maddr address.Address // lotus-miner RPC is single-actor
|
|
minerID abi.ActorID
|
|
|
|
ssize abi.SectorSize
|
|
|
|
pi lpmarket.Ingester
|
|
db *harmonydb.DB
|
|
confLayer string
|
|
}
|
|
|
|
func NewLMRPCProvider(si paths.SectorIndex, full api.FullNode, maddr address.Address, minerID abi.ActorID, ssize abi.SectorSize, pi lpmarket.Ingester, db *harmonydb.DB, confLayer string) *LMRPCProvider {
|
|
return &LMRPCProvider{
|
|
si: si,
|
|
full: full,
|
|
maddr: maddr,
|
|
minerID: minerID,
|
|
ssize: ssize,
|
|
pi: pi,
|
|
db: db,
|
|
confLayer: confLayer,
|
|
}
|
|
}
|
|
|
|
func (l *LMRPCProvider) ActorAddress(ctx context.Context) (address.Address, error) {
|
|
return l.maddr, nil
|
|
}
|
|
|
|
func (l *LMRPCProvider) WorkerJobs(ctx context.Context) (map[uuid.UUID][]storiface.WorkerJob, error) {
|
|
// correct enough
|
|
return map[uuid.UUID][]storiface.WorkerJob{}, nil
|
|
}
|
|
|
|
func (l *LMRPCProvider) SectorsStatus(ctx context.Context, sid abi.SectorNumber, showOnChainInfo bool) (api.SectorInfo, error) {
|
|
si, err := l.si.StorageFindSector(ctx, abi.SectorID{Miner: l.minerID, Number: sid}, storiface.FTSealed|storiface.FTCache, 0, false)
|
|
if err != nil {
|
|
return api.SectorInfo{}, err
|
|
}
|
|
|
|
var ssip []struct {
|
|
PieceCID *string `db:"piece_cid"`
|
|
DealID *int64 `db:"f05_deal_id"`
|
|
}
|
|
|
|
err = l.db.Select(ctx, &ssip, "SELECT ssip.piece_cid, ssip.f05_deal_id FROM sectors_sdr_pipeline p LEFT JOIN sectors_sdr_initial_pieces ssip ON p.sp_id = ssip.sp_id AND p.sector_number = ssip.sector_number WHERE p.sp_id = $1 AND p.sector_number = $2", l.minerID, sid)
|
|
if err != nil {
|
|
return api.SectorInfo{}, err
|
|
}
|
|
|
|
var deals []abi.DealID
|
|
if len(ssip) > 0 {
|
|
for _, d := range ssip {
|
|
if d.DealID != nil {
|
|
deals = append(deals, abi.DealID(*d.DealID))
|
|
}
|
|
}
|
|
} else {
|
|
osi, err := l.full.StateSectorGetInfo(ctx, l.maddr, sid, types.EmptyTSK)
|
|
if err != nil {
|
|
return api.SectorInfo{}, err
|
|
}
|
|
|
|
if osi != nil {
|
|
deals = osi.DealIDs
|
|
}
|
|
}
|
|
|
|
spt, err := miner.SealProofTypeFromSectorSize(l.ssize, network.Version20, false) // good enough, just need this for ssize anyways
|
|
if err != nil {
|
|
return api.SectorInfo{}, err
|
|
}
|
|
|
|
if len(si) == 0 {
|
|
state := api.SectorState(sealing.UndefinedSectorState)
|
|
if len(ssip) > 0 {
|
|
state = api.SectorState(sealing.PreCommit1)
|
|
}
|
|
|
|
return api.SectorInfo{
|
|
SectorID: sid,
|
|
State: state,
|
|
CommD: nil,
|
|
CommR: nil,
|
|
Proof: nil,
|
|
Deals: deals,
|
|
Pieces: nil,
|
|
Ticket: api.SealTicket{},
|
|
Seed: api.SealSeed{},
|
|
PreCommitMsg: nil,
|
|
CommitMsg: nil,
|
|
Retries: 0,
|
|
ToUpgrade: false,
|
|
ReplicaUpdateMessage: nil,
|
|
LastErr: "",
|
|
Log: nil,
|
|
SealProof: spt,
|
|
Activation: 0,
|
|
Expiration: 0,
|
|
DealWeight: big.Zero(),
|
|
VerifiedDealWeight: big.Zero(),
|
|
InitialPledge: big.Zero(),
|
|
OnTime: 0,
|
|
Early: 0,
|
|
}, nil
|
|
}
|
|
|
|
var state = api.SectorState(sealing.Proving)
|
|
if !si[0].CanStore {
|
|
state = api.SectorState(sealing.PreCommit2)
|
|
}
|
|
|
|
// todo improve this with on-chain info
|
|
return api.SectorInfo{
|
|
SectorID: sid,
|
|
State: state,
|
|
CommD: nil,
|
|
CommR: nil,
|
|
Proof: nil,
|
|
Deals: deals,
|
|
Pieces: nil,
|
|
Ticket: api.SealTicket{},
|
|
Seed: api.SealSeed{},
|
|
PreCommitMsg: nil,
|
|
CommitMsg: nil,
|
|
Retries: 0,
|
|
ToUpgrade: false,
|
|
ReplicaUpdateMessage: nil,
|
|
LastErr: "",
|
|
Log: nil,
|
|
|
|
SealProof: spt,
|
|
Activation: 0,
|
|
Expiration: 0,
|
|
DealWeight: big.Zero(),
|
|
VerifiedDealWeight: big.Zero(),
|
|
InitialPledge: big.Zero(),
|
|
OnTime: 0,
|
|
Early: 0,
|
|
}, nil
|
|
}
|
|
|
|
func (l *LMRPCProvider) SectorsList(ctx context.Context) ([]abi.SectorNumber, error) {
|
|
decls, err := l.si.StorageList(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var out []abi.SectorNumber
|
|
for _, decl := range decls {
|
|
for _, s := range decl {
|
|
if s.Miner != l.minerID {
|
|
continue
|
|
}
|
|
|
|
out = append(out, s.SectorID.Number)
|
|
}
|
|
}
|
|
|
|
return out, nil
|
|
}
|
|
|
|
type sectorParts struct {
|
|
sealed, unsealed, cache bool
|
|
inStorage bool
|
|
}
|
|
|
|
func (l *LMRPCProvider) SectorsSummary(ctx context.Context) (map[api.SectorState]int, error) {
|
|
decls, err := l.si.StorageList(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
states := map[abi.SectorID]sectorParts{}
|
|
for si, decll := range decls {
|
|
sinfo, err := l.si.StorageInfo(ctx, si)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for _, decl := range decll {
|
|
if decl.Miner != l.minerID {
|
|
continue
|
|
}
|
|
|
|
state := states[abi.SectorID{Miner: decl.Miner, Number: decl.SectorID.Number}]
|
|
state.sealed = state.sealed || decl.Has(storiface.FTSealed)
|
|
state.unsealed = state.unsealed || decl.Has(storiface.FTUnsealed)
|
|
state.cache = state.cache || decl.Has(storiface.FTCache)
|
|
state.inStorage = state.inStorage || sinfo.CanStore
|
|
states[abi.SectorID{Miner: decl.Miner, Number: decl.SectorID.Number}] = state
|
|
}
|
|
}
|
|
|
|
out := map[api.SectorState]int{}
|
|
for _, state := range states {
|
|
switch {
|
|
case state.sealed && state.inStorage:
|
|
out[api.SectorState(sealing.Proving)]++
|
|
default:
|
|
// not even close to correct, but good enough for now
|
|
out[api.SectorState(sealing.PreCommit1)]++
|
|
}
|
|
}
|
|
|
|
return out, nil
|
|
}
|
|
|
|
func (l *LMRPCProvider) SectorsListInStates(ctx context.Context, want []api.SectorState) ([]abi.SectorNumber, error) {
|
|
decls, err := l.si.StorageList(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
wantProving, wantPrecommit1 := false, false
|
|
for _, s := range want {
|
|
switch s {
|
|
case api.SectorState(sealing.Proving):
|
|
wantProving = true
|
|
case api.SectorState(sealing.PreCommit1):
|
|
wantPrecommit1 = true
|
|
}
|
|
}
|
|
|
|
states := map[abi.SectorID]sectorParts{}
|
|
|
|
for si, decll := range decls {
|
|
sinfo, err := l.si.StorageInfo(ctx, si)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for _, decl := range decll {
|
|
if decl.Miner != l.minerID {
|
|
continue
|
|
}
|
|
|
|
state := states[abi.SectorID{Miner: decl.Miner, Number: decl.SectorID.Number}]
|
|
state.sealed = state.sealed || decl.Has(storiface.FTSealed)
|
|
state.unsealed = state.unsealed || decl.Has(storiface.FTUnsealed)
|
|
state.cache = state.cache || decl.Has(storiface.FTCache)
|
|
state.inStorage = state.inStorage || sinfo.CanStore
|
|
states[abi.SectorID{Miner: decl.Miner, Number: decl.SectorID.Number}] = state
|
|
}
|
|
}
|
|
var out []abi.SectorNumber
|
|
|
|
for id, state := range states {
|
|
switch {
|
|
case state.sealed && state.inStorage:
|
|
if wantProving {
|
|
out = append(out, id.Number)
|
|
}
|
|
default:
|
|
// not even close to correct, but good enough for now
|
|
if wantPrecommit1 {
|
|
out = append(out, id.Number)
|
|
}
|
|
}
|
|
}
|
|
|
|
return out, nil
|
|
}
|
|
|
|
func (l *LMRPCProvider) StorageRedeclareLocal(ctx context.Context, id *storiface.ID, b bool) error {
|
|
// so this rescans and redeclares sectors on lotus-miner; whyyy is boost even calling this?
|
|
|
|
return nil
|
|
}
|
|
|
|
func (l *LMRPCProvider) IsUnsealed(ctx context.Context, sectorNum abi.SectorNumber, offset abi.UnpaddedPieceSize, length abi.UnpaddedPieceSize) (bool, error) {
|
|
sectorID := abi.SectorID{Miner: l.minerID, Number: sectorNum}
|
|
|
|
si, err := l.si.StorageFindSector(ctx, sectorID, storiface.FTUnsealed, 0, false)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
// yes, yes, technically sectors can be partially unsealed, but that is never done in practice
|
|
// and can't even be easily done with the current implementation
|
|
return len(si) > 0, nil
|
|
}
|
|
|
|
func (l *LMRPCProvider) ComputeDataCid(ctx context.Context, pieceSize abi.UnpaddedPieceSize, pieceData storiface.Data) (abi.PieceInfo, error) {
|
|
return abi.PieceInfo{}, xerrors.Errorf("not supported")
|
|
}
|
|
|
|
func (l *LMRPCProvider) SectorAddPieceToAny(ctx context.Context, size abi.UnpaddedPieceSize, r storiface.Data, d api.PieceDealInfo) (api.SectorOffset, error) {
|
|
if d.DealProposal.PieceSize != abi.PaddedPieceSize(l.ssize) {
|
|
return api.SectorOffset{}, xerrors.Errorf("only full-sector pieces are supported")
|
|
}
|
|
|
|
return api.SectorOffset{}, xerrors.Errorf("not supported, use AllocatePieceToSector")
|
|
}
|
|
|
|
func (l *LMRPCProvider) AllocatePieceToSector(ctx context.Context, maddr address.Address, piece api.PieceDealInfo, rawSize int64, source url.URL, header http.Header) (api.SectorOffset, error) {
|
|
return l.pi.AllocatePieceToSector(ctx, maddr, piece, rawSize, source, header)
|
|
}
|
|
|
|
func (l *LMRPCProvider) AuthNew(ctx context.Context, perms []auth.Permission) ([]byte, error) {
|
|
var cs []struct {
|
|
Config string
|
|
}
|
|
|
|
err := l.db.Select(ctx, &cs, "select config from harmony_config where title = $1", l.confLayer)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if len(cs) == 0 {
|
|
return nil, xerrors.Errorf("no harmony config found")
|
|
}
|
|
|
|
lp := config.DefaultLotusProvider()
|
|
if _, err := toml.Decode(cs[0].Config, lp); err != nil {
|
|
return nil, xerrors.Errorf("decode harmony config: %w", err)
|
|
}
|
|
|
|
type jwtPayload struct {
|
|
Allow []auth.Permission
|
|
}
|
|
|
|
p := jwtPayload{
|
|
Allow: perms,
|
|
}
|
|
|
|
sk, err := base64.StdEncoding.DecodeString(lp.Apis.StorageRPCSecret)
|
|
if err != nil {
|
|
return nil, xerrors.Errorf("decode secret: %w", err)
|
|
}
|
|
|
|
return jwt.Sign(&p, jwt.NewHS256(sk))
|
|
}
|
|
|
|
var _ MinimalLMApi = &LMRPCProvider{}
|