2020-02-13 00:28:23 +00:00
|
|
|
package paychmgr
|
2019-08-09 21:42:56 +00:00
|
|
|
|
|
|
|
import (
|
2019-09-25 12:56:04 +00:00
|
|
|
"bytes"
|
2019-08-09 21:42:56 +00:00
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
|
2020-09-01 14:33:44 +00:00
|
|
|
"github.com/filecoin-project/lotus/api"
|
|
|
|
|
2020-08-11 21:01:16 +00:00
|
|
|
"github.com/filecoin-project/specs-actors/actors/util/adt"
|
|
|
|
|
2020-07-28 23:16:47 +00:00
|
|
|
"github.com/ipfs/go-cid"
|
2019-09-09 13:59:07 +00:00
|
|
|
|
2019-12-19 20:13:17 +00:00
|
|
|
"github.com/filecoin-project/go-address"
|
2020-07-28 23:16:47 +00:00
|
|
|
cborutil "github.com/filecoin-project/go-cbor-util"
|
2019-10-18 04:47:41 +00:00
|
|
|
"github.com/filecoin-project/lotus/chain/actors"
|
|
|
|
"github.com/filecoin-project/lotus/chain/types"
|
2020-01-30 23:48:25 +00:00
|
|
|
"github.com/filecoin-project/lotus/lib/sigs"
|
2020-07-28 23:16:47 +00:00
|
|
|
"github.com/filecoin-project/specs-actors/actors/abi/big"
|
|
|
|
"github.com/filecoin-project/specs-actors/actors/builtin"
|
|
|
|
"github.com/filecoin-project/specs-actors/actors/builtin/account"
|
|
|
|
"github.com/filecoin-project/specs-actors/actors/builtin/paych"
|
|
|
|
xerrors "golang.org/x/xerrors"
|
2019-08-09 21:42:56 +00:00
|
|
|
)
|
|
|
|
|
2020-09-01 14:33:44 +00:00
|
|
|
// insufficientFundsErr indicates that there are not enough funds in the
|
|
|
|
// channel to create a voucher
|
|
|
|
type insufficientFundsErr interface {
|
|
|
|
Shortfall() types.BigInt
|
|
|
|
}
|
|
|
|
|
|
|
|
type ErrInsufficientFunds struct {
|
|
|
|
shortfall types.BigInt
|
|
|
|
}
|
|
|
|
|
|
|
|
func newErrInsufficientFunds(shortfall types.BigInt) *ErrInsufficientFunds {
|
|
|
|
return &ErrInsufficientFunds{shortfall: shortfall}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *ErrInsufficientFunds) Error() string {
|
|
|
|
return fmt.Sprintf("not enough funds in channel to cover voucher - shortfall: %d", e.shortfall)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *ErrInsufficientFunds) Shortfall() types.BigInt {
|
|
|
|
return e.shortfall
|
|
|
|
}
|
|
|
|
|
2020-07-28 23:16:47 +00:00
|
|
|
// channelAccessor is used to simplify locking when accessing a channel
|
|
|
|
type channelAccessor struct {
|
2020-09-01 14:33:44 +00:00
|
|
|
from address.Address
|
|
|
|
to address.Address
|
|
|
|
|
|
|
|
// chctx is used by background processes (eg when waiting for things to be
|
|
|
|
// confirmed on chain)
|
|
|
|
chctx context.Context
|
2020-07-28 23:16:47 +00:00
|
|
|
sa *stateAccessor
|
2020-08-11 14:20:05 +00:00
|
|
|
api managerAPI
|
2020-07-28 23:16:47 +00:00
|
|
|
store *Store
|
|
|
|
lk *channelLock
|
|
|
|
fundsReqQueue []*fundsReq
|
|
|
|
msgListeners msgListeners
|
2019-09-16 16:40:26 +00:00
|
|
|
}
|
|
|
|
|
2020-09-01 14:33:44 +00:00
|
|
|
func newChannelAccessor(pm *Manager, from address.Address, to address.Address) *channelAccessor {
|
2020-07-28 23:16:47 +00:00
|
|
|
return &channelAccessor{
|
2020-09-01 14:33:44 +00:00
|
|
|
from: from,
|
|
|
|
to: to,
|
|
|
|
chctx: pm.ctx,
|
2020-08-11 14:20:05 +00:00
|
|
|
sa: pm.sa,
|
2020-08-10 21:52:59 +00:00
|
|
|
api: pm.pchapi,
|
|
|
|
store: pm.store,
|
2020-09-01 14:33:44 +00:00
|
|
|
lk: &channelLock{globalLock: &pm.lk},
|
2020-08-10 21:52:59 +00:00
|
|
|
msgListeners: newMsgListeners(),
|
2019-08-09 21:42:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-28 23:16:47 +00:00
|
|
|
func (ca *channelAccessor) getChannelInfo(addr address.Address) (*ChannelInfo, error) {
|
|
|
|
ca.lk.Lock()
|
|
|
|
defer ca.lk.Unlock()
|
2019-09-10 13:43:01 +00:00
|
|
|
|
2020-07-28 23:16:47 +00:00
|
|
|
return ca.store.ByAddress(addr)
|
2019-09-16 17:23:48 +00:00
|
|
|
}
|
|
|
|
|
2020-09-01 14:33:44 +00:00
|
|
|
// createVoucher creates a voucher with the given specification, setting its
|
|
|
|
// nonce, signing the voucher and storing it in the local datastore.
|
|
|
|
// If there are not enough funds in the channel to create the voucher, returns
|
|
|
|
// the shortfall in funds.
|
|
|
|
func (ca *channelAccessor) createVoucher(ctx context.Context, ch address.Address, voucher paych.SignedVoucher) (*api.VoucherCreateResult, error) {
|
|
|
|
ca.lk.Lock()
|
|
|
|
defer ca.lk.Unlock()
|
|
|
|
|
|
|
|
// Find the channel for the voucher
|
|
|
|
ci, err := ca.store.ByAddress(ch)
|
|
|
|
if err != nil {
|
|
|
|
return nil, xerrors.Errorf("failed to get channel info by address: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the voucher channel
|
|
|
|
sv := &voucher
|
|
|
|
sv.ChannelAddr = ch
|
|
|
|
|
|
|
|
// Get the next nonce on the given lane
|
|
|
|
sv.Nonce = ca.nextNonceForLane(ci, voucher.Lane)
|
|
|
|
|
|
|
|
// Sign the voucher
|
|
|
|
vb, err := sv.SigningBytes()
|
|
|
|
if err != nil {
|
|
|
|
return nil, xerrors.Errorf("failed to get voucher signing bytes: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
sig, err := ca.api.WalletSign(ctx, ci.Control, vb)
|
|
|
|
if err != nil {
|
|
|
|
return nil, xerrors.Errorf("failed to sign voucher: %w", err)
|
|
|
|
}
|
|
|
|
sv.Signature = sig
|
|
|
|
|
|
|
|
// Store the voucher
|
|
|
|
if _, err := ca.addVoucherUnlocked(ctx, ch, sv, nil, types.NewInt(0)); err != nil {
|
|
|
|
// If there are not enough funds in the channel to cover the voucher,
|
|
|
|
// return a voucher create result with the shortfall
|
|
|
|
var ife insufficientFundsErr
|
|
|
|
if xerrors.As(err, &ife) {
|
|
|
|
return &api.VoucherCreateResult{
|
|
|
|
Shortfall: ife.Shortfall(),
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, xerrors.Errorf("failed to persist voucher: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &api.VoucherCreateResult{Voucher: sv, Shortfall: types.NewInt(0)}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ca *channelAccessor) nextNonceForLane(ci *ChannelInfo, lane uint64) uint64 {
|
|
|
|
var maxnonce uint64
|
|
|
|
for _, v := range ci.Vouchers {
|
|
|
|
if v.Voucher.Lane == lane {
|
|
|
|
if v.Voucher.Nonce > maxnonce {
|
|
|
|
maxnonce = v.Voucher.Nonce
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return maxnonce + 1
|
|
|
|
}
|
|
|
|
|
2020-07-28 23:16:47 +00:00
|
|
|
func (ca *channelAccessor) checkVoucherValid(ctx context.Context, ch address.Address, sv *paych.SignedVoucher) (map[uint64]*paych.LaneState, error) {
|
|
|
|
ca.lk.Lock()
|
|
|
|
defer ca.lk.Unlock()
|
2019-09-16 17:23:48 +00:00
|
|
|
|
2020-07-28 23:16:47 +00:00
|
|
|
return ca.checkVoucherValidUnlocked(ctx, ch, sv)
|
2019-08-12 17:09:56 +00:00
|
|
|
}
|
|
|
|
|
2020-07-28 23:16:47 +00:00
|
|
|
func (ca *channelAccessor) checkVoucherValidUnlocked(ctx context.Context, ch address.Address, sv *paych.SignedVoucher) (map[uint64]*paych.LaneState, error) {
|
2020-07-15 09:12:03 +00:00
|
|
|
if sv.ChannelAddr != ch {
|
|
|
|
return nil, xerrors.Errorf("voucher ChannelAddr doesn't match channel address, got %s, expected %s", sv.ChannelAddr, ch)
|
|
|
|
}
|
|
|
|
|
2020-08-17 15:13:13 +00:00
|
|
|
// Load payment channel actor state
|
|
|
|
act, pchState, err := ca.sa.loadPaychActorState(ctx, ch)
|
2019-08-09 21:42:56 +00:00
|
|
|
if err != nil {
|
2020-07-09 22:27:39 +00:00
|
|
|
return nil, err
|
2019-08-09 21:42:56 +00:00
|
|
|
}
|
|
|
|
|
2020-08-17 15:13:13 +00:00
|
|
|
// Load channel "From" account actor state
|
2020-07-28 23:16:47 +00:00
|
|
|
var actState account.State
|
2020-08-11 14:20:05 +00:00
|
|
|
_, err = ca.api.LoadActorState(ctx, pchState.From, &actState, nil)
|
2020-03-03 02:12:36 +00:00
|
|
|
if err != nil {
|
2020-07-09 22:27:39 +00:00
|
|
|
return nil, err
|
2020-03-03 02:12:36 +00:00
|
|
|
}
|
2020-07-28 23:16:47 +00:00
|
|
|
from := actState.Address
|
2020-03-03 02:12:36 +00:00
|
|
|
|
2020-08-17 15:13:13 +00:00
|
|
|
// verify voucher signature
|
2019-08-09 21:42:56 +00:00
|
|
|
vb, err := sv.SigningBytes()
|
|
|
|
if err != nil {
|
2020-07-09 22:27:39 +00:00
|
|
|
return nil, err
|
2019-08-09 21:42:56 +00:00
|
|
|
}
|
2019-08-13 18:47:40 +00:00
|
|
|
|
|
|
|
// TODO: technically, either party may create and sign a voucher.
|
|
|
|
// However, for now, we only accept them from the channel creator.
|
|
|
|
// More complex handling logic can be added later
|
2020-03-03 02:12:36 +00:00
|
|
|
if err := sigs.Verify(sv.Signature, from, vb); err != nil {
|
2020-07-09 22:27:39 +00:00
|
|
|
return nil, err
|
2019-08-09 21:42:56 +00:00
|
|
|
}
|
|
|
|
|
2020-07-09 22:27:39 +00:00
|
|
|
// Check the voucher against the highest known voucher nonce / value
|
2020-08-11 21:01:16 +00:00
|
|
|
laneStates, err := ca.laneState(ctx, pchState, ch)
|
2020-07-09 22:27:39 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-07-10 18:06:52 +00:00
|
|
|
|
|
|
|
// If the new voucher nonce value is less than the highest known
|
|
|
|
// nonce for the lane
|
|
|
|
ls, lsExists := laneStates[sv.Lane]
|
|
|
|
if lsExists && sv.Nonce <= ls.Nonce {
|
2020-07-09 22:27:39 +00:00
|
|
|
return nil, fmt.Errorf("nonce too low")
|
|
|
|
}
|
2020-07-10 18:06:52 +00:00
|
|
|
|
2020-07-09 22:27:39 +00:00
|
|
|
// If the voucher amount is less than the highest known voucher amount
|
2020-07-10 18:06:52 +00:00
|
|
|
if lsExists && sv.Amount.LessThanEqual(ls.Redeemed) {
|
2020-07-09 22:27:39 +00:00
|
|
|
return nil, fmt.Errorf("voucher amount is lower than amount for voucher with lower nonce")
|
2019-08-09 21:42:56 +00:00
|
|
|
}
|
|
|
|
|
2020-07-10 18:06:52 +00:00
|
|
|
// Total redeemed is the total redeemed amount for all lanes, including
|
|
|
|
// the new voucher
|
|
|
|
// eg
|
|
|
|
//
|
|
|
|
// lane 1 redeemed: 3
|
|
|
|
// lane 2 redeemed: 2
|
|
|
|
// voucher for lane 1: 5
|
|
|
|
//
|
|
|
|
// Voucher supersedes lane 1 redeemed, therefore
|
|
|
|
// effective lane 1 redeemed: 5
|
|
|
|
//
|
|
|
|
// lane 1: 5
|
|
|
|
// lane 2: 2
|
|
|
|
// -
|
|
|
|
// total: 7
|
2020-07-28 23:16:47 +00:00
|
|
|
totalRedeemed, err := ca.totalRedeemedWithVoucher(laneStates, sv)
|
2020-07-10 18:06:52 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-07-09 22:27:39 +00:00
|
|
|
|
2020-07-10 18:06:52 +00:00
|
|
|
// Total required balance = total redeemed + toSend
|
|
|
|
// Must not exceed actor balance
|
|
|
|
newTotal := types.BigAdd(totalRedeemed, pchState.ToSend)
|
2019-09-23 17:11:44 +00:00
|
|
|
if act.Balance.LessThan(newTotal) {
|
2020-09-01 14:33:44 +00:00
|
|
|
return nil, newErrInsufficientFunds(types.BigSub(newTotal, act.Balance))
|
2019-08-09 21:42:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(sv.Merges) != 0 {
|
2020-07-09 22:27:39 +00:00
|
|
|
return nil, fmt.Errorf("dont currently support paych lane merges")
|
2019-08-09 21:42:56 +00:00
|
|
|
}
|
|
|
|
|
2020-07-10 18:06:52 +00:00
|
|
|
return laneStates, nil
|
2019-08-09 21:42:56 +00:00
|
|
|
}
|
|
|
|
|
2020-07-28 23:16:47 +00:00
|
|
|
func (ca *channelAccessor) checkVoucherSpendable(ctx context.Context, ch address.Address, sv *paych.SignedVoucher, secret []byte, proof []byte) (bool, error) {
|
|
|
|
ca.lk.Lock()
|
|
|
|
defer ca.lk.Unlock()
|
|
|
|
|
|
|
|
recipient, err := ca.getPaychRecipient(ctx, ch)
|
2019-08-09 21:42:56 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
2020-08-20 16:09:52 +00:00
|
|
|
ci, err := ca.store.ByAddress(ch)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if voucher has already been submitted
|
|
|
|
submitted, err := ci.wasVoucherSubmitted(sv)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
if submitted {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// If proof is needed and wasn't supplied as a parameter, get it from the
|
|
|
|
// datastore
|
2019-09-09 13:59:07 +00:00
|
|
|
if sv.Extra != nil && proof == nil {
|
2020-08-20 16:09:52 +00:00
|
|
|
vi, err := ci.infoForVoucher(sv)
|
2019-09-09 13:59:07 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
2020-08-20 16:09:52 +00:00
|
|
|
if vi.Proof != nil {
|
|
|
|
log.Info("CheckVoucherSpendable: using stored proof")
|
|
|
|
proof = vi.Proof
|
|
|
|
} else {
|
2019-09-09 13:59:07 +00:00
|
|
|
log.Warn("CheckVoucherSpendable: nil proof for voucher with validation")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-12 23:52:19 +00:00
|
|
|
enc, err := actors.SerializeParams(&paych.UpdateChannelStateParams{
|
2019-08-09 21:42:56 +00:00
|
|
|
Sv: *sv,
|
|
|
|
Secret: secret,
|
|
|
|
Proof: proof,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
2020-08-11 14:20:05 +00:00
|
|
|
ret, err := ca.api.Call(ctx, &types.Message{
|
2020-07-22 17:55:31 +00:00
|
|
|
From: recipient,
|
2019-08-09 21:42:56 +00:00
|
|
|
To: ch,
|
2020-02-12 23:52:19 +00:00
|
|
|
Method: builtin.MethodsPaych.UpdateChannelState,
|
2019-08-09 21:42:56 +00:00
|
|
|
Params: enc,
|
|
|
|
}, nil)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
2020-03-03 23:32:17 +00:00
|
|
|
if ret.MsgRct.ExitCode != 0 {
|
2019-08-09 21:42:56 +00:00
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
|
2020-07-28 23:16:47 +00:00
|
|
|
func (ca *channelAccessor) getPaychRecipient(ctx context.Context, ch address.Address) (address.Address, error) {
|
2020-02-12 23:52:19 +00:00
|
|
|
var state paych.State
|
2020-08-11 14:20:05 +00:00
|
|
|
if _, err := ca.api.LoadActorState(ctx, ch, &state, nil); err != nil {
|
2020-02-12 23:52:19 +00:00
|
|
|
return address.Address{}, err
|
2019-08-09 21:42:56 +00:00
|
|
|
}
|
|
|
|
|
2020-07-22 17:55:31 +00:00
|
|
|
return state.To, nil
|
2019-08-09 21:42:56 +00:00
|
|
|
}
|
2019-08-12 17:09:56 +00:00
|
|
|
|
2020-07-28 23:16:47 +00:00
|
|
|
func (ca *channelAccessor) addVoucher(ctx context.Context, ch address.Address, sv *paych.SignedVoucher, proof []byte, minDelta types.BigInt) (types.BigInt, error) {
|
|
|
|
ca.lk.Lock()
|
|
|
|
defer ca.lk.Unlock()
|
2019-09-25 12:56:04 +00:00
|
|
|
|
2020-09-01 14:33:44 +00:00
|
|
|
return ca.addVoucherUnlocked(ctx, ch, sv, proof, minDelta)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ca *channelAccessor) addVoucherUnlocked(ctx context.Context, ch address.Address, sv *paych.SignedVoucher, proof []byte, minDelta types.BigInt) (types.BigInt, error) {
|
2020-07-28 23:16:47 +00:00
|
|
|
ci, err := ca.store.ByAddress(ch)
|
2019-09-25 12:56:04 +00:00
|
|
|
if err != nil {
|
2020-07-28 23:16:47 +00:00
|
|
|
return types.BigInt{}, err
|
2019-09-25 12:56:04 +00:00
|
|
|
}
|
|
|
|
|
2020-07-09 22:27:39 +00:00
|
|
|
// Check if the voucher has already been added
|
2019-09-25 12:56:04 +00:00
|
|
|
for i, v := range ci.Vouchers {
|
2020-03-03 02:12:36 +00:00
|
|
|
eq, err := cborutil.Equals(sv, v.Voucher)
|
2020-02-12 23:52:19 +00:00
|
|
|
if err != nil {
|
|
|
|
return types.BigInt{}, err
|
|
|
|
}
|
|
|
|
if !eq {
|
2019-09-25 12:56:04 +00:00
|
|
|
continue
|
|
|
|
}
|
2020-07-09 22:27:39 +00:00
|
|
|
|
|
|
|
// This is a duplicate voucher.
|
|
|
|
// Update the proof on the existing voucher
|
|
|
|
if len(proof) > 0 && !bytes.Equal(v.Proof, proof) {
|
|
|
|
log.Warnf("AddVoucher: adding proof to stored voucher")
|
|
|
|
ci.Vouchers[i] = &VoucherInfo{
|
|
|
|
Voucher: v.Voucher,
|
|
|
|
Proof: proof,
|
2019-09-25 12:56:04 +00:00
|
|
|
}
|
|
|
|
|
2020-07-28 23:16:47 +00:00
|
|
|
return types.NewInt(0), ca.store.putChannelInfo(ci)
|
2019-09-25 12:56:04 +00:00
|
|
|
}
|
|
|
|
|
2020-07-09 22:27:39 +00:00
|
|
|
// Otherwise just ignore the duplicate voucher
|
|
|
|
log.Warnf("AddVoucher: voucher re-added with matching proof")
|
|
|
|
return types.NewInt(0), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check voucher validity
|
2020-07-28 23:16:47 +00:00
|
|
|
laneStates, err := ca.checkVoucherValidUnlocked(ctx, ch, sv)
|
2020-07-09 22:27:39 +00:00
|
|
|
if err != nil {
|
|
|
|
return types.NewInt(0), err
|
|
|
|
}
|
|
|
|
|
|
|
|
// The change in value is the delta between the voucher amount and
|
2020-07-10 18:06:52 +00:00
|
|
|
// the highest previous voucher amount for the lane
|
|
|
|
laneState, exists := laneStates[sv.Lane]
|
|
|
|
redeemed := big.NewInt(0)
|
|
|
|
if exists {
|
|
|
|
redeemed = laneState.Redeemed
|
2019-09-25 12:56:04 +00:00
|
|
|
}
|
|
|
|
|
2020-07-10 18:06:52 +00:00
|
|
|
delta := types.BigSub(sv.Amount, redeemed)
|
2019-09-25 12:56:04 +00:00
|
|
|
if minDelta.GreaterThan(delta) {
|
2020-07-10 18:06:52 +00:00
|
|
|
return delta, xerrors.Errorf("addVoucher: supplied token amount too low; minD=%s, D=%s; laneAmt=%s; v.Amt=%s", minDelta, delta, redeemed, sv.Amount)
|
2019-09-25 12:56:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ci.Vouchers = append(ci.Vouchers, &VoucherInfo{
|
|
|
|
Voucher: sv,
|
|
|
|
Proof: proof,
|
|
|
|
})
|
|
|
|
|
2020-07-09 21:20:17 +00:00
|
|
|
if ci.NextLane <= sv.Lane {
|
2020-02-21 17:26:44 +00:00
|
|
|
ci.NextLane = sv.Lane + 1
|
2019-09-25 12:56:04 +00:00
|
|
|
}
|
|
|
|
|
2020-07-28 23:16:47 +00:00
|
|
|
return delta, ca.store.putChannelInfo(ci)
|
2019-08-12 17:09:56 +00:00
|
|
|
}
|
2019-08-12 19:51:01 +00:00
|
|
|
|
2020-08-20 16:09:52 +00:00
|
|
|
func (ca *channelAccessor) submitVoucher(ctx context.Context, ch address.Address, sv *paych.SignedVoucher, secret []byte, proof []byte) (cid.Cid, error) {
|
|
|
|
ca.lk.Lock()
|
|
|
|
defer ca.lk.Unlock()
|
|
|
|
|
|
|
|
ci, err := ca.store.ByAddress(ch)
|
|
|
|
if err != nil {
|
|
|
|
return cid.Undef, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// If voucher needs proof, and none was supplied, check datastore for proof
|
|
|
|
if sv.Extra != nil && proof == nil {
|
|
|
|
vi, err := ci.infoForVoucher(sv)
|
|
|
|
if err != nil {
|
|
|
|
return cid.Undef, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if vi.Proof != nil {
|
|
|
|
log.Info("SubmitVoucher: using stored proof")
|
|
|
|
proof = vi.Proof
|
|
|
|
} else {
|
|
|
|
log.Warn("SubmitVoucher: nil proof for voucher with validation")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
has, err := ci.hasVoucher(sv)
|
|
|
|
if err != nil {
|
|
|
|
return cid.Undef, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the channel has the voucher
|
|
|
|
if has {
|
|
|
|
// Check that the voucher hasn't already been submitted
|
|
|
|
submitted, err := ci.wasVoucherSubmitted(sv)
|
|
|
|
if err != nil {
|
|
|
|
return cid.Undef, err
|
|
|
|
}
|
|
|
|
if submitted {
|
|
|
|
return cid.Undef, xerrors.Errorf("cannot submit voucher that has already been submitted")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
enc, err := actors.SerializeParams(&paych.UpdateChannelStateParams{
|
|
|
|
Sv: *sv,
|
|
|
|
Secret: secret,
|
|
|
|
Proof: proof,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return cid.Undef, err
|
|
|
|
}
|
|
|
|
|
|
|
|
msg := &types.Message{
|
|
|
|
From: ci.Control,
|
|
|
|
To: ch,
|
|
|
|
Value: types.NewInt(0),
|
|
|
|
Method: builtin.MethodsPaych.UpdateChannelState,
|
|
|
|
Params: enc,
|
|
|
|
}
|
|
|
|
|
|
|
|
smsg, err := ca.api.MpoolPushMessage(ctx, msg, nil)
|
|
|
|
if err != nil {
|
|
|
|
return cid.Undef, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the channel didn't already have the voucher
|
|
|
|
if !has {
|
|
|
|
// Add the voucher to the channel
|
|
|
|
ci.Vouchers = append(ci.Vouchers, &VoucherInfo{
|
|
|
|
Voucher: sv,
|
|
|
|
Proof: proof,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mark the voucher and any lower-nonce vouchers as having been submitted
|
|
|
|
err = ca.store.MarkVoucherSubmitted(ci, sv)
|
|
|
|
if err != nil {
|
|
|
|
return cid.Undef, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return smsg.Cid(), nil
|
|
|
|
}
|
|
|
|
|
2020-07-28 23:16:47 +00:00
|
|
|
func (ca *channelAccessor) allocateLane(ch address.Address) (uint64, error) {
|
|
|
|
ca.lk.Lock()
|
|
|
|
defer ca.lk.Unlock()
|
|
|
|
|
2020-07-09 20:35:43 +00:00
|
|
|
// TODO: should this take into account lane state?
|
2020-07-28 23:16:47 +00:00
|
|
|
return ca.store.AllocateLane(ch)
|
2019-09-10 13:43:01 +00:00
|
|
|
}
|
|
|
|
|
2020-07-28 23:16:47 +00:00
|
|
|
func (ca *channelAccessor) listVouchers(ctx context.Context, ch address.Address) ([]*VoucherInfo, error) {
|
|
|
|
ca.lk.Lock()
|
|
|
|
defer ca.lk.Unlock()
|
|
|
|
|
2019-08-12 19:51:01 +00:00
|
|
|
// TODO: just having a passthrough method like this feels odd. Seems like
|
|
|
|
// there should be some filtering we're doing here
|
2020-07-28 23:16:47 +00:00
|
|
|
return ca.store.VouchersForPaych(ch)
|
2019-08-12 19:51:01 +00:00
|
|
|
}
|
|
|
|
|
2020-07-28 23:16:47 +00:00
|
|
|
// laneState gets the LaneStates from chain, then applies all vouchers in
|
|
|
|
// the data store over the chain state
|
2020-08-11 21:01:16 +00:00
|
|
|
func (ca *channelAccessor) laneState(ctx context.Context, state *paych.State, ch address.Address) (map[uint64]*paych.LaneState, error) {
|
2020-07-28 23:16:47 +00:00
|
|
|
// TODO: we probably want to call UpdateChannelState with all vouchers to be fully correct
|
|
|
|
// (but technically dont't need to)
|
|
|
|
|
|
|
|
// Get the lane state from the chain
|
2020-08-11 21:01:16 +00:00
|
|
|
store := ca.api.AdtStore(ctx)
|
|
|
|
lsamt, err := adt.AsArray(store, state.LaneStates)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2020-07-28 23:16:47 +00:00
|
|
|
}
|
|
|
|
|
2020-08-11 21:01:16 +00:00
|
|
|
// Note: we use a map instead of an array to store laneStates because the
|
|
|
|
// client sets the lane ID (the index) and potentially they could use a
|
|
|
|
// very large index.
|
|
|
|
var ls paych.LaneState
|
|
|
|
laneStates := make(map[uint64]*paych.LaneState, lsamt.Length())
|
2020-08-13 00:27:04 +00:00
|
|
|
err = lsamt.ForEach(&ls, func(i int64) error {
|
2020-08-11 21:01:16 +00:00
|
|
|
current := ls
|
2020-08-13 00:27:04 +00:00
|
|
|
laneStates[uint64(i)] = ¤t
|
2020-08-11 21:01:16 +00:00
|
|
|
return nil
|
|
|
|
})
|
2020-08-13 00:27:04 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-08-11 21:01:16 +00:00
|
|
|
|
2020-07-28 23:16:47 +00:00
|
|
|
// Apply locally stored vouchers
|
|
|
|
vouchers, err := ca.store.VouchersForPaych(ch)
|
|
|
|
if err != nil && err != ErrChannelNotTracked {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, v := range vouchers {
|
|
|
|
for range v.Voucher.Merges {
|
|
|
|
return nil, xerrors.Errorf("paych merges not handled yet")
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there's a voucher for a lane that isn't in chain state just
|
|
|
|
// create it
|
|
|
|
ls, ok := laneStates[v.Voucher.Lane]
|
|
|
|
if !ok {
|
|
|
|
ls = &paych.LaneState{
|
|
|
|
Redeemed: types.NewInt(0),
|
|
|
|
Nonce: 0,
|
|
|
|
}
|
|
|
|
laneStates[v.Voucher.Lane] = ls
|
|
|
|
}
|
|
|
|
|
|
|
|
if v.Voucher.Nonce < ls.Nonce {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
ls.Nonce = v.Voucher.Nonce
|
|
|
|
ls.Redeemed = v.Voucher.Amount
|
|
|
|
}
|
|
|
|
|
|
|
|
return laneStates, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the total redeemed amount across all lanes, after applying the voucher
|
|
|
|
func (ca *channelAccessor) totalRedeemedWithVoucher(laneStates map[uint64]*paych.LaneState, sv *paych.SignedVoucher) (big.Int, error) {
|
|
|
|
// TODO: merges
|
|
|
|
if len(sv.Merges) != 0 {
|
|
|
|
return big.Int{}, xerrors.Errorf("dont currently support paych lane merges")
|
|
|
|
}
|
|
|
|
|
|
|
|
total := big.NewInt(0)
|
|
|
|
for _, ls := range laneStates {
|
|
|
|
total = big.Add(total, ls.Redeemed)
|
|
|
|
}
|
|
|
|
|
|
|
|
lane, ok := laneStates[sv.Lane]
|
|
|
|
if ok {
|
|
|
|
// If the voucher is for an existing lane, and the voucher nonce
|
2020-09-01 14:33:44 +00:00
|
|
|
// is higher than the lane nonce
|
2020-07-28 23:16:47 +00:00
|
|
|
if sv.Nonce > lane.Nonce {
|
|
|
|
// Add the delta between the redeemed amount and the voucher
|
|
|
|
// amount to the total
|
|
|
|
delta := big.Sub(sv.Amount, lane.Redeemed)
|
|
|
|
total = big.Add(total, delta)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// If the voucher is *not* for an existing lane, just add its
|
|
|
|
// value (implicitly a new lane will be created for the voucher)
|
|
|
|
total = big.Add(total, sv.Amount)
|
|
|
|
}
|
|
|
|
|
|
|
|
return total, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ca *channelAccessor) settle(ctx context.Context, ch address.Address) (cid.Cid, error) {
|
|
|
|
ca.lk.Lock()
|
|
|
|
defer ca.lk.Unlock()
|
|
|
|
|
|
|
|
ci, err := ca.store.ByAddress(ch)
|
|
|
|
if err != nil {
|
|
|
|
return cid.Undef, err
|
|
|
|
}
|
|
|
|
|
|
|
|
msg := &types.Message{
|
|
|
|
To: ch,
|
|
|
|
From: ci.Control,
|
|
|
|
Value: types.NewInt(0),
|
|
|
|
Method: builtin.MethodsPaych.Settle,
|
|
|
|
}
|
2020-08-12 20:17:21 +00:00
|
|
|
smgs, err := ca.api.MpoolPushMessage(ctx, msg, nil)
|
2020-07-28 23:16:47 +00:00
|
|
|
if err != nil {
|
|
|
|
return cid.Undef, err
|
|
|
|
}
|
|
|
|
|
|
|
|
ci.Settling = true
|
|
|
|
err = ca.store.putChannelInfo(ci)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("Error marking channel as settled: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return smgs.Cid(), err
|
|
|
|
}
|
2020-08-10 17:21:25 +00:00
|
|
|
|
|
|
|
func (ca *channelAccessor) collect(ctx context.Context, ch address.Address) (cid.Cid, error) {
|
|
|
|
ca.lk.Lock()
|
|
|
|
defer ca.lk.Unlock()
|
|
|
|
|
|
|
|
ci, err := ca.store.ByAddress(ch)
|
|
|
|
if err != nil {
|
|
|
|
return cid.Undef, err
|
|
|
|
}
|
|
|
|
|
|
|
|
msg := &types.Message{
|
|
|
|
To: ch,
|
|
|
|
From: ci.Control,
|
|
|
|
Value: types.NewInt(0),
|
|
|
|
Method: builtin.MethodsPaych.Collect,
|
|
|
|
}
|
|
|
|
|
2020-08-12 20:17:21 +00:00
|
|
|
smsg, err := ca.api.MpoolPushMessage(ctx, msg, nil)
|
2020-08-10 17:21:25 +00:00
|
|
|
if err != nil {
|
|
|
|
return cid.Undef, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return smsg.Cid(), nil
|
|
|
|
}
|