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

feat(rfq-guard): v2 guard logic [SLT-387] #3324

Merged
merged 18 commits into from
Nov 26, 2024
Merged

Conversation

dwasse
Copy link
Collaborator

@dwasse dwasse commented Oct 22, 2024

Summary by CodeRabbit

  • New Features

    • Support for multiple RFQ bridge contract addresses with separate handling for versions V1 and V2.
    • Enhanced event handling for the FastBridge contract, including new dispute methods for both contract versions.
    • Integration of FastBridgeV2, including updated methods and structures for improved transaction handling.
  • Bug Fixes

    • Improved validation logic for RFQ addresses to ensure accurate processing.
  • Documentation

    • Updated comments in configuration structures to clarify the purpose of new fields.

These changes enhance the system's flexibility and reliability in managing bridge contracts and their associated processes.

Copy link
Contributor

coderabbitai bot commented Oct 22, 2024

Walkthrough

The pull request introduces significant changes to the RFQ (Request for Quote) functionality across several components. Key modifications include the restructuring of the ChainConfig to support multiple RFQ addresses, the addition of a FastBridgeAddress field in various models, and enhancements to the Guard service to manage dual contract versions. New methods and updated validation logic accommodate these changes, ensuring compatibility with both versions of the FastBridge contract while maintaining existing functionality.

Changes

File Path Change Summary
services/rfq/guard/guardconfig/config.go - Updated ChainConfig: replaced RFQAddress with RFQAddressV1 (*string) and added RFQAddressV2 (string).
- Renamed GetRFQAddress to GetRFQAddressV1 and added GetRFQAddressV2.
- Updated Validate method to validate both RFQ addresses.
services/rfq/guard/guarddb/base/model.go - Added FastBridgeAddress field to PendingProvenModel struct.
- Updated FromPendingProven to populate the new field.
services/rfq/guard/guarddb/db.go - Added FastBridgeAddress field to PendingProven struct.
services/rfq/guard/service/guard.go - Updated Guard struct to include separate maps for contracts and listeners for both versions.
- Modified NewGuard to initialize new maps.
- Introduced isV2Address method and modified runChainIndexer to handle both contract versions.
services/rfq/guard/service/handlers.go - Updated methods to support FastBridgeV2, including changes to handle proof and dispute logs for both versions.
- Introduced disputeV1 and disputeV2 methods.
services/rfq/relayer/relconfig/config.go - Renamed RFQAddress with an updated comment and added RFQAddressV1 as an optional string pointer.
services/rfq/e2e/rfq_test.go - Updated tests to use GetFastBridgeV2 and added new dispute tests.
- Changed method signatures to reflect the new FastBridge functionality.
services/rfq/e2e/setup_test.go - Updated setup methods to reference FastBridgeV2Type and handle both FastBridge versions.
services/rfq/testutil/contracttype.go - Added FastBridgeV2Type to contract type enumeration and updated related logic.
services/rfq/testutil/contracttypeimpl_string.go - Updated constant names and indices to include FastBridgeV2Type.
services/rfq/testutil/deployers.go - Introduced FastBridgeV2Deployer and updated deployment logic to support the new contract.
services/rfq/testutil/typecast.go - Updated GetFastBridge method and added GetFastBridgeV2 method for retrieving the new contract.

Possibly related PRs

Suggested labels

size/l

Suggested reviewers

  • aureliusbtc
  • trajan0x
  • ChiTimesChi

Poem

🐇 In the garden where the RFQ blooms,
New addresses sprout, dispelling the glooms.
With bridges fast and contracts anew,
The Guard stands strong, ready to pursue.
Hopping through code, with a joyful cheer,
Celebrating changes, we hold so dear! 🌼


Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media?

❤️ Share
🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

  • Review comments: Directly reply to a review comment made by CodeRabbit. Example:
    • I pushed a fix in commit <commit_id>, please review it.
    • Generate unit testing code for this file.
    • Open a follow-up GitHub issue for this discussion.
  • Files and specific lines of code (under the "Files changed" tab): Tag @coderabbitai in a new review comment at the desired location with your query. Examples:
    • @coderabbitai generate unit testing code for this file.
    • @coderabbitai modularize this function.
  • PR comments: Tag @coderabbitai in a new PR comment to ask questions about the PR branch. For the best results, please provide a very specific query, as very limited context is provided in this mode. Examples:
    • @coderabbitai gather interesting stats about this repository and render them as a table. Additionally, render a pie chart showing the language distribution in the codebase.
    • @coderabbitai read src/utils.ts and generate unit testing code.
    • @coderabbitai read the files in the src/scheduler package and generate a class diagram using mermaid and a README in the markdown format.
    • @coderabbitai help me debug CodeRabbit configuration file.

Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments.

CodeRabbit Commands (Invoked using PR comments)

  • @coderabbitai pause to pause the reviews on a PR.
  • @coderabbitai resume to resume the paused reviews.
  • @coderabbitai review to trigger an incremental review. This is useful when automatic reviews are disabled for the repository.
  • @coderabbitai full review to do a full review from scratch and review all the files again.
  • @coderabbitai summary to regenerate the summary of the PR.
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai configuration to show the current CodeRabbit configuration for the repository.
  • @coderabbitai help to get help.

Other keywords and placeholders

  • Add @coderabbitai ignore anywhere in the PR description to prevent this PR from being reviewed.
  • Add @coderabbitai summary to generate the high-level summary at a specific location in the PR description.
  • Add @coderabbitai anywhere in the PR title to generate the title automatically.

Documentation and Community

  • Visit our Documentation for detailed information on how to use CodeRabbit.
  • Join our Discord Community to get help, request features, and share feedback.
  • Follow us on X/Twitter for updates and announcements.

@github-actions github-actions bot added go Pull requests that update Go code size/l labels Oct 22, 2024
@dwasse dwasse changed the base branch from master to feat/relayer-arb-call October 22, 2024 17:37
Copy link

codecov bot commented Oct 22, 2024

Codecov Report

All modified and coverable lines are covered by tests ✅

Project coverage is 31.25438%. Comparing base (2132369) to head (4fe0be0).
Report is 1 commits behind head on feat/relayer-arb-call.

Additional details and impacted files
@@                    Coverage Diff                    @@
##           feat/relayer-arb-call       #3324   +/-   ##
=========================================================
  Coverage               31.25438%   31.25438%           
=========================================================
  Files                        107         107           
  Lines                       4281        4281           
  Branches                      82          82           
=========================================================
  Hits                        1338        1338           
  Misses                      2937        2937           
  Partials                       6           6           
Flag Coverage Δ
opbot 0.18282% <ø> (ø)
packages 90.44834% <ø> (ø)
promexporter 6.81642% <ø> (ø)
solidity 98.76161% <ø> (ø)

Flags with carried forward coverage won't be shown. Click here to find out more.

☔ View full report in Codecov by Sentry.
📢 Have feedback on the report? Share it here.

Copy link

cloudflare-workers-and-pages bot commented Oct 22, 2024

Deploying sanguine-fe with  Cloudflare Pages  Cloudflare Pages

Latest commit: a677ffe
Status: ✅  Deploy successful!
Preview URL: https://27c90a1c.sanguine-fe.pages.dev
Branch Preview URL: https://feat-guard-v2.sanguine-fe.pages.dev

View logs

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Caution

Inline review comments failed to post. This is likely due to GitHub's limits when posting large numbers of comments.

🛑 Comments failed to post (32)
services/rfq/contracts/testcontracts/recipientmock/recipientmock.metadata.go (1)

18-25: 🛠️ Refactor suggestion

Enhance error handling and add data validation.

While panicking in init() is acceptable for critical startup requirements, consider:

  1. Adding more context to the error message
  2. Validating the unmarshalled data before proceeding

Here's a suggested improvement:

 func init() {
-	// load contract metadata
-	var err error
-	err = json.Unmarshal(rawContracts, &Contracts)
-	if err != nil {
-		panic(err)
-	}
+	// Load and validate contract metadata
+	if err := json.Unmarshal(rawContracts, &Contracts); err != nil {
+		panic(fmt.Errorf("failed to unmarshal recipientmock contract metadata: %w", err))
+	}
+	
+	// Validate contract data
+	if len(Contracts) == 0 {
+		panic("no contracts found in recipientmock metadata")
+	}
+	for name, contract := range Contracts {
+		if contract == nil {
+			panic(fmt.Sprintf("nil contract found in metadata: %s", name))
+		}
+	}
 }
📝 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.

func init() {
	// Load and validate contract metadata
	if err := json.Unmarshal(rawContracts, &Contracts); err != nil {
		panic(fmt.Errorf("failed to unmarshal recipientmock contract metadata: %w", err))
	}
	
	// Validate contract data
	if len(Contracts) == 0 {
		panic("no contracts found in recipientmock metadata")
	}
	for name, contract := range Contracts {
		if contract == nil {
			panic(fmt.Sprintf("nil contract found in metadata: %s", name))
		}
	}
}
services/rfq/contracts/testcontracts/fastbridgemockv2/fastbridgemockv2.metadata.go (1)

18-25: 🛠️ Refactor suggestion

Consider improving error handling and adding data validation.

While panicking in init() is acceptable, consider these improvements:

  1. More descriptive error message
  2. Basic validation of unmarshalled data

Consider this enhanced implementation:

 func init() {
-	// load contract metadata
-	var err error
-	err = json.Unmarshal(rawContracts, &Contracts)
-	if err != nil {
-		panic(err)
-	}
+	// Initialize contract metadata from embedded JSON
+	if err := json.Unmarshal(rawContracts, &Contracts); err != nil {
+		panic(fmt.Errorf("failed to unmarshal contract metadata: %w", err))
+	}
+	
+	// Validate contract metadata
+	if len(Contracts) == 0 {
+		panic("no contracts found in metadata")
+	}
+	
+	for name, contract := range Contracts {
+		if contract == nil || len(contract.Info.AbiDefinition) == 0 {
+			panic(fmt.Errorf("invalid contract metadata for %s: missing ABI", name))
+		}
+	}
📝 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.

func init() {
	// Initialize contract metadata from embedded JSON
	if err := json.Unmarshal(rawContracts, &Contracts); err != nil {
		panic(fmt.Errorf("failed to unmarshal contract metadata: %w", err))
	}
	
	// Validate contract metadata
	if len(Contracts) == 0 {
		panic("no contracts found in metadata")
	}
	
	for name, contract := range Contracts {
		if contract == nil || len(contract.Info.AbiDefinition) == 0 {
			panic(fmt.Errorf("invalid contract metadata for %s: missing ABI", name))
		}
	}
}
packages/contracts-rfq/package.json (1)

27-27: 💡 Codebase verification

Warning: Missing flatten.sh script in the expected location

The build:go script in package.json references flatten.sh, but this script is missing in the root of the packages/contracts-rfq directory. The repository contains other flatten scripts (e.g., in make/scripts/flatten.sh and packages/contracts-core/script/sh/flatten.sh), but none at the expected location.

  • The script should be copied or symlinked to packages/contracts-rfq/flatten.sh
  • The test/mocks directory exists and contains Solidity files that need to be included in the flattening process
🔗 Analysis chain

LGTM! Verify flatten.sh script.

The addition of mock contracts to the build process is appropriate for testing the new FastBridge v2 functionality.

Let's verify the flatten.sh script exists and has appropriate permissions:

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Check if flatten.sh exists and is executable
if [ -x "./flatten.sh" ]; then
    echo "flatten.sh exists and is executable"
else
    echo "Warning: flatten.sh might be missing or not executable"
fi

# Verify the mock contracts directory exists
fd -t d "mocks" test/

Length of output: 259


Script:

#!/bin/bash
# Let's first check if the file exists, regardless of permissions
ls -la flatten.sh 2>/dev/null || echo "flatten.sh not found"

# Check the directory structure to understand where we are
ls -la

# Look for flatten.sh in the entire repository
fd "flatten.sh"

# Check if test directory exists and its structure
ls -la test/ 2>/dev/null || echo "test directory not found"

# Search for .sol files in the repository to verify the paths
fd -e sol

Length of output: 19892


Script:

#!/bin/bash
# Let's check the content of the flatten.sh in contracts-rfq package
cat packages/contracts-rfq/flatten.sh 2>/dev/null || echo "File not found in expected location"

# Let's also check if the test and mocks directories exist in contracts-rfq
ls -la packages/contracts-rfq/test/ 2>/dev/null
ls -la packages/contracts-rfq/test/mocks/ 2>/dev/null

# Check for .sol files in test/mocks directory
fd -e sol . packages/contracts-rfq/test/mocks/

Length of output: 4665

services/rfq/contracts/testcontracts/fastbridgemockv2/helper.go (1)

23-33: 🛠️ Refactor suggestion

Add input validation for the address parameter.

Consider validating that the provided address is not zero to prevent initialization with an invalid contract address.

 func NewFastBridgeMockRef(address common.Address, backend bind.ContractBackend) (*FastBridgeMockRef, error) {
+	if address == (common.Address{}) {
+		return nil, errors.New("invalid contract address: zero address")
+	}
 	fastbridgemock, err := NewFastBridgeMock(address, backend)

Committable suggestion was skipped due to low confidence.

services/rfq/contracts/fastbridgev2/status.go (1)

27-36: ⚠️ Potential issue

Add validation in init function.

The init function lacks validation and has an incomplete assertion comment. Consider adding proper validation to ensure the integrity of the bridge statuses.

 func init() {
+	expectedCount := 5 // Number of defined statuses
 	for i := 0; i < len(_BridgeStatus_index)-1; i++ {
 		contractType := BridgeStatus(i)
+		// Validate that the status is within expected range
+		if contractType > REFUNDED {
+			panic("invalid bridge status encountered during initialization")
+		}
 		allBridgeStatuses = append(allBridgeStatuses, contractType)
-		// assert type is correct
 	}
+	if len(allBridgeStatuses) != expectedCount {
+		panic("unexpected number of bridge statuses")
+	}
 }

Committable suggestion was skipped due to low confidence.

services/rfq/contracts/fastbridgev2/status_test.go (2)

37-37: 🛠️ Refactor suggestion

Consider making chainID configurable.

The chainID is hardcoded to 1 (mainnet). Consider making it configurable through a constant or test parameter to support testing against different networks.

+const defaultTestChainID = 1
+
 func (s *FastBridgeSuite) SetupTest() {
 	s.TestSuite.SetupTest()
-	s.backend = simulated.NewSimulatedBackendWithChainID(s.GetTestContext(), s.T(), big.NewInt(1))
+	s.backend = simulated.NewSimulatedBackendWithChainID(s.GetTestContext(), s.T(), big.NewInt(defaultTestChainID))

Committable suggestion was skipped due to low confidence.


41-49: 🛠️ Refactor suggestion

Enhance test robustness and cleanup.

The test could benefit from some improvements:

  1. More descriptive error messages
  2. Proper cleanup of resources
  3. Consider adding test cases for invalid status values

Here's a suggested improvement:

 func (s *FastBridgeSuite) TestStatusEnum() {
+	defer s.backend.Close()
+
 	_, fb := s.manager.GetMockFastBridge(s.GetTestContext(), s.backend)
 	for _, status := range fastbridgev2.GetAllBridgeStatuses() {
 		solstatus, err := fb.GetEnumKeyByValue(&bind.CallOpts{Context: s.GetTestContext()}, status.Int())
-		s.Require().NoError(err, "error getting enum key by value")
-		s.Require().Equal(solstatus, status.String(), "status %s does not match. BridgeStatus enums out of sync.", status)
+		s.Require().NoError(err, "failed to get enum key for status value %d", status.Int())
+		s.Require().Equal(solstatus, status.String(), 
+			"status enum mismatch: got %s, want %s (value: %d)", 
+			solstatus, status.String(), status.Int())
 	}
+
+	// Test invalid status
+	invalidStatus := big.NewInt(999999)
+	_, err := fb.GetEnumKeyByValue(&bind.CallOpts{Context: s.GetTestContext()}, invalidStatus)
+	s.Require().Error(err, "expected error for invalid status value %d", invalidStatus)
 }
📝 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.

// TestStatusEnum makes sure.
func (s *FastBridgeSuite) TestStatusEnum() {
	defer s.backend.Close()

	_, fb := s.manager.GetMockFastBridge(s.GetTestContext(), s.backend)
	for _, status := range fastbridgev2.GetAllBridgeStatuses() {
		solstatus, err := fb.GetEnumKeyByValue(&bind.CallOpts{Context: s.GetTestContext()}, status.Int())
		s.Require().NoError(err, "failed to get enum key for status value %d", status.Int())
		s.Require().Equal(solstatus, status.String(), 
			"status enum mismatch: got %s, want %s (value: %d)", 
			solstatus, status.String(), status.Int())
	}

	// Test invalid status
	invalidStatus := big.NewInt(999999)
	_, err := fb.GetEnumKeyByValue(&bind.CallOpts{Context: s.GetTestContext()}, invalidStatus)
	s.Require().Error(err, "expected error for invalid status value %d", invalidStatus)
}
services/rfq/relayer/reldb/base/model_test.go (1)

26-40: 💡 Codebase verification

Update test coverage for V2 transaction fields across test files

Based on the verification results, I found that:

  1. services/rfq/relayer/limiter/limiter_test.go needs to be updated as it's not setting the new V2 fields at all
  2. services/rfq/relayer/relapi/server_test.go is using zero values for the new fields, similar to the file under review
  3. services/rfq/relayer/quoter/quoter_test.go needs to be updated as it's not setting the new V2 fields

The original review comment's suggestion to improve test coverage is valid and should be extended to these additional files:

  • limiter_test.go and quoter_test.go should be updated to include the new V2 fields
  • All test files should use meaningful non-zero values for better test coverage, especially for:
    • ExclusivityEndTime (using future timestamps)
    • OriginFeeAmount (using non-zero amounts)
    • CallValue (using non-zero values)
    • CallParams (using non-empty byte arrays)
🔗 Analysis chain

Enhance test coverage for new V2 fields

While the basic roundtrip test is implemented, consider improving the test coverage:

  1. The new V2 fields (ExclusivityEndTime, OriginFeeAmount, CallValue, CallParams) are currently initialized with zero values, which might not catch potential serialization issues.
  2. Add test cases with non-zero values, especially for CallParams which is a byte array.

Let's check if there are other test files that might need similar updates:

Here's a suggested improvement to make the test more robust:

 Transaction: fastbridgev2.IFastBridgeV2BridgeTransactionV2{
   OriginChainId:      1,
   DestChainId:        2,
   OriginSender:       common.Address{},
   DestRecipient:      common.Address{},
   OriginToken:        common.Address{},
   DestToken:          common.Address{},
   OriginAmount:       big.NewInt(1000),
   DestAmount:         big.NewInt(2000),
   Deadline:           big.NewInt(time.Now().Unix()),
   Nonce:              big.NewInt(1),
-  ExclusivityEndTime: big.NewInt(0),
-  OriginFeeAmount:    big.NewInt(0),
-  CallValue:          big.NewInt(0),
-  CallParams:         []byte{},
+  ExclusivityEndTime: big.NewInt(time.Now().Add(1 * time.Hour).Unix()),
+  OriginFeeAmount:    big.NewInt(100),
+  CallValue:          big.NewInt(50),
+  CallParams:         []byte{0x1, 0x2, 0x3}, // Add some test data
 },

Let me check the usage of FastBridge transaction in these test files to verify if they need similar updates.


Let me fix the regex and try again to check the transaction usage in other test files.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Find other test files that might use the FastBridge transaction type
rg -l "IFastBridge.*Transaction" --type go --glob "*_test.go"

Length of output: 240


Script:

#!/bin/bash
# Check how FastBridge transaction is used in other test files
for file in services/rfq/relayer/limiter/limiter_test.go services/rfq/relayer/relapi/server_test.go services/rfq/relayer/quoter/quoter_test.go; do
    echo "=== $file ==="
    rg "IFastBridge.*Transaction.*{" -A 15 "$file"
done

Length of output: 1379


Script:

#!/bin/bash
# Check how FastBridge transaction is used in other test files
for file in services/rfq/relayer/limiter/limiter_test.go services/rfq/relayer/relapi/server_test.go services/rfq/relayer/quoter/quoter_test.go; do
    echo "=== $file ==="
    rg "IFastBridgeV2BridgeTransactionV2" -A 15 "$file"
done

Length of output: 4738

services/rfq/testutil/typecast.go (2)

34-38: 💡 Codebase verification

Potential inconsistency in FastBridge version usage

Based on the verification results:

  • The relayer service tests correctly use fastbridgemockv2
  • However, both fastbridge/status_test.go and fastbridgev2/status_test.go appear to be using the same GetMockFastBridge method but with different package imports (fastbridge vs fastbridgev2)

This suggests a potential version mismatch that needs attention:

  • services/rfq/contracts/fastbridge/status_test.go might need updating to use v1-specific mock bridge
  • services/rfq/contracts/fastbridgev2/status_test.go is correctly using v2 imports
🔗 Analysis chain

Verify test cases using GetMockFastBridge are updated for v2

The mock bridge changes correctly align with the v2 upgrade.


Let me gather more information to verify if the test cases are properly updated for v2.


Let me check one more thing to verify the imports and usage in the status test files.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for GetMockFastBridge usage in test files
rg -t go -l "GetMockFastBridge\(" | grep "_test.go" | xargs rg -A 2 "GetMockFastBridge\("

Length of output: 1804


Script:

#!/bin/bash
# Check imports in test files using GetMockFastBridge to verify they're using v2
rg -t go "GetMockFastBridge\(" | grep "_test.go" | cut -d: -f1 | xargs rg -l "fastbridgemock"

Length of output: 263


Script:

#!/bin/bash
# Check the actual mock bridge type being used in these test files
rg -t go "fastbridgemock" services/rfq/contracts/fastbridge/status_test.go services/rfq/contracts/fastbridgev2/status_test.go services/rfq/relayer/service/suite_test.go

Length of output: 697


Script:

#!/bin/bash
# Check imports in status test files
rg -t go "^import \(" -A 10 services/rfq/contracts/fastbridge/status_test.go services/rfq/contracts/fastbridgev2/status_test.go

Length of output: 2051


20-23: 💡 Codebase verification

Issues found with FastBridge v2 migration

The verification reveals inconsistencies in the FastBridge version usage:

  • services/explorer/testutil/typecast.go and services/explorer/testutil/typecast_test.go are still using v1 FastBridge imports and types
  • services/rfq/e2e/rfq_test.go has mixed imports of both v1 and v2 FastBridge packages
  • services/rfq/relayer/relapi/suite_test.go is still using v1 FastBridge import

While the services/rfq/testutil/typecast.go correctly uses v2, the changes need to be propagated to all dependent services to maintain consistency.

🔗 Analysis chain

Verify callers of GetFastBridge are updated for v2

The method changes look good, correctly updating to the v2 contract reference.


Let me analyze the findings and generate additional verification scripts to ensure all callers are properly updated.

The first command will help verify the import statements to ensure they're using the v2 package, and the second command will show how the returned handle is being used in the code. This will help us confirm if all callers are properly updated for v2.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for GetFastBridge calls to ensure they handle the new v2 return type
rg -l "GetFastBridge\(" | xargs rg -A 2 "GetFastBridge\("

Length of output: 3652


Script:

#!/bin/bash
# Check the import statements in files using GetFastBridge to verify they're using v2
rg -l "GetFastBridge\(" | xargs rg "fastbridge.*\"$" 

# Check the type assertions or direct usage of the returned handle
rg -l "GetFastBridge\(" | xargs rg -A 3 "GetFastBridge.*\)\s*$"

Length of output: 4597

services/rfq/relayer/service/relayer.go (1)

102-102: 💡 Codebase verification

Based on the search results, I can now provide a final response as there's clear evidence of potential issues:

FastBridge V1 to V2 migration is incomplete

The codebase shows multiple components still using FastBridge V1 that need to be updated:

  • Guard service components:
    • services/rfq/guard/service/handlers.go
    • services/rfq/guard/service/guard.go
  • Explorer service components:
    • services/explorer/api/server.go
    • services/explorer/contracts/fastbridge/helpers.go
  • Other RFQ components:
    • services/rfq/relayer/chain/chain.go
    • services/rfq/relayer/relapi/server.go

While the relayer service has been updated to use FastBridgeV2, other dependent services are still using V1, which could lead to compatibility issues. This change requires a coordinated update across all services.

🔗 Analysis chain

Verify FastBridge V2 contract compatibility

The contract instantiation has been correctly updated to use FastBridgeV2Ref. However, let's verify the compatibility of the V2 contract with existing callers.

Let's check for any potential interface mismatches or unconverted V1 contract usage:


Let me gather more specific information about the V1 to V2 contract usage to better understand the impact.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check for potential V1/V2 contract compatibility issues

# Test 1: Search for any remaining V1 contract references
echo "Checking for remaining V1 contract references..."
rg -l "fastbridge\.FastBridge[^V]"

# Test 2: Search for V1-specific method calls that might need updates
echo "Checking for V1-specific method calls..."
rg -l "NewFastBridgeRef"

# Test 3: Check interface differences between V1 and V2
echo "Checking V1/V2 interface differences..."
ast-grep --pattern 'contract FastBridge {
  $$$
}'

echo "Checking V2 interface..."
ast-grep --pattern 'contract FastBridgeV2 {
  $$$
}'

Length of output: 1190


Script:

#!/bin/bash
# Let's examine the actual usage patterns and changes

# Check for specific V1 method calls that might need updates
echo "Checking V1 method call patterns..."
rg -A 2 "NewFastBridgeRef\(" 

# Look for any V2 migration patterns
echo "Checking V2 migration patterns..."
rg -A 2 "NewFastBridgeV2Ref\("

# Check the contract helper implementations
echo "Checking contract helper implementations..."
rg -A 5 "func NewFastBridge" services/rfq/contracts/fastbridge/

# Check for any potential interface mismatches
echo "Checking interface usage patterns..."
rg -A 3 "fastbridge\.FastBridge\." services/rfq/relayer/

Length of output: 7622

services/rfq/contracts/fastbridgev2/events.go (3)

11-23: 🛠️ Refactor suggestion

Consider unexporting variables to prevent accidental mutation

There is a TODO comment suggesting that the variables should not be exported to avoid accidental mutation. Since these variables are intended to be constants after initialization and should not be modified externally, consider unexporting them by renaming them to start with a lowercase letter or defining them as constants if appropriate.


76-79: ⚠️ Potential issue

Avoid returning the address of a loop variable

In Go, taking the address of a loop variable and returning it can lead to unexpected behavior because the loop variable eventType is reused in each iteration. The returned pointer may point to the last value assigned in the loop.

To fix this issue, create a new variable inside the loop to capture the current value of eventType and return its address.

Apply the following diff:

func eventTypeFromTopic(ogTopic common.Hash) *EventType {
	for eventType, topic := range topicMap() {
		if bytes.Equal(ogTopic.Bytes(), topic.Bytes()) {
+			e := eventType
-			return &eventType
+			return &e
		}
	}
	return 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.

	for eventType, topic := range topicMap() {
		if bytes.Equal(ogTopic.Bytes(), topic.Bytes()) {
			e := eventType
			return &e
		}

37-58: ⚠️ Potential issue

Add missing validation for BridgeDepositClaimedTopic

While you have assigned BridgeDepositClaimedTopic at line 37, its existence is not validated with parsedABI.EventByID as done for the other topics. To ensure consistency and catch potential errors, add a validation step for BridgeDepositClaimedTopic.

Apply the following diff to include the missing validation:

    BridgeProofDisputedTopic = parsedABI.Events["BridgeProofDisputed"].ID

+	_, err = parsedABI.EventByID(BridgeDepositClaimedTopic)
+	if err != nil {
+		panic(err)
+	}

    _, err = parsedABI.EventByID(BridgeProofDisputedTopic)
📝 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.

	BridgeDepositClaimedTopic = parsedABI.Events["BridgeDepositClaimed"].ID
	BridgeProofDisputedTopic = parsedABI.Events["BridgeProofDisputed"].ID

	_, err = parsedABI.EventByID(BridgeRequestedTopic)
	if err != nil {
		panic(err)
	}

	_, err = parsedABI.EventByID(BridgeRelayedTopic)
	if err != nil {
		panic(err)
	}

	_, err = parsedABI.EventByID(BridgeProofProvidedTopic)
	if err != nil {
		panic(err)
	}

	_, err = parsedABI.EventByID(BridgeDepositClaimedTopic)
	if err != nil {
		panic(err)
	}

	_, err = parsedABI.EventByID(BridgeProofDisputedTopic)
	if err != nil {
		panic(err)
	}
services/rfq/contracts/fastbridgev2/parser.go (5)

47-97: 🛠️ Refactor suggestion

Consider refactoring 'ParseEvent' method to reduce complexity

The ParseEvent method has a high cyclomatic complexity, as indicated by the // nolint: cyclop directive. To improve maintainability and readability, consider refactoring this method to reduce its complexity.

One approach is to create separate helper functions for each event parsing case or use a mapping of event types to parsing functions.


51-51: ⚠️ Potential issue

Undefined function 'topicMap()' used

The function topicMap() is used but not defined within this file or imported from another package. This will result in a compilation error. Ensure that topicMap() is defined or properly imported.


89-94: ⚠️ Potential issue

Inconsistent naming between event type and parsing method

The event type is named BridgeDisputeEvent, but the corresponding parsing method is ParseBridgeProofDisputed. To maintain consistency and avoid confusion, consider renaming either the event type or the parsing method to match each other.

For example, you could rename the event type to BridgeProofDisputedEvent to align with the parsing method:

-// BridgeDisputeEvent is the event type for the BridgeDispute event.
-BridgeDisputeEvent
+// BridgeProofDisputedEvent is the event type for the BridgeProofDisputed event.
+BridgeProofDisputedEvent

Committable suggestion was skipped due to low confidence.


48-48: 🛠️ Refactor suggestion

Use pointer receiver for 'parserImpl' methods

The ParseEvent method has a value receiver (p parserImpl). To follow best practices and ensure consistency, consider using a pointer receiver (p *parserImpl), especially since parserImpl contains a pointer field.

Apply the following diff:

-func (p parserImpl) ParseEvent(log ethTypes.Log) (_ EventType, event interface{}, ok bool) {
+func (p *parserImpl) ParseEvent(log ethTypes.Log) (_ EventType, event interface{}, ok bool) {
📝 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.

func (p *parserImpl) ParseEvent(log ethTypes.Log) (_ EventType, event interface{}, ok bool) {

28-28: ⚠️ Potential issue

Typo in comment: 'contracat' should be 'contract'

There's a typographical error in the comment. The word "contracat" should be corrected to "contract".

Apply the following diff to fix the typo:

-// Parser parses events from the fastbridge contracat.
+// Parser parses events from the fastbridge contract.
📝 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.

// Parser parses events from the fastbridge contract.
services/rfq/guard/guardconfig/config.go (2)

88-103: 🛠️ Refactor suggestion

Reduce code duplication in GetRFQAddressV1 and GetRFQAddressV2

The methods GetRFQAddressV1 and GetRFQAddressV2 share similar logic. Consider refactoring them into a single method to improve maintainability.

Example refactored method:

func (c Config) GetRFQAddress(chainID int, version int) (string, error) {
	chainCfg, ok := c.Chains[chainID]
	if !ok {
		return "", fmt.Errorf("chain config not found for chain %d", chainID)
	}
	switch version {
	case 1:
		return chainCfg.RFQAddressV1, nil
	case 2:
		return chainCfg.RFQAddressV2, nil
	default:
		return "", fmt.Errorf("invalid RFQ version %d", version)
	}
}

You can then adjust your calls accordingly.


37-40: 🛠️ Refactor suggestion

Clarify documentation comments for RFQAddressV1 and RFQAddressV2

The comments for RFQAddressV1 and RFQAddressV2 are identical and may cause confusion. Consider updating them to clearly specify which version each address corresponds to.

Apply this diff to improve the comments:

-    // RFQAddressV1 is the rfq bridge contract address.
+    // RFQAddressV1 is the RFQ bridge contract address for version 1.
-    // RFQAddressV2 is the rfq bridge contract address.
+    // RFQAddressV2 is the RFQ bridge contract address for version 2.
📝 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.

	// RFQAddressV1 is the RFQ bridge contract address for version 1.
	RFQAddressV1 string `yaml:"rfq_address_v1"`
	// RFQAddressV2 is the RFQ bridge contract address for version 2.
	RFQAddressV2 string `yaml:"rfq_address_v2"`
services/rfq/testutil/deployers.go (2)

138-140: 🛠️ Refactor suggestion

Inconsistent function naming in fastbridgemockv2 package

The function DeployFastBridgeMock in the fastbridgemockv2 package does not include the 'V2' suffix in its name, whereas the corresponding function in fastbridgev2 is named DeployFastBridgeV2. For consistency and to avoid confusion, consider renaming the function to DeployFastBridgeMockV2.


140-141: 🛠️ Refactor suggestion

Inconsistent reference function naming in fastbridgemockv2 package

Similarly, the function NewFastBridgeMockRef does not include the 'V2' suffix. To maintain consistency with fastbridgev2.NewFastBridgeV2Ref, consider renaming it to NewFastBridgeMockV2Ref.

services/rfq/relayer/reldb/db.go (1)

84-87: 🛠️ Refactor suggestion

Consider unifying 'TransactionV1' and 'Transaction' fields

Adding both TransactionV1 and Transaction fields to the QuoteRequest struct may increase complexity and potential for confusion. Since SendChainGas is the only difference between v1 and v2, consider using a single Transaction field and handling version differences within the field or through interface abstraction.

services/rfq/relayer/reldb/base/model.go (1)

241-242: ⚠️ Potential issue

Possible loss of precision in integer division

Dividing big.Int amounts to scale them can result in integer division, potentially causing precision loss if amounts are not exact multiples of the scaling factor.

Consider using precise decimal arithmetic or alternative methods to ensure accurate amount representation without losing precision due to integer division.

services/rfq/guard/service/handlers.go (2)

244-253: ⚠️ Potential issue

Verify the correctness of field assignments in relayDetails.

In both isProveValidV1 and isProveValidV2, the relayDetails struct is populated from event data. Please ensure that the event fields are correctly assigned to the corresponding relayDetails fields.

For instance:

  • The To field in relayDetails is assigned event.OriginToken, which represents a token address. Should this be event.To, representing the recipient address?
  • Similarly, check that OriginToken and DestToken are correctly assigned.

Also applies to: 293-302


266-269: ⚠️ Potential issue

Use the correct parser for FastBridgeV2 events.

In the isProveValidV2 function, the parser is initialized using fastbridge.NewParser, which is intended for FastBridgeV1 events. To correctly parse FastBridgeV2 events, use fastbridgev2.NewParser instead.

Apply this diff to fix the parser initialization:

-	parser, err := fastbridge.NewParser(common.HexToAddress(rfqAddr))
+	parser, err := fastbridgev2.NewParser(common.HexToAddress(rfqAddr))
📝 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.

	parser, err := fastbridgev2.NewParser(common.HexToAddress(rfqAddr))
	if err != nil {
		return false, fmt.Errorf("could not get parser: %w", err)
	}
services/rfq/guard/service/guard.go (3)

63-71: 🛠️ Refactor suggestion

Reduce code duplication in handler initialization

The initialization of fastBridgeHandlerV1 and fastBridgeHandlerV2 is nearly identical. Refactoring this into a loop or a helper function can reduce duplication and improve maintainability.

You could refactor the handler initialization as follows:

handlers := map[string]*fastBridgeHandler{}

for _, version := range []string{"v1", "v2"} {
	handlers[version] = &fastBridgeHandler{
		listeners: make(map[int]listener.ContractListener),
		contracts: make(map[int]*fastbridge.FastBridgeRef),
	}
}

Update the Guard struct and usage accordingly to utilize the handlers map.


321-349: ⚠️ Potential issue

Avoid casting V2 events to V1 types

In runChainIndexerV2, V2 events are being converted to V1 event structs before handling. This approach may lead to errors if the event structures diverge in the future. It's safer to handle V2 events directly or define a common interface for shared handling.

Consider updating the event handlers to accept interfaces or create separate handling functions for V2 events to ensure type safety and future compatibility.


389-394: 🛠️ Refactor suggestion

Optimize isV2Address method by utilizing existing data

The isV2Address method fetches the RFQ address from the config each time it's called. Since the address is already stored in fastBridgeHandlerV2.contracts, consider checking against the stored contract address to avoid redundant configuration calls.

For example:

func (g *Guard) isV2Address(chainID int, addr common.Address) bool {
	contract, ok := g.fastBridgeHandlerV2.contracts[chainID]
	if !ok {
		return false
	}
	return contract.Address() == addr
}
services/rfq/relayer/service/handlers.go (2)

85-85: 🛠️ Refactor suggestion

Address the TODO: Retrieve decimals from inventory and handle invalid tokens

The TODO comment at line 85 suggests improving the retrieval of originDecimals and destDecimals by pulling them from the inventory. Additionally, tokens that are not present in the inventory should be marked as invalid to prevent processing invalid requests.

Would you like assistance in implementing this change, or should I open a GitHub issue to track this task?


103-104: 🛠️ Refactor suggestion

Clarify field naming by renaming 'Transaction' to 'TransactionV2'

In the dbReq struct, the field Transaction is assigned txV2, while TransactionV1 is assigned txV1. To improve code clarity and maintain consistency, consider renaming the Transaction field to TransactionV2.

Apply this diff to update the field name:

  dbReq := reldb.QuoteRequest{
      BlockNumber:         req.Raw.BlockNumber,
      RawRequest:          req.Request,
      OriginTokenDecimals: *originDecimals,
      DestTokenDecimals:   *destDecimals,
      TransactionID:       req.TransactionId,
      Sender:              req.Sender,
      TransactionV1:       txV1,
-     Transaction:         txV2,
+     TransactionV2:       txV2,
      Status:              reldb.Seen,
      OriginTxHash:        req.Raw.TxHash,
  }

Ensure that all references to Transaction are updated accordingly throughout the codebase.

Committable suggestion was skipped due to low confidence.

@github-actions github-actions bot added size/m and removed size/l labels Oct 24, 2024
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 8

🧹 Outside diff range and nitpick comments (4)
services/rfq/guard/service/guard.go (2)

Line range hint 220-236: Improve error message in group.Wait()

The error message in group.Wait() is incomplete and doesn't include the error details.

Apply this fix:

 err = group.Wait()
 if err != nil {
-    return fmt.Errorf("could not run chain indexers")
+    return fmt.Errorf("could not run chain indexers: %w", err)
 }

Line range hint 239-294: Consider reducing code duplication in indexer implementations

The runChainIndexerV1 and runChainIndexerV2 functions share significant code structure. Consider extracting common functionality into a shared helper function to improve maintainability.

Example refactor:

func (g Guard) runChainIndexerCommon(
    ctx context.Context,
    chainID int,
    listener listener.ContractListener,
    parser eventParser,
    handleEvent func(context.Context, interface{}) error,
) error {
    return listener.Listen(ctx, func(parentCtx context.Context, log types.Log) error {
        et, parsedEvent, ok := parser.ParseEvent(log)
        if !ok {
            if len(log.Topics) != 0 {
                logger.Warnf("unknown event %s", log.Topics[0])
            }
            return nil
        }

        ctx, span := g.metrics.Tracer().Start(parentCtx, /* ... */)
        defer func() { metrics.EndSpanWithErr(span, err) }()
        
        return handleEvent(ctx, parsedEvent)
    })
}

Also applies to: 295-362

services/rfq/guard/guardconfig/config.go (1)

76-76: Differentiate Error Messages for V1 and V2 RFQ Addresses

In the Validate method, the error messages for invalid RFQ addresses are identical for both V1 and V2: "invalid rfq address: %s". This could lead to confusion when diagnosing issues. Consider specifying whether the invalid address is RFQAddressV1 or RFQAddressV2 for clearer debugging.

Suggested Change:

// For addrV1 validation
-	return fmt.Errorf("invalid rfq address: %s", *addrV1)
+	return fmt.Errorf("invalid RFQAddressV1: %s", *addrV1)

// For addrV2 validation
-	return fmt.Errorf("invalid rfq address: %s", addrV2)
+	return fmt.Errorf("invalid RFQAddressV2: %s", addrV2)

Also applies to: 83-83

services/rfq/guard/service/handlers.go (1)

356-374: Complete the exhaustive checks in relayMatchesBridgeRequest

The relayMatchesBridgeRequest function includes a TODO comment questioning the exhaustiveness of the checks. To ensure the relay details accurately match the bridge request, review all relevant fields and confirm that no critical data points are omitted.

Would you like assistance in identifying additional fields or conditions that should be included in the matching logic?

📜 Review details

Configuration used: .coderabbit.yaml
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between bdfb0a4 and bcb5b77.

📒 Files selected for processing (4)
  • services/rfq/guard/guardconfig/config.go (4 hunks)
  • services/rfq/guard/service/guard.go (8 hunks)
  • services/rfq/guard/service/handlers.go (7 hunks)
  • services/rfq/relayer/relconfig/config.go (1 hunks)
🔇 Additional comments (5)
services/rfq/guard/service/guard.go (3)

20-20: LGTM: Clean separation of v1 and v2 contract handling

The Guard struct's design cleanly separates v1 and v2 contract concerns using distinct maps, which will make version-specific operations clearer and easier to maintain.

Also applies to: 33-42


60-91: LGTM: Proper initialization of v1 contract handlers

The v1 contract initialization is well-structured with proper error handling and conditional setup based on address availability.


205-219: LGTM: Well-structured v1 indexer initialization

The v1 indexer initialization is properly guarded with null checks and has good error handling.

services/rfq/guard/guardconfig/config.go (1)

136-137: ⚠️ Potential issue

Verify Field Names in NewGuardConfigFromRelayer

In NewGuardConfigFromRelayer, you're assigning chainCfg.RFQAddressV1 to RFQAddressV1 and chainCfg.RFQAddress to RFQAddressV2. Please ensure that chainCfg.RFQAddressV1 exists and is correctly populated in relconfig.ChainConfig. There may be a mismatch in field names or missing fields between the relayer and guard configurations.

Verification Script:

This script searches the codebase for the definition of RFQAddressV1 within the ChainConfig struct of the relconfig package. Ensure that RFQAddressV1 is defined and properly used in relconfig.ChainConfig.

✅ Verification successful

Let me try a different approach to verify this by searching for the ChainConfig struct definition and checking its fields.

This will help us understand:

  1. The actual fields in ChainConfig struct
  2. All occurrences of RFQAddressV1 in the codebase to verify if there's a mismatch

Field names in NewGuardConfigFromRelayer are correct

Based on the search results, the field assignments in NewGuardConfigFromRelayer are correct:

  1. In services/rfq/relayer/relconfig/config.go, RFQAddressV1 is defined as:
RFQAddressV1 *string `yaml:"rfq_address_v1"`
  1. In services/rfq/guard/guardconfig/config.go, the same field exists:
RFQAddressV1 *string `yaml:"rfq_address_v1"`

The assignment RFQAddressV1: chainCfg.RFQAddressV1 is correct as both structs have matching field names and types. Similarly, RFQAddress in the relayer config maps correctly to RFQAddressV2 in the guard config.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify that `RFQAddressV1` exists in `relconfig.ChainConfig`

# Search for the definition of RFQAddressV1 in relconfig package
ast-grep --pattern $'type ChainConfig {
    $$$
    RFQAddressV1 $_
    $$$
}'

Length of output: 79


Script:

#!/bin/bash
# Search for ChainConfig struct definition
rg "type ChainConfig struct" -A 20

# Also search for any usage of RFQAddressV1
rg "RFQAddressV1"

Length of output: 17280

services/rfq/guard/service/handlers.go (1)

298-301: Verify consistent usage of FastBridgeV2 methods and types

Ensure that all FastBridgeV2 specific methods and types are consistently used throughout the code, particularly in the isProveValidV2 function and related logic.

Run the following script to identify usages of FastBridgeV2 methods and confirm correct implementation:

This will help ensure that all V2 contract interactions are correctly handled.

✅ Verification successful

Let me analyze the code further to verify the consistent usage of FastBridgeV2 methods and types.

This will help us verify the consistent usage of V2-specific methods in the guard service, particularly around the code snippet in question.


Let me verify one more aspect to ensure complete analysis.

This will help us verify the implementation details of the V2-specific validation method.


Based on the analysis of the codebase through the executed scripts, I can now provide a final response:

FastBridgeV2 methods and types are consistently implemented

The code shows proper and consistent implementation of FastBridgeV2-specific functionality:

  • Correct usage of GetRFQAddressV2 throughout the guard service
  • Proper V2 contract interactions in the relayer service
  • Consistent type handling for FastBridgeV2BridgeTransactionV2 and related structures
  • Proper event handling for V2-specific events

The code in question correctly uses GetRFQAddressV2 which aligns with the rest of the V2 implementation patterns seen across the codebase.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify usage of FastBridgeV2 methods and types.

# Search for references to FastBridgeV2 in the codebase.
rg 'fastbridgev2\.' -A 3

Length of output: 16350


Script:

#!/bin/bash
# Search for all V2-specific method calls and types in the guard service
rg -A 3 'GetRFQAddressV2|isProveValidV2|BridgeTransactionV2' services/rfq/guard/

Length of output: 2304


Script:

#!/bin/bash
# Search for the implementation of isProveValidV2 and related V2 methods
ast-grep --pattern 'func $_ isProveValidV2($_) {
  $$$
}'

Length of output: 59

Comment on lines +380 to +386
func (g *Guard) isV2Address(chainID int, addr common.Address) bool {
rfqAddr, err := g.cfg.GetRFQAddressV2(chainID)
if err != nil {
return false
}
return addr == common.HexToAddress(rfqAddr)
}
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 caching the v2 address lookup

The isV2Address function performs a config lookup and address comparison on every call. Consider caching the v2 addresses in a map during initialization to improve performance.

Example implementation:

type Guard struct {
    // ... existing fields ...
    v2AddressCache map[int]common.Address
}

func (g *Guard) isV2Address(chainID int, addr common.Address) bool {
    if cachedAddr, ok := g.v2AddressCache[chainID]; ok {
        return addr == cachedAddr
    }
    return false
}

Comment on lines 93 to 115
// setup v2
rfqAddrV2, err := cfg.GetRFQAddressV2(chainID)
if err != nil {
return nil, fmt.Errorf("could not get rfq address: %w", err)
}
chainClient, err := omniClient.GetChainClient(ctx, chainID)
if err != nil {
return nil, fmt.Errorf("could not get chain client: %w", err)
}

contract, err := fastbridge.NewFastBridgeRef(common.HexToAddress(rfqAddr), chainClient)
contract, err := fastbridgev2.NewFastBridgeV2Ref(common.HexToAddress(rfqAddrV2), chainClient)
if err != nil {
return nil, fmt.Errorf("could not create fast bridge contract: %w", err)
}
startBlock, err := contract.DeployBlock(&bind.CallOpts{Context: ctx})
if err != nil {
return nil, fmt.Errorf("could not get deploy block: %w", err)
}
chainListener, err := listener.NewChainListener(chainClient, store, common.HexToAddress(rfqAddr), uint64(startBlock.Int64()), metricHandler, listener.WithName("guard"))
chainListener, err := listener.NewChainListener(chainClient, store, common.HexToAddress(rfqAddrV2), uint64(startBlock.Int64()), metricHandler, listener.WithName("guard"))
if err != nil {
return nil, fmt.Errorf("could not get chain listener: %w", err)
}
chainListeners[chainID] = chainListener

// setup FastBridge contract on this chain
contracts[chainID], err = fastbridge.NewFastBridgeRef(common.HexToAddress(rfqAddr), chainClient)
if err != nil {
return nil, fmt.Errorf("could not create bridge contract: %w", err)
}
listenersV2[chainID] = chainListener
contractsV2[chainID] = contract
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 null check for v2 address

The v2 contract initialization assumes rfqAddrV2 is always non-empty. Consider adding a null check similar to v1 to handle cases where v2 contracts might not be deployed yet.

Apply this pattern:

 rfqAddrV2, err := cfg.GetRFQAddressV2(chainID)
 if err != nil {
     return nil, fmt.Errorf("could not get rfq address: %w", err)
 }
+if rfqAddrV2 == "" {
+    continue
+}
 chainClient, err := omniClient.GetChainClient(ctx, chainID)
📝 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
// setup v2
rfqAddrV2, err := cfg.GetRFQAddressV2(chainID)
if err != nil {
return nil, fmt.Errorf("could not get rfq address: %w", err)
}
chainClient, err := omniClient.GetChainClient(ctx, chainID)
if err != nil {
return nil, fmt.Errorf("could not get chain client: %w", err)
}
contract, err := fastbridge.NewFastBridgeRef(common.HexToAddress(rfqAddr), chainClient)
contract, err := fastbridgev2.NewFastBridgeV2Ref(common.HexToAddress(rfqAddrV2), chainClient)
if err != nil {
return nil, fmt.Errorf("could not create fast bridge contract: %w", err)
}
startBlock, err := contract.DeployBlock(&bind.CallOpts{Context: ctx})
if err != nil {
return nil, fmt.Errorf("could not get deploy block: %w", err)
}
chainListener, err := listener.NewChainListener(chainClient, store, common.HexToAddress(rfqAddr), uint64(startBlock.Int64()), metricHandler, listener.WithName("guard"))
chainListener, err := listener.NewChainListener(chainClient, store, common.HexToAddress(rfqAddrV2), uint64(startBlock.Int64()), metricHandler, listener.WithName("guard"))
if err != nil {
return nil, fmt.Errorf("could not get chain listener: %w", err)
}
chainListeners[chainID] = chainListener
// setup FastBridge contract on this chain
contracts[chainID], err = fastbridge.NewFastBridgeRef(common.HexToAddress(rfqAddr), chainClient)
if err != nil {
return nil, fmt.Errorf("could not create bridge contract: %w", err)
}
listenersV2[chainID] = chainListener
contractsV2[chainID] = contract
// setup v2
rfqAddrV2, err := cfg.GetRFQAddressV2(chainID)
if err != nil {
return nil, fmt.Errorf("could not get rfq address: %w", err)
}
if rfqAddrV2 == "" {
continue
}
chainClient, err := omniClient.GetChainClient(ctx, chainID)
if err != nil {
return nil, fmt.Errorf("could not get chain client: %w", err)
}
contract, err := fastbridgev2.NewFastBridgeV2Ref(common.HexToAddress(rfqAddrV2), chainClient)
if err != nil {
return nil, fmt.Errorf("could not create fast bridge contract: %w", err)
}
startBlock, err := contract.DeployBlock(&bind.CallOpts{Context: ctx})
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"))
if err != nil {
return nil, fmt.Errorf("could not get chain listener: %w", err)
}
listenersV2[chainID] = chainListener
contractsV2[chainID] = contract

Comment on lines +76 to +79
// RFQAddress is the rfq bridge contract address (v2).
RFQAddress string `yaml:"rfq_address"`
// RFQAddressV1 is the rfq bridge contract address (v1, optional).
RFQAddressV1 *string `yaml:"rfq_address_v1"`
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 validation logic for RFQ addresses

The addition of dual RFQ addresses (v1 and v2) requires additional validation to ensure proper configuration:

  1. When RFQAddressV1 is present, both addresses should be valid Ethereum addresses
  2. Consider adding validation to ensure proper migration state (e.g., at least one address must be configured)
  3. Document the migration path for existing configurations

Consider adding validation in the Validate method:

func (c Config) Validate(ctx context.Context, omniclient omniClient.RPCClient) (err error) {
+       // Validate RFQ addresses
+       for chainID, chainCfg := range c.Chains {
+               if !common.IsHexAddress(chainCfg.RFQAddress) {
+                       return fmt.Errorf("invalid RFQAddress (v2) for chain %d: %s", chainID, chainCfg.RFQAddress)
+               }
+               if chainCfg.RFQAddressV1 != nil {
+                       if !common.IsHexAddress(*chainCfg.RFQAddressV1) {
+                               return fmt.Errorf("invalid RFQAddressV1 for chain %d: %s", chainID, *chainCfg.RFQAddressV1)
+                       }
+               }
+       }

        // Existing validation logic...
        maintenancePctSums := map[string]float64{}

Consider adding documentation in the struct to clarify:

  1. The relationship between v1 and v2 addresses
  2. Migration requirements and timeline
  3. Configuration examples for different scenarios (v2 only, dual-version support)

Committable suggestion was skipped due to low confidence.

Comment on lines 38 to 40
RFQAddressV1 *string `yaml:"rfq_address_v1"`
// RFQAddressV2 is the rfq bridge contract address.
RFQAddressV2 string `yaml:"rfq_address_v2"`
Copy link
Contributor

@coderabbitai coderabbitai bot Oct 24, 2024

Choose a reason for hiding this comment

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

⚠️ Potential issue

Ensure Consistent Types for RFQAddressV1 and RFQAddressV2

RFQAddressV1 is declared as a pointer to string (*string), whereas RFQAddressV2 is declared as a string (string). This inconsistency may lead to confusion and potential errors in handling these fields. If both addresses are required configurations, consider using string for both. If RFQAddressV1 is optional and may be nil, ensure this distinction is clearly documented and consistently handled throughout the codebase.

Suggested Change:

Option 1: If both addresses are required:

-	RFQAddressV1 *string `yaml:"rfq_address_v1"`
+	RFQAddressV1 string  `yaml:"rfq_address_v1"`

Option 2: If both addresses can be optional:

-	RFQAddressV2 string  `yaml:"rfq_address_v2"`
+	RFQAddressV2 *string `yaml:"rfq_address_v2"`

Committable suggestion was skipped due to low confidence.

Copy link
Collaborator

Choose a reason for hiding this comment

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

@dwasse I agree with the Rabbit that highlighting the fields as optional/required would be helpful here.

Copy link
Contributor

Choose a reason for hiding this comment

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

Seems like the humans are having a chat. I'll hop back into my burrow for now. If you need me again, just tag @coderabbitai in a new comment, and I'll come hopping out!

Comment on lines 95 to 110
// GetRFQAddressV1 returns the RFQ address for the given chain.
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 chainCfg.RFQAddressV1, nil
}

// GetRFQAddressV2 returns the RFQ address for the given chain.
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 chainCfg.RFQAddress, nil
return chainCfg.RFQAddressV2, nil
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

Standardize Return Types of GetRFQAddress Methods

GetRFQAddressV1 returns a pointer to string (*string, error), while GetRFQAddressV2 returns a string (string, error). This inconsistency might cause confusion and complicate error handling. Consider standardizing the return types of both methods to improve consistency and reduce potential errors.

Suggested Change:

Option 1: Have both methods return (*string, error):

// For GetRFQAddressV2
-func (c Config) GetRFQAddressV2(chainID int) (string, error) {
+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 nil, fmt.Errorf("chain config not found for chain %d", chainID)
     }
-    return chainCfg.RFQAddressV2, nil
+    return &chainCfg.RFQAddressV2, nil
}

Option 2: Have both methods return (string, error) and adjust handling of optional values accordingly.

Committable suggestion was skipped due to low confidence.

Comment on lines +171 to +209
func (g *Guard) disputeV1(ctx context.Context, proven *guarddb.PendingProven, bridgeRequest *guarddb.BridgeRequest) error {
contract, ok := g.contractsV1[int(bridgeRequest.Transaction.OriginChainId)]
if !ok {
return errors.New("could not get contract")
}
_, err := g.txSubmitter.SubmitTransaction(ctx, big.NewInt(int64(bridgeRequest.Transaction.OriginChainId)), func(transactor *bind.TransactOpts) (tx *types.Transaction, err error) {
tx, err = contract.Dispute(transactor, proven.TransactionID)
if err != nil {
return nil, fmt.Errorf("could not dispute: %w", err)
}

return tx, nil
})
if err != nil {
return fmt.Errorf("could not dispute: %w", err)
}

return nil
}

func (g *Guard) disputeV2(ctx context.Context, proven *guarddb.PendingProven, bridgeRequest *guarddb.BridgeRequest) error {
contract, ok := g.contractsV2[int(bridgeRequest.Transaction.OriginChainId)]
if !ok {
return errors.New("could not get contract")
}
_, err := g.txSubmitter.SubmitTransaction(ctx, big.NewInt(int64(bridgeRequest.Transaction.OriginChainId)), func(transactor *bind.TransactOpts) (tx *types.Transaction, err error) {
tx, err = contract.Dispute(transactor, proven.TransactionID)
if err != nil {
return nil, fmt.Errorf("could not dispute: %w", err)
}

return tx, nil
})
if err != nil {
return fmt.Errorf("could not dispute: %w", err)
}

return nil
}
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

Refactor disputeV1 and disputeV2 to eliminate code duplication

The disputeV1 and disputeV2 functions share similar logic, differing mainly in the contract version used. Refactoring these into a single function can reduce redundancy and improve maintainability.

Consider creating a unified dispute function:

func (g *Guard) dispute(ctx context.Context, proven *guarddb.PendingProven, bridgeRequest *guarddb.BridgeRequest, version int) error {
    var contract YourContractInterface
    switch version {
    case 1:
        contract, ok = g.contractsV1[int(bridgeRequest.Transaction.OriginChainId)]
    case 2:
        contract, ok = g.contractsV2[int(bridgeRequest.Transaction.OriginChainId)]
    default:
        return errors.New("invalid contract version")
    }
    if !ok {
        return errors.New("could not get contract")
    }
    _, err := g.txSubmitter.SubmitTransaction(ctx, big.NewInt(int64(bridgeRequest.Transaction.OriginChainId)), func(transactor *bind.TransactOpts) (*types.Transaction, error) {
        tx, err := contract.Dispute(transactor, proven.TransactionID)
        if err != nil {
            return nil, fmt.Errorf("could not dispute: %w", err)
        }
        return tx, nil
    })
    if err != nil {
        return fmt.Errorf("could not submit dispute transaction: %w", err)
    }
    return nil
}

Then, in your handleProveCalled function, you can call:

if g.isV2Address(int(bridgeRequest.Transaction.OriginChainId), proven.FastBridgeAddress) {
    err = g.dispute(ctx, proven, bridgeRequest, 2)
} else {
    err = g.dispute(ctx, proven, bridgeRequest, 1)
}

Comment on lines 243 to 293
func (g *Guard) isProveValidV1(ctx context.Context, proven *guarddb.PendingProven, bridgeRequest *guarddb.BridgeRequest, receipt *types.Receipt) (bool, error) {
span := trace.SpanFromContext(ctx)

rfqAddr, err := g.cfg.GetRFQAddressV1(int(bridgeRequest.Transaction.DestChainId))
if err != nil {
return false, fmt.Errorf("could not get rfq address v1: %w", err)
}
if rfqAddr == nil {
return false, errors.New("rfq address v1 is nil")
}
parser, err := fastbridge.NewParser(common.HexToAddress(*rfqAddr))
if err != nil {
return false, fmt.Errorf("could not get rfq address: %w", err)
return false, fmt.Errorf("could not get parser: %w", err)
}

for _, log := range receipt.Logs {
_, parsedEvent, ok := parser.ParseEvent(*log)
if !ok {
continue
}

if log.Address != common.HexToAddress(*rfqAddr) {
span.AddEvent(fmt.Sprintf("log address %s does not match rfq address %s", log.Address.Hex(), *rfqAddr))
continue
}

event, ok := parsedEvent.(*fastbridge.FastBridgeBridgeRelayed)
if !ok {
span.AddEvent("event is not a BridgeRelayed event")
continue
}

if event.Relayer != proven.RelayerAddress {
span.AddEvent(fmt.Sprintf("relayer address %s does not match prover address %s", event.Relayer.Hex(), proven.RelayerAddress.Hex()))
continue
}

details := relayDetails{
TransactionID: event.TransactionId,
OriginAmount: event.OriginAmount,
DestAmount: event.DestAmount,
OriginChainID: event.OriginChainId,
To: event.OriginToken,
DestToken: event.DestToken,
}

return relayMatchesBridgeRequest(details, bridgeRequest), nil
}

return false, nil
}
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

Verify field mappings in relayDetails assignments

In both isProveValidV1 and isProveValidV2, ensure that the fields from the events are correctly mapped to the relayDetails struct. Inconsistent field assignments may lead to incorrect validation outcomes.

Review the assignments, particularly:

  • Confirm whether To should map to event.OriginToken or another field.
  • Ensure OriginToken and DestToken are assigned the correct values from the event.

For example, in isProveValidV2:

details := relayDetails{
    TransactionID: event.TransactionId,
    OriginAmount:  event.OriginAmount,
    DestAmount:    event.DestAmount,
    OriginChainID: event.OriginChainId,
    To:            event.OriginToken,       // Verify if this is correct
    DestToken:     event.DestToken,
    OriginToken:   event.OriginToken,       // Is this duplicated?
}

Consider clarifying the purpose of each field and adjusting the assignments accordingly.

Also applies to: 295-343

Comment on lines 298 to 301
rfqAddr, err := g.cfg.GetRFQAddressV2(int(bridgeRequest.Transaction.DestChainId))
if err != nil {
return false, fmt.Errorf("could not get rfq address v2: %w", err)
}
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

Use fastbridgev2.NewParser instead of fastbridge.NewParser

In the isProveValidV2 function, the parser is instantiated using fastbridge.NewParser, which is intended for the V1 contract. To correctly parse events from the V2 contract, it should use fastbridgev2.NewParser.

Apply this diff to fix the parser instantiation:

 func (g *Guard) isProveValidV2(ctx context.Context, proven *guarddb.PendingProven, bridgeRequest *guarddb.BridgeRequest, receipt *types.Receipt) (bool, error) {
     span := trace.SpanFromContext(ctx)

     rfqAddr, err := g.cfg.GetRFQAddressV2(int(bridgeRequest.Transaction.DestChainId))
     if err != nil {
         return false, fmt.Errorf("could not get rfq address v2: %w", err)
     }
-    parser, err := fastbridge.NewParser(common.HexToAddress(rfqAddr))
+    parser, err := fastbridgev2.NewParser(common.HexToAddress(rfqAddr))
     if err != nil {
         return false, fmt.Errorf("could not get parser: %w", err)
     }

Committable suggestion was skipped due to low confidence.

@trajan0x trajan0x changed the title feat(rfq-guard): v2 guard logic feat(rfq-guard): v2 guard logic SLT-387 Oct 24, 2024
@trajan0x trajan0x changed the title feat(rfq-guard): v2 guard logic SLT-387 feat(rfq-guard): v2 guard logic [SLT-387] Oct 24, 2024
Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

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

Actionable comments posted: 1

🧹 Outside diff range and nitpick comments (1)
services/rfq/guard/service/handlers.go (1)

231-233: Enhance error messages with more context.

The error messages could be more specific about what failed during validation.

Apply this diff to improve error messages:

-			return false, fmt.Errorf("could not check prove validity v2: %w", err)
+			return false, fmt.Errorf("could not validate V2 proof for transaction %s: %w", hexutil.Encode(proven.TransactionID[:]), err)
-			return false, fmt.Errorf("could not check prove validity v1: %w", err)
+			return false, fmt.Errorf("could not validate V1 proof for transaction %s: %w", hexutil.Encode(proven.TransactionID[:]), err)

Also applies to: 236-238

📜 Review details

Configuration used: .coderabbit.yaml
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between bcb5b77 and 873659e.

📒 Files selected for processing (3)
  • services/rfq/guard/guarddb/base/model.go (3 hunks)
  • services/rfq/guard/service/guard.go (8 hunks)
  • services/rfq/guard/service/handlers.go (8 hunks)
🚧 Files skipped from review as they are similar to previous changes (1)
  • services/rfq/guard/guarddb/base/model.go
🔇 Additional comments (4)
services/rfq/guard/service/guard.go (3)

Line range hint 205-228: LGTM! Well-structured chain indexer setup

The separation of v1 and v2 indexers with clear error messages and proper goroutine management is well implemented.


391-397: Previous review comment about caching is still valid


93-115: ⚠️ Potential issue

Add null check for v2 address

The v2 contract initialization should handle cases where v2 contracts might not be deployed yet, similar to how v1 is handled.

Apply this diff:

 // setup v2
 rfqAddrV2, err := cfg.GetRFQAddressV2(chainID)
 if err != nil {
     return nil, fmt.Errorf("could not get rfq address: %w", err)
 }
+if rfqAddrV2 == "" {
+    continue
+}
 chainClient, err := omniClient.GetChainClient(ctx, chainID)

Likely invalid or redundant comment.

services/rfq/guard/service/handlers.go (1)

356-357: Address TODO comment about validation completeness.

The TODO comment suggests uncertainty about whether the validation checks are exhaustive. Let's verify the completeness of the validation.

Let's analyze the bridge request fields to ensure we're not missing any critical validations:

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

parser, err := fastbridgev2.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 *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)
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)
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)
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
}
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 extracting event conversion logic

The event conversion logic in runChainIndexerV2 could be moved to separate helper methods to improve maintainability and reduce the cyclomatic complexity.

Consider creating helper methods for event conversions:

func convertToV1BridgeRequestedEvent(v2Event *fastbridgev2.FastBridgeV2BridgeRequested) *fastbridge.FastBridgeBridgeRequested {
    return &fastbridge.FastBridgeBridgeRequested{
        TransactionId: v2Event.TransactionId,
        Sender:        v2Event.Sender,
        Request:       v2Event.Request,
        DestChainId:   v2Event.DestChainId,
        OriginToken:   v2Event.OriginToken,
        DestToken:     v2Event.DestToken,
        DestAmount:    v2Event.DestAmount,
        SendChainGas:  v2Event.SendChainGas,
        Raw:           v2Event.Raw,
    }
}

@@ -0,0 +1,109 @@
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;

Check warning

Code scanning / Slither

Different pragma directives are used

3 different versions of Solidity are used: - Version constraint ^0.8.20 is used by: -^0.8.20 (contracts/AdminV2.sol#2) -^0.8.20 (node_modules/@openzeppelin/contracts/access/AccessControl.sol#4) -^0.8.20 (node_modules/@openzeppelin/contracts/access/IAccessControl.sol#4) -^0.8.20 (node_modules/@openzeppelin/contracts/access/extensions/AccessControlEnumerable.sol#4) -^0.8.20 (node_modules/@openzeppelin/contracts/access/extensions/IAccessControlEnumerable.sol#4) -^0.8.20 (node_modules/@openzeppelin/contracts/token/ERC20/IERC20.sol#4) -^0.8.20 (node_modules/@openzeppelin/contracts/token/ERC20/extensions/IERC20Permit.sol#4) -^0.8.20 (node_modules/@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol#4) -^0.8.20 (node_modules/@openzeppelin/contracts/utils/Address.sol#4) -^0.8.20 (node_modules/@openzeppelin/contracts/utils/Context.sol#4) -^0.8.20 (node_modules/@openzeppelin/contracts/utils/introspection/ERC165.sol#4) -^0.8.20 (node_modules/@openzeppelin/contracts/utils/introspection/IERC165.sol#4) -^0.8.20 (node_modules/@openzeppelin/contracts/utils/structs/EnumerableSet.sol#5) - Version constraint 0.8.24 is used by: -0.8.24 (contracts/FastBridgeV2.sol#2) -0.8.24 (contracts/zaps/TokenZapV1.sol#2) - Version constraint ^0.8.4 is used by: -^0.8.4 (contracts/interfaces/IAdminV2.sol#2) -^0.8.4 (contracts/interfaces/IAdminV2Errors.sol#2) -^0.8.4 (contracts/interfaces/IFastBridge.sol#2) -^0.8.4 (contracts/interfaces/IFastBridgeV2.sol#2) -^0.8.4 (contracts/interfaces/IFastBridgeV2Errors.sol#2) -^0.8.4 (contracts/interfaces/IMulticallTarget.sol#2) -^0.8.4 (contracts/interfaces/IZapRecipient.sol#2) -^0.8.4 (contracts/libs/BridgeTransactionV2.sol#2) -^0.8.4 (contracts/libs/ZapDataV1.sol#2) -^0.8.4 (contracts/utils/MulticallTarget.sol#2)
@@ -0,0 +1,109 @@
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;

Check warning

Code scanning / Slither

Incorrect versions of Solidity

Version constraint ^0.8.20 contains known severe issues (https://solidity.readthedocs.io/en/latest/bugs.html) - VerbatimInvalidDeduplication - FullInlinerNonExpressionSplitArgumentEvaluationOrder - MissingSideEffectsOnSelectorAccess. It is used by: - ^0.8.20 (contracts/AdminV2.sol#2) - ^0.8.20 (node_modules/@openzeppelin/contracts/access/AccessControl.sol#4) - ^0.8.20 (node_modules/@openzeppelin/contracts/access/IAccessControl.sol#4) - ^0.8.20 (node_modules/@openzeppelin/contracts/access/extensions/AccessControlEnumerable.sol#4) - ^0.8.20 (node_modules/@openzeppelin/contracts/access/extensions/IAccessControlEnumerable.sol#4) - ^0.8.20 (node_modules/@openzeppelin/contracts/token/ERC20/IERC20.sol#4) - ^0.8.20 (node_modules/@openzeppelin/contracts/token/ERC20/extensions/IERC20Permit.sol#4) - ^0.8.20 (node_modules/@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol#4) - ^0.8.20 (node_modules/@openzeppelin/contracts/utils/Address.sol#4) - ^0.8.20 (node_modules/@openzeppelin/contracts/utils/Context.sol#4) - ^0.8.20 (node_modules/@openzeppelin/contracts/utils/introspection/ERC165.sol#4) - ^0.8.20 (node_modules/@openzeppelin/contracts/utils/introspection/IERC165.sol#4) - ^0.8.20 (node_modules/@openzeppelin/contracts/utils/structs/EnumerableSet.sol#5)
Comment on lines 243 to 276
function cancelV2(bytes calldata request) public {
// Decode the request and check that it could be cancelled.
request.validateV2();
bytes32 transactionId = keccak256(request);
BridgeTransactionV2 memory transaction = getBridgeTransactionV2(request);
_validateRelayParams(transaction, transactionId, relayer);
// mark bridge transaction as relayed
bridgeRelayDetails[transactionId] =
BridgeRelay({blockNumber: uint48(block.number), blockTimestamp: uint48(block.timestamp), relayer: relayer});

// transfer tokens to recipient on destination chain and do an arbitrary call if requested
address to = transaction.destRecipient;
address token = transaction.destToken;
uint256 amount = transaction.destAmount;
uint256 callValue = transaction.callValue;

// Emit the event before any external calls

// Can only cancel a REQUESTED transaction after its deadline expires.
BridgeTxDetails storage $ = bridgeTxDetails[transactionId];
if ($.status != BridgeStatus.REQUESTED) revert StatusIncorrect();

// Permissionless cancel is only allowed after `cancelDelay` on top of the deadline.
uint256 deadline = request.deadline();
if (!hasRole(CANCELER_ROLE, msg.sender)) deadline += cancelDelay;
if (block.timestamp <= deadline) revert DeadlineNotExceeded();

// Update status to REFUNDED.
// Note: this is a storage write.
$.status = BridgeStatus.REFUNDED;

// Return the full amount (collateral + protocol fees) to the original sender.
// The protocol fees are only accumulated when the transaction is claimed, so we don't need to update them here.
address to = request.originSender();
address token = request.originToken();
uint256 amount = request.originAmount() + request.originFeeAmount();

// Emit the event before any external calls.
emit BridgeDepositRefunded(transactionId, to, token, amount);

// Return the funds to the original sender as last transaction action.
if (token == NATIVE_GAS_TOKEN) {
Address.sendValue(payable(to), amount);
} else {
IERC20(token).safeTransfer(to, amount);
}
}

Check notice

Code scanning / Slither

Block timestamp

FastBridgeV2.cancelV2(bytes) (contracts/FastBridgeV2.sol#243-276) uses timestamp for comparisons Dangerous comparisons: - block.timestamp <= deadline (contracts/FastBridgeV2.sol#255)
@@ -1,5 +1,5 @@
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;
pragma solidity ^0.8.4;

Check warning

Code scanning / Slither

Incorrect versions of Solidity

Version constraint ^0.8.4 contains known severe issues (https://solidity.readthedocs.io/en/latest/bugs.html) - FullInlinerNonExpressionSplitArgumentEvaluationOrder - MissingSideEffectsOnSelectorAccess - AbiReencodingHeadOverflowWithStaticArrayCleanup - DirtyBytesArrayToStorage - DataLocationChangeInInternalOverride - NestedCalldataArrayAbiReencodingSizeValidation - SignedImmutables. It is used by: - ^0.8.4 (contracts/interfaces/IAdmin.sol#2) - ^0.8.4 (contracts/interfaces/IAdminV2.sol#2) - ^0.8.4 (contracts/interfaces/IAdminV2Errors.sol#2) - ^0.8.4 (contracts/interfaces/IFastBridge.sol#2) - ^0.8.4 (contracts/interfaces/IFastBridgeV2.sol#2) - ^0.8.4 (contracts/interfaces/IFastBridgeV2Errors.sol#2) - ^0.8.4 (contracts/interfaces/IMulticallTarget.sol#2) - ^0.8.4 (contracts/interfaces/IZapRecipient.sol#2) - ^0.8.4 (contracts/libs/BridgeTransactionV2.sol#2) - ^0.8.4 (contracts/libs/Errors.sol#2) - ^0.8.4 (contracts/libs/ZapDataV1.sol#2) - ^0.8.4 (contracts/utils/MulticallTarget.sol#2)
@@ -1,5 +1,5 @@
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;
pragma solidity ^0.8.4;

Check warning

Code scanning / Slither

Incorrect versions of Solidity

Version constraint ^0.8.4 contains known severe issues (https://solidity.readthedocs.io/en/latest/bugs.html) - FullInlinerNonExpressionSplitArgumentEvaluationOrder - MissingSideEffectsOnSelectorAccess - AbiReencodingHeadOverflowWithStaticArrayCleanup - DirtyBytesArrayToStorage - DataLocationChangeInInternalOverride - NestedCalldataArrayAbiReencodingSizeValidation - SignedImmutables. It is used by: - ^0.8.4 (contracts/interfaces/IAdmin.sol#2) - ^0.8.4 (contracts/interfaces/IFastBridge.sol#2) - ^0.8.4 (contracts/interfaces/IMulticallTarget.sol#2) - ^0.8.4 (contracts/libs/Errors.sol#2) - ^0.8.4 (contracts/utils/MulticallTarget.sol#2)
Comment on lines 227 to 232
function zapNative(bytes calldata encodedTx) internal pure returns (uint256 zapNative_) {
// Load 32 bytes from the offset. No shift is applied, as we need the full 256 bits.
assembly {
zapNative_ := calldataload(add(encodedTx.offset, OFFSET_ZAP_NATIVE))
}
}

Check warning

Code scanning / Slither

Assembly usage

BridgeTransactionV2Lib.zapNative(bytes) (contracts/libs/BridgeTransactionV2.sol#227-232) uses assembly - INLINE ASM (contracts/libs/BridgeTransactionV2.sol#229-231)
@dwasse dwasse merged commit 57efbd0 into feat/relayer-arb-call Nov 26, 2024
46 of 49 checks passed
@dwasse dwasse deleted the feat/guard-v2 branch November 26, 2024 19:50
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
go Pull requests that update Go code javascript Pull requests that update Javascript code needs-go-generate-contrib/promexporter needs-go-generate-services/rfq size/m
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants