mirror of
https://github.com/grassrootseconomics/cic-custodial.git
synced 2025-01-10 16:47:33 +01:00
197 lines
4.5 KiB
Go
197 lines
4.5 KiB
Go
|
package task
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"encoding/json"
|
||
|
"math/big"
|
||
|
|
||
|
"github.com/bsm/redislock"
|
||
|
"github.com/celo-org/celo-blockchain/common/hexutil"
|
||
|
"github.com/grassrootseconomics/celoutils"
|
||
|
"github.com/grassrootseconomics/cic-custodial/internal/custodial"
|
||
|
"github.com/grassrootseconomics/cic-custodial/internal/store"
|
||
|
"github.com/grassrootseconomics/cic-custodial/internal/tasker"
|
||
|
"github.com/grassrootseconomics/cic-custodial/pkg/enum"
|
||
|
"github.com/grassrootseconomics/w3-celo-patch/module/eth"
|
||
|
"github.com/hibiken/asynq"
|
||
|
)
|
||
|
|
||
|
type TransferAuthPayload struct {
|
||
|
Amount uint64 `json:"amount"`
|
||
|
Authorizer string `json:"authorizer"`
|
||
|
AuthorizedAddress string `json:"authorizedAddress"`
|
||
|
TrackingId string `json:"trackingId"`
|
||
|
VoucherAddress string `json:"voucherAddress"`
|
||
|
}
|
||
|
|
||
|
func SignTransferAuthorizationProcessor(cu *custodial.Custodial) func(context.Context, *asynq.Task) error {
|
||
|
return func(ctx context.Context, t *asynq.Task) error {
|
||
|
var (
|
||
|
err error
|
||
|
networkBalance big.Int
|
||
|
payload TransferAuthPayload
|
||
|
)
|
||
|
|
||
|
if err := json.Unmarshal(t.Payload(), &payload); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
lock, err := cu.LockProvider.Obtain(
|
||
|
ctx,
|
||
|
lockPrefix+payload.Authorizer,
|
||
|
lockTimeout,
|
||
|
&redislock.Options{
|
||
|
RetryStrategy: lockRetry(),
|
||
|
},
|
||
|
)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
defer lock.Release(ctx)
|
||
|
|
||
|
key, err := cu.Store.LoadPrivateKey(ctx, payload.Authorizer)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
nonce, err := cu.Noncestore.Acquire(ctx, payload.Authorizer)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
defer func() {
|
||
|
if err != nil {
|
||
|
if nErr := cu.Noncestore.Return(ctx, payload.Authorizer); nErr != nil {
|
||
|
err = nErr
|
||
|
}
|
||
|
}
|
||
|
}()
|
||
|
|
||
|
input, err := cu.Abis[custodial.Approve].EncodeArgs(
|
||
|
celoutils.HexToAddress(payload.AuthorizedAddress),
|
||
|
new(big.Int).SetUint64(payload.Amount),
|
||
|
)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
builtTx, err := cu.CeloProvider.SignContractExecutionTx(
|
||
|
key,
|
||
|
celoutils.ContractExecutionTxOpts{
|
||
|
ContractAddress: celoutils.HexToAddress(payload.VoucherAddress),
|
||
|
InputData: input,
|
||
|
GasFeeCap: celoutils.SafeGasFeeCap,
|
||
|
GasTipCap: celoutils.SafeGasTipCap,
|
||
|
GasLimit: uint64(celoutils.SafeGasLimit),
|
||
|
Nonce: nonce,
|
||
|
},
|
||
|
)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
rawTx, err := builtTx.MarshalBinary()
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
id, err := cu.Store.CreateOtx(ctx, store.Otx{
|
||
|
TrackingId: payload.TrackingId,
|
||
|
Type: enum.TRANSFER_AUTH,
|
||
|
RawTx: hexutil.Encode(rawTx),
|
||
|
TxHash: builtTx.Hash().Hex(),
|
||
|
From: cu.SystemPublicKey,
|
||
|
Data: hexutil.Encode(builtTx.Data()),
|
||
|
GasPrice: builtTx.GasPrice(),
|
||
|
GasLimit: builtTx.Gas(),
|
||
|
TransferValue: 0,
|
||
|
Nonce: builtTx.Nonce(),
|
||
|
})
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
if err := cu.CeloProvider.Client.CallCtx(
|
||
|
ctx,
|
||
|
eth.Balance(celoutils.HexToAddress(payload.Authorizer), nil).Returns(&networkBalance),
|
||
|
); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
disptachJobPayload, err := json.Marshal(TxPayload{
|
||
|
OtxId: id,
|
||
|
Tx: builtTx,
|
||
|
})
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
_, err = cu.TaskerClient.CreateTask(
|
||
|
ctx,
|
||
|
tasker.DispatchTxTask,
|
||
|
tasker.HighPriority,
|
||
|
&tasker.Task{
|
||
|
Payload: disptachJobPayload,
|
||
|
},
|
||
|
)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
// Auto-revoke every session (15 min)
|
||
|
// Check if already a revoke request
|
||
|
if payload.Amount > 0 {
|
||
|
taskPayload, err := json.Marshal(TransferAuthPayload{
|
||
|
TrackingId: payload.TrackingId,
|
||
|
Amount: 0,
|
||
|
Authorizer: payload.Authorizer,
|
||
|
AuthorizedAddress: payload.AuthorizedAddress,
|
||
|
VoucherAddress: payload.VoucherAddress,
|
||
|
})
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
_, err = cu.TaskerClient.CreateTask(
|
||
|
ctx,
|
||
|
tasker.SignTransferTaskAuth,
|
||
|
tasker.DefaultPriority,
|
||
|
&tasker.Task{
|
||
|
Payload: taskPayload,
|
||
|
},
|
||
|
asynq.ProcessIn(cu.ApprovalTimeout),
|
||
|
)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
gasRefillPayload, err := json.Marshal(AccountPayload{
|
||
|
PublicKey: payload.Authorizer,
|
||
|
TrackingId: payload.TrackingId,
|
||
|
})
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
if !balanceCheck(networkBalance) {
|
||
|
if err := cu.Store.GasLock(ctx, payload.Authorizer); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
_, err = cu.TaskerClient.CreateTask(
|
||
|
ctx,
|
||
|
tasker.AccountRefillGasTask,
|
||
|
tasker.DefaultPriority,
|
||
|
&tasker.Task{
|
||
|
Payload: gasRefillPayload,
|
||
|
},
|
||
|
)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
}
|