mirror of
https://github.com/grassrootseconomics/cic-custodial.git
synced 2025-01-27 07:47:32 +01:00
81 lines
2.4 KiB
Go
81 lines
2.4 KiB
Go
package task
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
|
|
"github.com/celo-org/celo-blockchain/common"
|
|
"github.com/celo-org/celo-blockchain/core/types"
|
|
"github.com/grassrootseconomics/celoutils"
|
|
"github.com/grassrootseconomics/cic-custodial/internal/custodial"
|
|
"github.com/grassrootseconomics/cic-custodial/internal/pub"
|
|
"github.com/grassrootseconomics/cic-custodial/internal/store"
|
|
"github.com/grassrootseconomics/cic-custodial/pkg/enum"
|
|
"github.com/grassrootseconomics/w3-celo-patch/module/eth"
|
|
"github.com/hibiken/asynq"
|
|
)
|
|
|
|
type TxPayload struct {
|
|
OtxId uint `json:"otxId"`
|
|
Tx *types.Transaction `json:"tx"`
|
|
}
|
|
|
|
func DispatchTx(cu *custodial.Custodial) func(context.Context, *asynq.Task) error {
|
|
return func(ctx context.Context, t *asynq.Task) error {
|
|
var (
|
|
payload TxPayload
|
|
dispatchStatus store.DispatchStatus
|
|
eventPayload pub.EventPayload
|
|
dispathchTx common.Hash
|
|
)
|
|
|
|
if err := json.Unmarshal(t.Payload(), &payload); err != nil {
|
|
return fmt.Errorf("dispatch: failed %v: %w", err, asynq.SkipRetry)
|
|
}
|
|
|
|
txHash := payload.Tx.Hash().Hex()
|
|
|
|
dispatchStatus.OtxId, eventPayload.OtxId = payload.OtxId, payload.OtxId
|
|
eventPayload.TxHash = txHash
|
|
|
|
if err := cu.CeloProvider.Client.CallCtx(
|
|
ctx,
|
|
eth.SendTx(payload.Tx).Returns(&dispathchTx),
|
|
); err != nil {
|
|
switch err.Error() {
|
|
case celoutils.ErrGasPriceLow:
|
|
dispatchStatus.Status = enum.FAIL_LOW_GAS_PRICE
|
|
case celoutils.ErrInsufficientGas:
|
|
dispatchStatus.Status = enum.FAIL_NO_GAS
|
|
case celoutils.ErrNonceLow:
|
|
dispatchStatus.Status = enum.FAIL_LOW_NONCE
|
|
default:
|
|
dispatchStatus.Status = enum.FAIL_UNKNOWN_RPC_ERROR
|
|
}
|
|
|
|
if err := cu.PgStore.CreateDispatchStatus(ctx, dispatchStatus); err != nil {
|
|
return fmt.Errorf("dispatch: failed %v: %w", err, asynq.SkipRetry)
|
|
}
|
|
|
|
if err := cu.Pub.Publish(pub.DispatchFail, txHash, eventPayload); err != nil {
|
|
return fmt.Errorf("dispatch: failed %v: %w", err, asynq.SkipRetry)
|
|
}
|
|
|
|
return fmt.Errorf("dispatch: failed %v: %w", err, asynq.SkipRetry)
|
|
}
|
|
|
|
dispatchStatus.Status = enum.IN_NETWORK
|
|
|
|
if err := cu.PgStore.CreateDispatchStatus(ctx, dispatchStatus); err != nil {
|
|
return fmt.Errorf("dispatch: failed %v: %w", err, asynq.SkipRetry)
|
|
}
|
|
|
|
if err := cu.Pub.Publish(pub.DispatchSuccess, txHash, eventPayload); err != nil {
|
|
return fmt.Errorf("dispatch: failed %v: %w", err, asynq.SkipRetry)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
}
|