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

proposed mods for feat/guard-v2 [SLT-422] #3364

Merged
merged 11 commits into from
Nov 6, 2024
22 changes: 11 additions & 11 deletions services/rfq/guard/guardconfig/config.go
Original file line number Diff line number Diff line change
Expand Up @@ -34,10 +34,10 @@

// ChainConfig represents the configuration for a chain.
type ChainConfig struct {
// RFQAddressV1 is the rfq bridge contract address.
// RFQAddressV1 is the legacy V1 rfq bridge contract address. OPTIONAL. Only populate if also guarding a deprecated V1 contract.
RFQAddressV1 *string `yaml:"rfq_address_v1"`
// RFQAddressV2 is the rfq bridge contract address.
RFQAddressV2 string `yaml:"rfq_address_v2"`
// RFQAddress is the current/latest rfq bridge contract address. REQUIRED.
RFQAddress string `yaml:"rfq_address"`
Comment on lines +37 to +40
Copy link
Contributor

Choose a reason for hiding this comment

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

💡 Codebase verification

Update required: RFQAddressV2 references need to be modified

The consolidation of RFQAddress fields requires corresponding updates to the codebase. Several files still contain references to RFQAddressV2:

  • services/rfq/guard/guardconfig/config.go: 3 references including the method GetRFQAddressV2
  • services/rfq/guard/service/handlers.go: 1 reference in bridge request handling
  • services/rfq/guard/service/guard.go: 2 references in contract address retrieval

These references should be updated to use the new consolidated RFQAddress field to maintain consistency with the configuration changes.

🔗 Analysis chain

LGTM! Verify impact of RFQAddress consolidation.

The struct changes improve clarity by marking V1 as legacy/optional and consolidating the primary RFQAddress field. This aligns well with the V2 transition.

Let's verify no direct references to the old RFQAddressV2 field remain:

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for any remaining references to RFQAddressV2
rg "RFQAddressV2" --type go

Length of output: 620


Script:

#!/bin/bash
# Let's check the changes in the PR to see if these references are being updated
git diff HEAD^ HEAD -- services/rfq/guard/guardconfig/config.go services/rfq/guard/service/handlers.go services/rfq/guard/service/guard.go

Length of output: 923


Script:

#!/bin/bash
# Let's check if there are any other files in the PR that might be updating these references
git diff --name-only HEAD^ HEAD | grep -i rfq

Length of output: 88

// Confirmations is the number of required confirmations.
Confirmations uint64 `yaml:"confirmations"`
}
Expand Down Expand Up @@ -70,17 +70,17 @@
for chainID := range c.Chains {
addrV1, err := c.GetRFQAddressV1(chainID)
if err != nil {
return fmt.Errorf("could not get rfq address: %w", err)
return fmt.Errorf("could not get v1 rfq address: %w", err)

Check warning on line 73 in services/rfq/guard/guardconfig/config.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/guard/guardconfig/config.go#L73

Added line #L73 was not covered by tests
}
if addrV1 != nil && !common.IsHexAddress(*addrV1) {
return fmt.Errorf("invalid rfq address: %s", *addrV1)
return fmt.Errorf("invalid rfq v1 address: %s", *addrV1)

Check warning on line 76 in services/rfq/guard/guardconfig/config.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/guard/guardconfig/config.go#L76

Added line #L76 was not covered by tests
}
addrV2, err := c.GetRFQAddressV2(chainID)
if err != nil {
return fmt.Errorf("could not get rfq address: %w", err)
return fmt.Errorf("could not get v2 rfq address: %w", err)

Check warning on line 80 in services/rfq/guard/guardconfig/config.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/guard/guardconfig/config.go#L80

Added line #L80 was not covered by tests
}
if !common.IsHexAddress(addrV2) {
return fmt.Errorf("invalid rfq address: %s", addrV2)
return fmt.Errorf("invalid rfq v2 address: %s", addrV2)

Check warning on line 83 in services/rfq/guard/guardconfig/config.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/guard/guardconfig/config.go#L83

Added line #L83 was not covered by tests
}
}

Expand All @@ -96,7 +96,7 @@
func (c Config) GetRFQAddressV1(chainID int) (*string, error) {
chainCfg, ok := c.Chains[chainID]
if !ok {
return nil, fmt.Errorf("chain config not found for chain %d", chainID)
return nil, fmt.Errorf("v1 chain config not found for chain %d", chainID)

Check warning on line 99 in services/rfq/guard/guardconfig/config.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/guard/guardconfig/config.go#L99

Added line #L99 was not covered by tests
}
return chainCfg.RFQAddressV1, nil
}
Expand All @@ -105,9 +105,9 @@
func (c Config) GetRFQAddressV2(chainID int) (string, error) {
chainCfg, ok := c.Chains[chainID]
if !ok {
return "", fmt.Errorf("chain config not found for chain %d", chainID)
return "", fmt.Errorf("v2 chain config not found for chain %d", chainID)

Check warning on line 108 in services/rfq/guard/guardconfig/config.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/guard/guardconfig/config.go#L108

Added line #L108 was not covered by tests
}
return chainCfg.RFQAddressV2, nil
return chainCfg.RFQAddress, nil

Check warning on line 110 in services/rfq/guard/guardconfig/config.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/guard/guardconfig/config.go#L110

Added line #L110 was not covered by tests
Comment on lines +108 to +110
Copy link
Contributor

Choose a reason for hiding this comment

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

🛠️ Refactor suggestion

Consider renaming GetRFQAddressV2.

Since this method now returns the main RFQAddress and not a V2-specific address, the method name is misleading.

Consider this change:

-func (c Config) GetRFQAddressV2(chainID int) (string, error) {
+func (c Config) GetRFQAddress(chainID int) (string, error) {

Committable suggestion skipped: line range outside the PR's diff.

}

const defaultDBSelectorIntervalSeconds = 1
Expand All @@ -134,7 +134,7 @@
for chainID, chainCfg := range relayerCfg.GetChains() {
cfg.Chains[chainID] = ChainConfig{
RFQAddressV1: chainCfg.RFQAddressV1,
RFQAddressV2: chainCfg.RFQAddress,
RFQAddress: chainCfg.RFQAddress,

Check warning on line 137 in services/rfq/guard/guardconfig/config.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/guard/guardconfig/config.go#L137

Added line #L137 was not covered by tests
Confirmations: chainCfg.FinalityConfirmations,
}
}
Expand Down
4 changes: 2 additions & 2 deletions services/rfq/guard/guarddb/base/model.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@
"github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/common/hexutil"
"github.com/synapsecns/sanguine/core/dbcommon"
"github.com/synapsecns/sanguine/services/rfq/contracts/fastbridge"
"github.com/synapsecns/sanguine/services/rfq/contracts/fastbridgev2"
"github.com/synapsecns/sanguine/services/rfq/guard/guarddb"
)

Expand Down Expand Up @@ -168,7 +168,7 @@
return &guarddb.BridgeRequest{
TransactionID: transactionID,
RawRequest: req,
Transaction: fastbridge.IFastBridgeBridgeTransaction{
Transaction: fastbridgev2.IFastBridgeBridgeTransaction{

Check warning on line 171 in services/rfq/guard/guarddb/base/model.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/guard/guarddb/base/model.go#L171

Added line #L171 was not covered by tests
OriginChainId: b.OriginChainID,
DestChainId: b.DestChainID,
OriginSender: common.HexToAddress(b.OriginSender),
Expand Down
4 changes: 2 additions & 2 deletions services/rfq/guard/guarddb/db.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ import (
"fmt"

"github.com/synapsecns/sanguine/ethergo/listener/db"
"github.com/synapsecns/sanguine/services/rfq/contracts/fastbridge"
"github.com/synapsecns/sanguine/services/rfq/contracts/fastbridgev2"

"github.com/ethereum/go-ethereum/common"
"github.com/synapsecns/sanguine/core/dbcommon"
Expand Down Expand Up @@ -53,7 +53,7 @@ type Service interface {
// BridgeRequest is the bridge request object.
type BridgeRequest struct {
TransactionID [32]byte
Transaction fastbridge.IFastBridgeBridgeTransaction
Transaction fastbridgev2.IFastBridgeBridgeTransaction
RawRequest []byte
}

Expand Down
96 changes: 7 additions & 89 deletions services/rfq/guard/service/guard.go
Original file line number Diff line number Diff line change
Expand Up @@ -107,7 +107,7 @@
if err != nil {
return nil, fmt.Errorf("could not get deploy block: %w", err)
}
chainListener, err := listener.NewChainListener(chainClient, store, common.HexToAddress(rfqAddrV2), uint64(startBlock.Int64()), metricHandler, listener.WithName("guard"))
chainListener, err := listener.NewChainListener(chainClient, store, common.HexToAddress(rfqAddrV2), uint64(startBlock.Int64()), metricHandler, listener.WithName("guardV2"))

Check warning on line 110 in services/rfq/guard/service/guard.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/guard/service/guard.go#L110

Added line #L110 was not covered by tests
if err != nil {
return nil, fmt.Errorf("could not get chain listener: %w", err)
}
Expand Down Expand Up @@ -210,7 +210,7 @@
}
if v1Addr != nil {
group.Go(func() error {
err := g.runChainIndexerV1(ctx, chainID)
err := g.runChainIndexer(ctx, chainID, g.listenersV1[chainID])

Check warning on line 213 in services/rfq/guard/service/guard.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/guard/service/guard.go#L213

Added line #L213 was not covered by tests
if err != nil {
return fmt.Errorf("could not runChainIndexer chain indexer for chain %d [v1]: %w", chainID, err)
}
Expand All @@ -219,7 +219,7 @@
}

group.Go(func() error {
err := g.runChainIndexerV2(ctx, chainID)
err := g.runChainIndexer(ctx, chainID, g.listenersV2[chainID])

Check warning on line 222 in services/rfq/guard/service/guard.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/guard/service/guard.go#L222

Added line #L222 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.

⚠️ Potential issue

Check for nil listener before starting V2 chain indexer

The code starts the V2 chain indexer without verifying if the listener exists. If g.listenersV2[chainID] is nil, this could result in a runtime error. Consider adding a check to ensure the listener is initialized before starting the indexer.

You can add a nil check before starting the chain indexer:

if g.listenersV2[chainID] != nil {
    group.Go(func() error {
        err := g.runChainIndexer(ctx, chainID, g.listenersV2[chainID])
        if err != nil {
            return fmt.Errorf("could not runChainIndexer chain indexer for chain %d [v2]: %w", chainID, err)
        }
        return nil
    })
}

if err != nil {
return fmt.Errorf("could not runChainIndexer chain indexer for chain %d [v2]: %w", chainID, err)
}
Expand All @@ -236,66 +236,7 @@
}

//nolint:cyclop
func (g Guard) runChainIndexerV1(ctx context.Context, chainID int) (err error) {
chainListener := g.listenersV1[chainID]

parser, err := fastbridge.NewParser(chainListener.Address())
if err != nil {
return fmt.Errorf("could not parse: %w", err)
}

err = chainListener.Listen(ctx, func(parentCtx context.Context, log types.Log) (err error) {
et, parsedEvent, ok := parser.ParseEvent(log)
// handle unknown event
if !ok {
if len(log.Topics) != 0 {
logger.Warnf("unknown event %s", log.Topics[0])
}
return nil
}

ctx, span := g.metrics.Tracer().Start(parentCtx, fmt.Sprintf("handleLog-%s", et), trace.WithAttributes(
attribute.String(metrics.TxHash, log.TxHash.String()),
attribute.Int(metrics.Origin, chainID),
attribute.String(metrics.Contract, log.Address.String()),
attribute.String("block_hash", log.BlockHash.String()),
attribute.Int64("block_number", int64(log.BlockNumber)),
))

defer func() {
metrics.EndSpanWithErr(span, err)
}()

switch event := parsedEvent.(type) {
case *fastbridge.FastBridgeBridgeRequested:
err = g.handleBridgeRequestedLog(ctx, event, chainID)
if err != nil {
return fmt.Errorf("could not handle request: %w", err)
}
case *fastbridge.FastBridgeBridgeProofProvided:
err = g.handleProofProvidedLog(ctx, event, chainID)
if err != nil {
return fmt.Errorf("could not handle request: %w", err)
}
case *fastbridge.FastBridgeBridgeProofDisputed:
err = g.handleProofDisputedLog(ctx, event)
if err != nil {
return fmt.Errorf("could not handle request: %w", err)
}
}

return nil
})
if err != nil {
return fmt.Errorf("listener failed: %w", err)
}
return nil
}

//nolint:cyclop,gosec
func (g Guard) runChainIndexerV2(ctx context.Context, chainID int) (err error) {
chainListener := g.listenersV2[chainID]

func (g Guard) runChainIndexer(ctx context.Context, chainID int, chainListener listener.ContractListener) (err error) {

Check warning on line 239 in services/rfq/guard/service/guard.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/guard/service/guard.go#L239

Added line #L239 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.

⚠️ Potential issue

Add defensive nil check for chainListener

The function should verify that chainListener is not nil before proceeding to prevent potential runtime panics.

 func (g Guard) runChainIndexer(ctx context.Context, chainID int, chainListener listener.ContractListener) (err error) {
+    if chainListener == nil {
+        return fmt.Errorf("chain listener is nil for chain %d", chainID)
+    }
     parser, err := fastbridgev2.NewParser(chainListener.Address())
📝 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
func (g Guard) runChainIndexer(ctx context.Context, chainID int, chainListener listener.ContractListener) (err error) {
func (g Guard) runChainIndexer(ctx context.Context, chainID int, chainListener listener.ContractListener) (err error) {
if chainListener == nil {
return fmt.Errorf("chain listener is nil for chain %d", chainID)
}
parser, err := fastbridgev2.NewParser(chainListener.Address())
🧰 Tools
🪛 GitHub Check: codecov/patch

[warning] 239-239: services/rfq/guard/service/guard.go#L239
Added line #L239 was not covered by tests

parser, err := fastbridgev2.NewParser(chainListener.Address())
if err != nil {
return fmt.Errorf("could not parse: %w", err)
Expand Down Expand Up @@ -325,40 +266,17 @@

switch event := parsedEvent.(type) {
case *fastbridgev2.FastBridgeV2BridgeRequested:
v1Event := &fastbridge.FastBridgeBridgeRequested{
TransactionId: event.TransactionId,
Sender: event.Sender,
Request: event.Request,
DestChainId: event.DestChainId,
OriginToken: event.OriginToken,
DestToken: event.DestToken,
DestAmount: event.DestAmount,
SendChainGas: event.SendChainGas,
Raw: event.Raw,
}
err = g.handleBridgeRequestedLog(ctx, v1Event, chainID)
err = g.handleBridgeRequestedLog(ctx, event, chainID)

Check warning on line 269 in services/rfq/guard/service/guard.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/guard/service/guard.go#L269

Added line #L269 was not covered by tests
if err != nil {
return fmt.Errorf("could not handle request: %w", err)
}
// following events match ABIs exactly, so no need to differentiate
case *fastbridgev2.FastBridgeV2BridgeProofProvided:
v1Event := &fastbridge.FastBridgeBridgeProofProvided{
TransactionId: event.TransactionId,
Relayer: event.Relayer,
TransactionHash: event.TransactionHash,
Raw: event.Raw,
}
err = g.handleProofProvidedLog(ctx, v1Event, chainID)
err = g.handleProofProvidedLog(ctx, event, chainID)

Check warning on line 274 in services/rfq/guard/service/guard.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/guard/service/guard.go#L274

Added line #L274 was not covered by tests
if err != nil {
return fmt.Errorf("could not handle request: %w", err)
}
case *fastbridgev2.FastBridgeV2BridgeProofDisputed:
v1Event := &fastbridge.FastBridgeBridgeProofDisputed{
TransactionId: event.TransactionId,
Relayer: event.Relayer,
Raw: event.Raw,
}
err = g.handleProofDisputedLog(ctx, v1Event)
err = g.handleProofDisputedLog(ctx, event)

Check warning on line 279 in services/rfq/guard/service/guard.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/guard/service/guard.go#L279

Added line #L279 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.

⚠️ Potential issue

Inconsistent parameters in handleProofDisputedLog call

Unlike handleBridgeRequestedLog and handleProofProvidedLog, the handleProofDisputedLog function is called without the chainID parameter. If chainID is required in handleProofDisputedLog, consider adding it to maintain consistency and ensure correct processing.

Apply this diff to include chainID in the function call:

-	err = g.handleProofDisputedLog(ctx, event)
+	err = g.handleProofDisputedLog(ctx, event, chainID)

Also, update the handleProofDisputedLog function signature to accept chainID if necessary.

Committable suggestion skipped: line range outside the PR's diff.

🧰 Tools
🪛 GitHub Check: codecov/patch

[warning] 279-279: services/rfq/guard/service/guard.go#L279
Added line #L279 was not covered by tests

if err != nil {
return fmt.Errorf("could not handle request: %w", err)
}
Comment on lines +269 to 282
Copy link
Contributor

Choose a reason for hiding this comment

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

⚠️ Potential issue

Standardize event handler parameters and improve error messages.

The event handling code has two issues:

  1. handleProofDisputedLog is missing the chainID parameter unlike other handlers
  2. Error messages are generic and don't differentiate between event types

Apply this diff:

 case *fastbridgev2.FastBridgeV2BridgeRequested:
     err = g.handleBridgeRequestedLog(ctx, event, chainID)
     if err != nil {
-        return fmt.Errorf("could not handle request: %w", err)
+        return fmt.Errorf("could not handle bridge request event for chain %d: %w", chainID, err)
     }
 case *fastbridgev2.FastBridgeV2BridgeProofProvided:
     err = g.handleProofProvidedLog(ctx, event, chainID)
     if err != nil {
-        return fmt.Errorf("could not handle request: %w", err)
+        return fmt.Errorf("could not handle proof provided event for chain %d: %w", chainID, err)
     }
 case *fastbridgev2.FastBridgeV2BridgeProofDisputed:
-    err = g.handleProofDisputedLog(ctx, event)
+    err = g.handleProofDisputedLog(ctx, event, chainID)
     if err != nil {
-        return fmt.Errorf("could not handle request: %w", err)
+        return fmt.Errorf("could not handle proof disputed event for chain %d: %w", chainID, 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 = g.handleBridgeRequestedLog(ctx, event, chainID)
if err != nil {
return fmt.Errorf("could not handle request: %w", err)
}
// following events match ABIs exactly, so no need to differentiate
case *fastbridgev2.FastBridgeV2BridgeProofProvided:
v1Event := &fastbridge.FastBridgeBridgeProofProvided{
TransactionId: event.TransactionId,
Relayer: event.Relayer,
TransactionHash: event.TransactionHash,
Raw: event.Raw,
}
err = g.handleProofProvidedLog(ctx, v1Event, chainID)
err = g.handleProofProvidedLog(ctx, event, chainID)
if err != nil {
return fmt.Errorf("could not handle request: %w", err)
}
case *fastbridgev2.FastBridgeV2BridgeProofDisputed:
v1Event := &fastbridge.FastBridgeBridgeProofDisputed{
TransactionId: event.TransactionId,
Relayer: event.Relayer,
Raw: event.Raw,
}
err = g.handleProofDisputedLog(ctx, v1Event)
err = g.handleProofDisputedLog(ctx, event)
if err != nil {
return fmt.Errorf("could not handle request: %w", err)
}
err = g.handleBridgeRequestedLog(ctx, event, chainID)
if err != nil {
return fmt.Errorf("could not handle bridge request event for chain %d: %w", chainID, err)
}
case *fastbridgev2.FastBridgeV2BridgeProofProvided:
err = g.handleProofProvidedLog(ctx, event, chainID)
if err != nil {
return fmt.Errorf("could not handle proof provided event for chain %d: %w", chainID, err)
}
case *fastbridgev2.FastBridgeV2BridgeProofDisputed:
err = g.handleProofDisputedLog(ctx, event, chainID)
if err != nil {
return fmt.Errorf("could not handle proof disputed event for chain %d: %w", chainID, err)
}
🧰 Tools
🪛 GitHub Check: codecov/patch

[warning] 269-269: services/rfq/guard/service/guard.go#L269
Added line #L269 was not covered by tests


[warning] 274-274: services/rfq/guard/service/guard.go#L274
Added line #L274 was not covered by tests


[warning] 279-279: services/rfq/guard/service/guard.go#L279
Added line #L279 was not covered by tests

Expand Down
Loading
Loading