Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

add refunder to opbot #2838

Merged
merged 16 commits into from
Jul 5, 2024
2 changes: 1 addition & 1 deletion agents/go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -203,7 +203,7 @@ require (
github.com/google/uuid v1.6.0 // indirect
github.com/googleapis/enterprise-certificate-proxy v0.3.2 // indirect
github.com/googleapis/gax-go/v2 v2.12.4 // indirect
github.com/gorilla/websocket v1.5.1 // indirect
github.com/gorilla/websocket v1.5.3 // indirect
github.com/grpc-ecosystem/grpc-gateway/v2 v2.20.0 // indirect
github.com/hashicorp/consul/sdk v0.14.0 // indirect
github.com/hashicorp/errwrap v1.1.0 // indirect
Expand Down
3 changes: 1 addition & 2 deletions agents/go.sum
Original file line number Diff line number Diff line change
Expand Up @@ -622,8 +622,7 @@ github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1/go.mod h1:wJfORR
github.com/gorilla/mux v1.8.0/go.mod h1:DVbg23sWSpFRCP0SfiEN6jmj59UnW/n46BH5rLB71So=
github.com/gorilla/websocket v1.4.1/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
github.com/gorilla/websocket v1.5.1 h1:gmztn0JnHVt9JZquRuzLw3g4wouNVzKL15iLr/zn/QY=
github.com/gorilla/websocket v1.5.1/go.mod h1:x3kM2JMyaluk02fnUJpQuwD2dCS5NDG2ZHL0uE0tcaY=
github.com/gorilla/websocket v1.5.3 h1:saDtZ6Pbx/0u+bgYQ3q96pZgCzfhKXGPqt7kZ72aNNg=
github.com/grafana/otel-profiling-go v0.5.1 h1:stVPKAFZSa7eGiqbYuG25VcqYksR6iWvF3YH66t4qL8=
github.com/grafana/otel-profiling-go v0.5.1/go.mod h1:ftN/t5A/4gQI19/8MoWurBEtC6gFw8Dns1sJZ9W4Tls=
github.com/grafana/pyroscope-go v1.1.1 h1:PQoUU9oWtO3ve/fgIiklYuGilvsm8qaGhlY4Vw6MAcQ=
Expand Down
44 changes: 42 additions & 2 deletions contrib/opbot/botmd/botmd.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,18 @@

import (
"context"
"fmt"
"github.com/slack-io/slacker"
"github.com/synapsecns/sanguine/contrib/opbot/config"
"github.com/synapsecns/sanguine/contrib/opbot/signoz"
"github.com/synapsecns/sanguine/core/dbcommon"
"github.com/synapsecns/sanguine/core/metrics"
signerConfig "github.com/synapsecns/sanguine/ethergo/signer/config"
"github.com/synapsecns/sanguine/ethergo/signer/signer"
"github.com/synapsecns/sanguine/ethergo/submitter"
cctpSql "github.com/synapsecns/sanguine/services/cctp-relayer/db/sql"
omnirpcClient "github.com/synapsecns/sanguine/services/omnirpc/client"
"golang.org/x/sync/errgroup"
)

// Bot represents the bot server.
Expand All @@ -15,6 +23,9 @@
cfg config.Config
signozClient *signoz.Client
signozEnabled bool
rpcClient omnirpcClient.RPCClient
signer signer.Signer
submitter submitter.TransactionSubmitter
}

// NewBot creates a new bot server.
Expand All @@ -32,8 +43,10 @@
bot.signozEnabled = true
}

bot.rpcClient = omnirpcClient.NewOmnirpcClient(cfg.OmniRPCURL, handler, omnirpcClient.WithCaptureReqRes())

Check warning on line 47 in contrib/opbot/botmd/botmd.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/botmd.go#L46-L47

Added lines #L46 - L47 were not covered by tests
Comment on lines +46 to +47
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Add tests for the new functionality.

The rpcClient initialization lacks test coverage. Ensure that tests are added to cover this initialization.

Do you want me to generate the unit testing code or open a GitHub issue to track this task?

Tools
GitHub Check: codecov/patch

[warning] 46-47: contrib/opbot/botmd/botmd.go#L46-L47
Added lines #L46 - L47 were not covered by tests

bot.addMiddleware(bot.tracingMiddleware(), bot.metricsMiddleware())
bot.addCommands(bot.traceCommand(), bot.rfqLookupCommand())
bot.addCommands(bot.traceCommand(), bot.rfqLookupCommand(), bot.rfqRefund())

Check warning on line 49 in contrib/opbot/botmd/botmd.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/botmd.go#L49

Added line #L49 was not covered by tests
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Add tests for the new functionality.

The addition of the rfqRefund command lacks test coverage. Ensure that tests are added to cover this functionality.

Do you want me to generate the unit testing code or open a GitHub issue to track this task?

Tools
GitHub Check: codecov/patch

[warning] 49-49: contrib/opbot/botmd/botmd.go#L49
Added line #L49 was not covered by tests


return bot
}
Expand All @@ -53,6 +66,33 @@
// Start starts the bot server.
// nolint: wrapcheck
func (b *Bot) Start(ctx context.Context) error {
var err error
b.signer, err = signerConfig.SignerFromConfig(ctx, b.cfg.Signer)
if err != nil {
return fmt.Errorf("failed to create signer: %w", err)
Comment on lines +70 to +72
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🧠 logic: Error handling for signer creation should be reviewed to ensure it covers all edge cases.

Comment on lines +70 to +72
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🧠 logic: Error handling for signer creation should be reviewed to ensure it covers all edge cases.

}

Check warning on line 73 in contrib/opbot/botmd/botmd.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/botmd.go#L69-L73

Added lines #L69 - L73 were not covered by tests
Comment on lines +69 to +73
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Add tests for the new functionality.

The initialization of the signer lacks test coverage. Ensure that tests are added to cover this initialization.

Do you want me to generate the unit testing code or open a GitHub issue to track this task?

Tools
GitHub Check: codecov/patch

[warning] 69-73: contrib/opbot/botmd/botmd.go#L69-L73
Added lines #L69 - L73 were not covered by tests


dbType, err := dbcommon.DBTypeFromString(b.cfg.Database.Type)
if err != nil {
return fmt.Errorf("could not get db type: %w", err)
Comment on lines +75 to +77
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🧠 logic: Error handling for database type conversion should be reviewed to ensure it covers all edge cases.

Comment on lines +75 to +77
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🧠 logic: Error handling for database type conversion should be reviewed to ensure it covers all edge cases.

}

Check warning on line 78 in contrib/opbot/botmd/botmd.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/botmd.go#L75-L78

Added lines #L75 - L78 were not covered by tests
Comment on lines +75 to +78
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Add tests for the new functionality.

The database connection lacks test coverage. Ensure that tests are added to cover this functionality.

Do you want me to generate the unit testing code or open a GitHub issue to track this task?

Tools
GitHub Check: codecov/patch

[warning] 75-78: contrib/opbot/botmd/botmd.go#L75-L78
Added lines #L75 - L78 were not covered by tests


store, err := cctpSql.Connect(ctx, dbType, b.cfg.Database.DSN, b.handler)
if err != nil {
return fmt.Errorf("could not connect to database: %w", err)
Comment on lines +80 to +82
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🧠 logic: Error handling for database connection should be reviewed to ensure it covers all edge cases.

Comment on lines +80 to +82
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🧠 logic: Error handling for database connection should be reviewed to ensure it covers all edge cases.

}

Check warning on line 83 in contrib/opbot/botmd/botmd.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/botmd.go#L80-L83

Added lines #L80 - L83 were not covered by tests
Comment on lines +80 to +83
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Add tests for the new functionality.

The connection to the database lacks test coverage. Ensure that tests are added to cover this functionality.

Do you want me to generate the unit testing code or open a GitHub issue to track this task?

Tools
GitHub Check: codecov/patch

[warning] 80-83: contrib/opbot/botmd/botmd.go#L80-L83
Added lines #L80 - L83 were not covered by tests


b.submitter = submitter.NewTransactionSubmitter(b.handler, b.signer, b.rpcClient, store.SubmitterDB(), &b.cfg.SubmitterConfig)

g, ctx := errgroup.WithContext(ctx)
g.Go(func() error {
return b.submitter.Start(ctx)
})

Check warning on line 90 in contrib/opbot/botmd/botmd.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/botmd.go#L85-L90

Added lines #L85 - L90 were not covered by tests
Comment on lines +85 to +90
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Add tests for the new functionality.

The initialization and starting of the submitter lack test coverage. Ensure that tests are added to cover this functionality.

Do you want me to generate the unit testing code or open a GitHub issue to track this task?

Tools
GitHub Check: codecov/patch

[warning] 85-90: contrib/opbot/botmd/botmd.go#L85-L90
Added lines #L85 - L90 were not covered by tests


g.Go(func() error {
return b.server.Listen(ctx)
})

Check warning on line 94 in contrib/opbot/botmd/botmd.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/botmd.go#L92-L94

Added lines #L92 - L94 were not covered by tests
Comment on lines +92 to +94
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Add tests for the new functionality.

The starting of the server lacks test coverage. Ensure that tests are added to cover this functionality.

Do you want me to generate the unit testing code or open a GitHub issue to track this task?

Tools
GitHub Check: codecov/patch

[warning] 92-94: contrib/opbot/botmd/botmd.go#L92-L94
Added lines #L92 - L94 were not covered by tests


// nolint: wrapcheck
return b.server.Listen(ctx)
return g.Wait()

Check warning on line 97 in contrib/opbot/botmd/botmd.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/botmd.go#L97

Added line #L97 was not covered by tests
}
135 changes: 135 additions & 0 deletions contrib/opbot/botmd/commands.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,16 +3,23 @@
package botmd

import (
"context"
"fmt"
"github.com/ethereum/go-ethereum/accounts/abi/bind"
"github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/core/types"
"github.com/hako/durafmt"
"github.com/slack-go/slack"
"github.com/slack-io/slacker"
"github.com/synapsecns/sanguine/contrib/opbot/signoz"
"github.com/synapsecns/sanguine/ethergo/chaindata"
rfqClient "github.com/synapsecns/sanguine/services/rfq/api/client"
"github.com/synapsecns/sanguine/services/rfq/contracts/fastbridge"
"github.com/synapsecns/sanguine/services/rfq/relayer/relapi"
"log"
"math/big"
"regexp"
"strconv"
"strings"
"sync"
"time"
Expand Down Expand Up @@ -239,6 +246,107 @@
}}
}

func (b *Bot) rfqRefund() *slacker.CommandDefinition {

Check failure on line 249 in contrib/opbot/botmd/commands.go

View workflow job for this annotation

GitHub Actions / Lint (contrib/opbot)

cognitive complexity 46 of func `(*Bot).rfqRefund` is high (> 30) (gocognit)
return &slacker.CommandDefinition{
Command: "refund <tx> <origin_chainid>",
Description: "refund a quote request",
Examples: []string{"refund 0x1234"},
Handler: func(ctx *slacker.CommandContext) {
client, err := rfqClient.NewUnauthenticatedClient(b.handler, b.cfg.RFQApiURL)
if err != nil {
log.Printf("error creating rfq client: %v\n", err)
return
}

Check warning on line 259 in contrib/opbot/botmd/commands.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/commands.go#L249-L259

Added lines #L249 - L259 were not covered by tests
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reduce the cognitive complexity of the rfqRefund function.

The function has a high cognitive complexity (46). Consider refactoring to break down the function into smaller, more manageable functions.

func (b *Bot) rfqRefund() *slacker.CommandDefinition {
	return &slacker.CommandDefinition{
		Command:     "refund <tx> <origin_chainid>",
		Description: "refund a quote request",
		Examples:    []string{"refund 0x1234"},
		Handler: func(ctx *slacker.CommandContext) {
			if err := b.handleRFQRefund(ctx); err != nil {
				log.Println(err)
			}
		},
	}
}

func (b *Bot) handleRFQRefund(ctx *slacker.CommandContext) error {
	client, err := rfqClient.NewUnauthenticatedClient(b.handler, b.cfg.RFQApiURL)
	if err != nil {
		log.Printf("error creating rfq client: %v\n", err)
		return err
	}

	contracts, err := client.GetRFQContracts(ctx.Context())
	if err != nil {
		log.Printf("error fetching rfq contracts: %v\n", err)
		_, err := ctx.Response().Reply("error fetching rfq contracts")
		if err != nil {
			return err
		}
		return err
	}

	tx := stripLinks(ctx.Request().Param("tx"))
	if len(tx) == 0 {
		_, err := ctx.Response().Reply("please provide a tx hash")
		if err != nil {
			log.Println(err)
		}
		return nil
	}

	originChainID, err := b.getOriginChainID(ctx)
	if err != nil {
		return err
	}

	chainClient, err := b.rpcClient.GetChainClient(ctx.Context(), originChainID)
	if err != nil {
		log.Printf("error getting chain client: %v\n", err)
		return err
	}

	contractAddress, ok := contracts.Contracts[uint32(originChainID)]
	if !ok {
		_, err := ctx.Response().Reply("contract address not found")
		if err != nil {
			log.Println(err)
		}
		return nil
	}

	fastBridgeHandle, err := fastbridge.NewFastBridge(common.HexToAddress(contractAddress), chainClient)
	if err != nil {
		log.Printf("error creating fast bridge: %v\n", err)
		return err
	}

	return b.submitRefund(ctx, fastBridgeHandle, tx, originChainID)
}

func (b *Bot) getOriginChainID(ctx *slacker.CommandContext) (int, error) {
	originChainIDStr := ctx.Request().Param("origin_chainid")
	if len(originChainIDStr) == 0 {
		_, err := ctx.Response().Reply("please provide an origin chain id")
		if err != nil {
			log.Println(err)
		}
		return 0, nil
	}

	originChainID, err := strconv.Atoi(convertChainName(originChainIDStr))
	if err != nil {
		_, err := ctx.Response().Reply("origin_chainid must be a number")
		if err != nil {
			log.Println(err)
		}
		return 0, err
	}

	return originChainID, nil
}

func (b *Bot) submitRefund(ctx *slacker.CommandContext, fastBridgeHandle *fastbridge.FastBridgeTransactor, tx string, originChainID int) error {
	for _, relayer := range b.cfg.RelayerURLS {
		relClient := relapi.NewRelayerClient(b.handler, relayer)

		rawRequest, err := getQuoteRequest(ctx.Context(), relClient, tx)
		if err != nil {
			_, err := ctx.Response().Reply("error fetching quote request")
			if err != nil {
				log.Println(err)
			}
			return err
		}

		nonce, err := b.submitter.SubmitTransaction(ctx.Context(), big.NewInt(int64(originChainID)), func(transactor *bind.TransactOpts) (*types.Transaction, error) {
			return fastBridgeHandle.Refund(transactor, common.Hex2Bytes(rawRequest.QuoteRequestRaw))
		})
		if err != nil {
			log.Printf("error submitting refund: %v\n", err)
			continue
		}

		if _, err := ctx.Response().Reply(fmt.Sprintf("refund submitted with nonce %d", nonce)); err != nil {
			log.Println(err)
		}
		return nil
	}
	return nil
}
Tools
GitHub Check: codecov/patch

[warning] 249-259: contrib/opbot/botmd/commands.go#L249-L259
Added lines #L249 - L259 were not covered by tests

GitHub Check: Lint (contrib/opbot)

[failure] 249-249:
cognitive complexity 46 of func (*Bot).rfqRefund is high (> 30) (gocognit)


contracts, err := client.GetRFQContracts(ctx.Context())
if err != nil {
log.Printf("error fetching rfq contracts: %v\n", err)
_, err = ctx.Response().Reply("error fetching rfq contracts")
if err != nil {
return
}
return

Check warning on line 268 in contrib/opbot/botmd/commands.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/commands.go#L261-L268

Added lines #L261 - L268 were not covered by tests
}

tx := stripLinks(ctx.Request().Param("tx"))

if len(tx) == 0 {
_, err := ctx.Response().Reply("please provide a tx hash")
if err != nil {
log.Println(err)
}
return

Check warning on line 278 in contrib/opbot/botmd/commands.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/commands.go#L271-L278

Added lines #L271 - L278 were not covered by tests
}

originChainIDStr := ctx.Request().Param("origin_chainid")
if len(originChainIDStr) == 0 {
_, err := ctx.Response().Reply("please provide an origin chain id")
if err != nil {
log.Println(err)
}
return

Check warning on line 287 in contrib/opbot/botmd/commands.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/commands.go#L281-L287

Added lines #L281 - L287 were not covered by tests
}

originChainID, err := strconv.Atoi(convertChainName(originChainIDStr))
if err != nil {
_, err := ctx.Response().Reply("origin_chainid must be a number")
if err != nil {
log.Println(err)
}
return

Check warning on line 296 in contrib/opbot/botmd/commands.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/commands.go#L290-L296

Added lines #L290 - L296 were not covered by tests
}

chainClient, err := b.rpcClient.GetChainClient(ctx.Context(), originChainID)
if err != nil {
log.Printf("error getting chain client: %v\n", err)
return
}

Check warning on line 303 in contrib/opbot/botmd/commands.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/commands.go#L299-L303

Added lines #L299 - L303 were not covered by tests

contractAddress, ok := contracts.Contracts[uint32(originChainID)]
if !ok {
_, err := ctx.Response().Reply("contract address not found")
if err != nil {
log.Println(err)
}
return

Check warning on line 311 in contrib/opbot/botmd/commands.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/commands.go#L305-L311

Added lines #L305 - L311 were not covered by tests
}

fastBridgeHandle, err := fastbridge.NewFastBridge(common.HexToAddress(contractAddress), chainClient)
if err != nil {
log.Printf("error creating fast bridge: %v\n", err)
return
}

Check warning on line 318 in contrib/opbot/botmd/commands.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/commands.go#L314-L318

Added lines #L314 - L318 were not covered by tests

for _, relayer := range b.cfg.RelayerURLS {
relClient := relapi.NewRelayerClient(b.handler, relayer)

rawRequest, err := getQuoteRequest(ctx.Context(), relClient, tx)
if err != nil {
_, err := ctx.Response().Reply("error fetching quote request")
if err != nil {
log.Println(err)
}
return

Check warning on line 329 in contrib/opbot/botmd/commands.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/commands.go#L320-L329

Added lines #L320 - L329 were not covered by tests
}

nonce, err := b.submitter.SubmitTransaction(ctx.Context(), big.NewInt(int64(originChainID)), func(transactor *bind.TransactOpts) (tx *types.Transaction, err error) {
return fastBridgeHandle.Refund(transactor, rawRequest)

Check failure on line 333 in contrib/opbot/botmd/commands.go

View workflow job for this annotation

GitHub Actions / Lint (contrib/opbot)

error returned from external package is unwrapped: sig: func (*github.com/synapsecns/sanguine/services/rfq/contracts/fastbridge.FastBridgeTransactor).Refund(opts *github.com/ethereum/go-ethereum/accounts/abi/bind.TransactOpts, request []byte) (*github.com/ethereum/go-ethereum/core/types.Transaction, error) (wrapcheck)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Wrap the error returned from the external package.

The error returned from fastBridgeHandle.Refund should be wrapped to provide more context.

-  return fastBridgeHandle.Refund(transactor, rawRequest)
+  tx, err := fastBridgeHandle.Refund(transactor, rawRequest)
+  if err != nil {
+    return nil, fmt.Errorf("failed to refund: %w", err)
+  }
+  return tx, nil
Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
return fastBridgeHandle.Refund(transactor, rawRequest)
tx, err := fastBridgeHandle.Refund(transactor, rawRequest)
if err != nil {
return nil, fmt.Errorf("failed to refund: %w", err)
}
return tx, nil
Tools
GitHub Check: Lint (contrib/opbot)

[failure] 333-333:
error returned from external package is unwrapped: sig: func (*github.com/synapsecns/sanguine/services/rfq/contracts/fastbridge.FastBridgeTransactor).Refund(opts *github.com/ethereum/go-ethereum/accounts/abi/bind.TransactOpts, request []byte) (*github.com/ethereum/go-ethereum/core/types.Transaction, error) (wrapcheck)

})
if err != nil {
log.Printf("error submitting refund: %v\n", err)
continue

Check warning on line 337 in contrib/opbot/botmd/commands.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/commands.go#L332-L337

Added lines #L332 - L337 were not covered by tests
}

_, err = ctx.Response().Reply(fmt.Sprintf("refund submitted with nonce %d", nonce))
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Fix ineffectual assignment to err.

The variable err is assigned but not used.

-  _, err = ctx.Response().Reply(fmt.Sprintf("refund submitted with nonce %d", nonce))
+  if _, err := ctx.Response().Reply(fmt.Sprintf("refund submitted with nonce %d", nonce)); err != nil {
+    log.Println(err)
+  }
Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
_, err = ctx.Response().Reply(fmt.Sprintf("refund submitted with nonce %d", nonce))
if _, err := ctx.Response().Reply(fmt.Sprintf("refund submitted with nonce %d", nonce)); err != nil {
log.Println(err)
}
Tools
GitHub Check: codecov/patch

[warning] 341-345: contrib/opbot/botmd/commands.go#L341-L345
Added lines #L341 - L345 were not covered by tests

if err != nil {
log.Println(err)
}
return

Check warning on line 344 in contrib/opbot/botmd/commands.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/commands.go#L340-L344

Added lines #L340 - L344 were not covered by tests
}
},
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reduce the cognitive complexity of the rfqRefund function.

The function has a high cognitive complexity (46). Consider refactoring to reduce complexity.

One way to reduce complexity is to break down the function into smaller, more manageable functions. For example:

func (b *Bot) rfqRefund() *slacker.CommandDefinition {
	return &slacker.CommandDefinition{
		Command:     "refund <tx> <origin_chainid>",
		Description: "refund a quote request",
		Examples:    []string{"refund 0x1234"},
		Handler: func(ctx *slacker.CommandContext) {
			if err := b.handleRFQRefund(ctx); err != nil {
				log.Println(err)
			}
		},
	}
}

func (b *Bot) handleRFQRefund(ctx *slacker.CommandContext) error {
	client, err := rfqClient.NewUnauthenticatedClient(b.handler, b.cfg.RFQApiURL)
	if err != nil {
		return fmt.Errorf("error creating rfq client: %w", err)
	}

	contracts, err := client.GetRFQContracts(ctx.Context())
	if err != nil {
		return b.respondWithError(ctx, "error fetching rfq contracts", err)
	}

	tx := stripLinks(ctx.Request().Param("tx"))
	if len(tx) == 0 {
		return b.respondWithError(ctx, "please provide a tx hash", nil)
	}

	originChainID, err := b.getOriginChainID(ctx)
	if err != nil {
		return err
	}

	chainClient, err := b.rpcClient.GetChainClient(ctx.Context(), originChainID)
	if err != nil {
		return fmt.Errorf("error getting chain client: %w", err)
	}

	contractAddress, ok := contracts.Contracts[uint32(originChainID)]
	if !ok {
		return b.respondWithError(ctx, "contract address not found", nil)
	}

	fastBridgeHandle, err := fastbridge.NewFastBridge(common.HexToAddress(contractAddress), chainClient)
	if err != nil {
		return fmt.Errorf("error creating fast bridge: %w", err)
	}

	return b.submitRefund(ctx, fastBridgeHandle, tx, originChainID)
}

func (b *Bot) respondWithError(ctx *slacker.CommandContext, message string, err error) error {
	if err != nil {
		log.Printf("%s: %v\n", message, err)
	}
	_, respErr := ctx.Response().Reply(message)
	if respErr != nil {
		log.Println(respErr)
	}
	return err
}

func (b *Bot) getOriginChainID(ctx *slacker.CommandContext) (int, error) {
	originChainIDStr := ctx.Request().Param("origin_chainid")
	if len(originChainIDStr) == 0 {
		return 0, b.respondWithError(ctx, "please provide an origin chain id", nil)
	}

	originChainID, err := strconv.Atoi(convertChainName(originChainIDStr))
	if err != nil {
		return 0, b.respondWithError(ctx, "origin_chainid must be a number", err)
	}

	return originChainID, nil
}

func (b *Bot) submitRefund(ctx *slacker.CommandContext, fastBridgeHandle *fastbridge.FastBridgeTransactor, tx string, originChainID int) error {
	for _, relayer := range b.cfg.RelayerURLS {
		relClient := relapi.NewRelayerClient(b.handler, relayer)

		rawRequest, err := getQuoteRequest(ctx.Context(), relClient, tx)
		if err != nil {
			return b.respondWithError(ctx, "error fetching quote request", err)
		}

		nonce, err := b.submitter.SubmitTransaction(ctx.Context(), big.NewInt(int64(originChainID)), func(transactor *bind.TransactOpts) (*types.Transaction, error) {
			return fastBridgeHandle.Refund(transactor, rawRequest)
		})
		if err != nil {
			log.Printf("error submitting refund: %v\n", err)
			continue
		}

		if _, err := ctx.Response().Reply(fmt.Sprintf("refund submitted with nonce %d", nonce)); err != nil {
			log.Println(err)
		}
		return nil
	}
	return nil
}
Tools
GitHub Check: codecov/patch

[warning] 249-259: contrib/opbot/botmd/commands.go#L249-L259
Added lines #L249 - L259 were not covered by tests


[warning] 261-263: contrib/opbot/botmd/commands.go#L261-L263
Added lines #L261 - L263 were not covered by tests


[warning] 268-269: contrib/opbot/botmd/commands.go#L268-L269
Added lines #L268 - L269 were not covered by tests


[warning] 271-271: contrib/opbot/botmd/commands.go#L271
Added line #L271 was not covered by tests


[warning] 281-281: contrib/opbot/botmd/commands.go#L281
Added line #L281 was not covered by tests


[warning] 289-289: contrib/opbot/botmd/commands.go#L289
Added line #L289 was not covered by tests


[warning] 291-296: contrib/opbot/botmd/commands.go#L291-L296
Added lines #L291 - L296 were not covered by tests


[warning] 299-303: contrib/opbot/botmd/commands.go#L299-L303
Added lines #L299 - L303 were not covered by tests


[warning] 305-311: contrib/opbot/botmd/commands.go#L305-L311
Added lines #L305 - L311 were not covered by tests


[warning] 314-318: contrib/opbot/botmd/commands.go#L314-L318
Added lines #L314 - L318 were not covered by tests


[warning] 320-321: contrib/opbot/botmd/commands.go#L320-L321
Added lines #L320 - L321 were not covered by tests


[warning] 324-324: contrib/opbot/botmd/commands.go#L324
Added line #L324 was not covered by tests


[warning] 332-332: contrib/opbot/botmd/commands.go#L332
Added line #L332 was not covered by tests


[warning] 334-337: contrib/opbot/botmd/commands.go#L334-L337
Added lines #L334 - L337 were not covered by tests


[warning] 340-344: contrib/opbot/botmd/commands.go#L340-L344
Added lines #L340 - L344 were not covered by tests

GitHub Check: Lint (contrib/opbot)

[failure] 249-249:
cognitive complexity 46 of func (*Bot).rfqRefund is high (> 30) (gocognit)


[failure] 333-333:
error returned from external package is unwrapped: sig: func (*github.com/synapsecns/sanguine/services/rfq/contracts/fastbridge.FastBridgeTransactor).Refund(opts *github.com/ethereum/go-ethereum/accounts/abi/bind.TransactOpts, request []byte) (*github.com/ethereum/go-ethereum/core/types.Transaction, error) (wrapcheck)

}

func toExplorerSlackLink(ogHash string) string {
rfqHash := strings.ToUpper(ogHash)
// cut off 0x
Expand All @@ -264,3 +372,30 @@
linkRegex := regexp.MustCompile(`<https?://[^|>]+\|([^>]+)>`)
return linkRegex.ReplaceAllString(input, "$1")
}

// convertChainName detects if the string contains letters and if it does, tries to convert it to a chain id.
// otherwise return the original string

Check failure on line 377 in contrib/opbot/botmd/commands.go

View workflow job for this annotation

GitHub Actions / Lint (contrib/opbot)

Comment should end in a period (godot)
func convertChainName(input string) string {
res := chaindata.ChainNameToChainID(input)
if res != 0 {
return strconv.Itoa(int(res))
}
return input

Check warning on line 383 in contrib/opbot/botmd/commands.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/commands.go#L378-L383

Added lines #L378 - L383 were not covered by tests
}

func getQuoteRequest(ctx context.Context, client relapi.RelayerClient, tx string) ([]byte, error) {
// at this point tx can be a txid or a has, we try both
txRequest, err := client.GetQuoteRequestStatusByTxHash(ctx, tx)
if err == nil {
// override tx with txid
tx = txRequest.TxID
}

Check warning on line 392 in contrib/opbot/botmd/commands.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/commands.go#L386-L392

Added lines #L386 - L392 were not covered by tests

// look up quote request
qr, err := client.GetQuoteRequestByTXID(ctx, tx)
if err != nil {
return nil, fmt.Errorf("error fetching quote request: %w", err)
}

Check warning on line 398 in contrib/opbot/botmd/commands.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/commands.go#L395-L398

Added lines #L395 - L398 were not covered by tests
Comment on lines +381 to +384
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ensure error handling for GetQuoteRequestByTXID.

The error handling for GetQuoteRequestByTXID should be improved to provide more context.

-  qr, err := client.GetQuoteRequestByTXID(ctx, tx)
-  if err != nil {
-    return nil, fmt.Errorf("error fetching quote request: %w", err)
+  qr, err := client.GetQuoteRequestByTXID(ctx, tx)
+  if err != nil {
+    return nil, fmt.Errorf("error fetching quote request by tx id: %w", err)
Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
qr, err := client.GetQuoteRequestByTXID(ctx, tx)
if err != nil {
return nil, fmt.Errorf("error fetching quote request: %w", err)
}
qr, err := client.GetQuoteRequestByTXID(ctx, tx)
if err != nil {
return nil, fmt.Errorf("error fetching quote request by tx id: %w", err)
}
Tools
GitHub Check: codecov/patch

[warning] 396-399: contrib/opbot/botmd/commands.go#L396-L399
Added lines #L396 - L399 were not covered by tests


return common.Hex2Bytes(qr.QuoteRequestRaw), nil

Check warning on line 400 in contrib/opbot/botmd/commands.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/commands.go#L400

Added line #L400 was not covered by tests
}
5 changes: 3 additions & 2 deletions contrib/opbot/botmd/middleware.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,12 +21,13 @@
func (b *Bot) tracingMiddleware() slacker.CommandMiddlewareHandler {
return func(next slacker.CommandHandler) slacker.CommandHandler {
return func(cmdCtx *slacker.CommandContext) {
// TODO: context is not inherited here.
_, span := b.handler.Tracer().Start(cmdCtx.Context(), fmt.Sprintf("command.%s", cmdCtx.Definition().Command), trace.WithAttributes(
ctx, span := b.handler.Tracer().Start(cmdCtx.Context(), fmt.Sprintf("command.%s", cmdCtx.Definition().Command), trace.WithAttributes(

Check warning on line 24 in contrib/opbot/botmd/middleware.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/middleware.go#L24

Added line #L24 was not covered by tests
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Tip

Codebase Verification

Add tests for new functionality.

The new functionality added to the tracingMiddleware method is not covered by tests. Ensure that tests are added to cover:

  • Context inheritance using ctx.

  • Span creation with appropriate attributes.

  • contrib/opbot/botmd/middleware.go

Analysis chain

Add tests for new functionality.

The new functionality added to the tracingMiddleware method is not covered by tests. Ensure that tests are added to cover:

  • Context inheritance using ctx.
  • Span creation with appropriate attributes.
Scripts executed

The following scripts were executed for the analysis:


Script:

#!/bin/bash
# Description: Verify test coverage for new functionality in `tracingMiddleware`.

# Test: Search for test cases covering the `tracingMiddleware` method. Expect: Test cases covering all new functionality.
rg --type go 'tracingMiddleware'

Length of output: 233



Script:

#!/bin/bash
# Description: Verify test coverage for the `tracingMiddleware` method by searching for test files and functions.

# Find all test files in the repository.
fd --type f --name '*_test.go'

# Search for test functions that might cover `tracingMiddleware`.
rg --type go 'func Test' | rg 'tracingMiddleware'

Length of output: 333



Script:

#!/bin/bash
# Description: Verify test coverage for the `tracingMiddleware` method by searching for test files and functions.

# Find all test files in the repository.
fd '_test.go$'

# Search for references to `tracingMiddleware` within test files.
rg 'tracingMiddleware' $(fd '_test.go$')

Length of output: 25161

Tools
GitHub Check: codecov/patch

[warning] 24-24: contrib/opbot/botmd/middleware.go#L24
Added line #L24 was not covered by tests

attribute.String("user_id", cmdCtx.Event().UserID),
attribute.String("channel_id", cmdCtx.Event().Channel.ID),
))

cmdCtx.WithContext(ctx)

Check warning on line 30 in contrib/opbot/botmd/middleware.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/botmd/middleware.go#L29-L30

Added lines #L29 - L30 were not covered by tests
Comment on lines +29 to +30
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Tip

Codebase Verification

Add tests for new functionality.

The tracingMiddleware method is not covered by any existing test cases. Please add tests to ensure the following functionalities are covered:

  • Context inheritance using ctx.

  • Span creation with appropriate attributes.

  • contrib/opbot/botmd/middleware.go

Analysis chain

Add tests for new functionality.

The new functionality added to the tracingMiddleware method is not covered by tests. Ensure that tests are added to cover:

  • Context inheritance using ctx.
  • Span creation with appropriate attributes.
Scripts executed

The following scripts were executed for the analysis:


Script:

#!/bin/bash
# Description: Verify test coverage for new functionality in `tracingMiddleware`.

# Test: Search for test cases covering the `tracingMiddleware` method. Expect: Test cases covering all new functionality.
rg --type go 'tracingMiddleware'

Length of output: 233



Script:

#!/bin/bash
# Description: Verify if there are tests covering the `tracingMiddleware` method.

# Find all Go test files in the repository
test_files=$(fd --type f --extension go --glob '*_test.go')

# Search for test cases covering the `tracingMiddleware` method in the test files
for file in $test_files; do
    echo "Checking $file"
    rg 'tracingMiddleware' "$file"
done

Length of output: 75047

Tools
GitHub Check: codecov/patch

[warning] 29-30: contrib/opbot/botmd/middleware.go#L29-L30
Added lines #L29 - L30 were not covered by tests

defer func() {
metrics.EndSpan(span)
}()
Expand Down
3 changes: 2 additions & 1 deletion contrib/opbot/cmd/commands.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
import (
"fmt"
"github.com/synapsecns/sanguine/contrib/opbot/botmd"
"github.com/synapsecns/sanguine/core"
"github.com/synapsecns/sanguine/core/metrics"

// used for testing.
Expand All @@ -23,7 +24,7 @@
Usage: "start the slack bot",
Flags: []cli.Flag{fileFlag},
Action: func(c *cli.Context) error {
configFile, err := os.ReadFile(c.String(fileFlag.Name))
configFile, err := os.ReadFile(core.ExpandOrReturnPath(c.String(fileFlag.Name)))

Check warning on line 27 in contrib/opbot/cmd/commands.go

View check run for this annotation

Codecov / codecov/patch

contrib/opbot/cmd/commands.go#L27

Added line #L27 was not covered by tests
if err != nil {
return fmt.Errorf("failed to open config file: %w", err)
}
Expand Down
21 changes: 21 additions & 0 deletions contrib/opbot/config/config.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,11 @@
// Package config provides a simple way to read and write configuration files.
package config

import (
"github.com/synapsecns/sanguine/ethergo/signer/config"
submitterConfig "github.com/synapsecns/sanguine/ethergo/submitter/config"
)

// Config represents the configuration of the application.
type Config struct {
// SlackBotToken is the token of the slack bot.
Expand All @@ -18,4 +23,20 @@ type Config struct {
SignozBaseURL string `yaml:"signoz_base_url"`
// RelayerURLS is the list of RFQ relayer URLs.
RelayerURLS []string `yaml:"rfq_relayer_urls"`
// RFQApiURL is the URL of the RFQ API.
RFQApiURL string `yaml:"rfq_api_url"`
// OmniRPCURL is the URL of the Omni RPC.
OmniRPCURL string `yaml:"omnirpc_url"`
// Signer is the signer config.
Signer config.SignerConfig `yaml:"signer"`
// SubmitterConfig is the submitter config.
SubmitterConfig submitterConfig.Config `yaml:"submitter_config"`
// Database is the database config.
Database DatabaseConfig `yaml:"database"`
}

// DatabaseConfig represents the configuration for the database.
type DatabaseConfig struct {
Type string `yaml:"type"`
DSN string `yaml:"dsn"` // Data Source Name
}
Loading
Loading