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

RFQ API: add GET /ack endpoint #2643

Merged
merged 24 commits into from
May 20, 2024
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
Show all changes
24 commits
Select commit Hold shift + click to select a range
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
16 changes: 14 additions & 2 deletions services/rfq/api/config/config.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import (
"fmt"
"os"
"path/filepath"
"time"

"github.com/jftuga/ellipsis"
"gopkg.in/yaml.v2"
Expand All @@ -21,8 +22,19 @@ type Config struct {
Database DatabaseConfig `yaml:"database"`
OmniRPCURL string `yaml:"omnirpc_url"`
// bridges is a map of chainid->address
Bridges map[uint32]string `yaml:"bridges"`
Port string `yaml:"port"`
Bridges map[uint32]string `yaml:"bridges"`
Port string `yaml:"port"`
RelayAckTimeout time.Duration `yaml:"relay_ack_timeout"`
Copy link
Contributor

Choose a reason for hiding this comment

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

Add a YAML tag for RelayAckTimeout.

- RelayAckTimeout time.Duration     `yaml:"relay_ack_timeout"`
+ RelayAckTimeout time.Duration     `yaml:"relay_ack_timeout" yaml:"relay_ack_timeout"`

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.

Suggested change
RelayAckTimeout time.Duration `yaml:"relay_ack_timeout"`
RelayAckTimeout time.Duration `yaml:"relay_ack_timeout" yaml:"relay_ack_timeout"`

}

const defaultRelayAckTimeout = 5 * time.Second
dwasse marked this conversation as resolved.
Show resolved Hide resolved

// GetRelayAckTimeout returns the relay ack timeout.
func (c Config) GetRelayAckTimeout() time.Duration {
if c.RelayAckTimeout == 0 {
return defaultRelayAckTimeout
}
return c.RelayAckTimeout
}

// LoadConfig loads the config from the given path.
Expand Down
48 changes: 47 additions & 1 deletion services/rfq/api/rest/server.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
"context"
"fmt"
"net/http"
"sync"
"time"

"github.com/ipfs/go-log"
Expand All @@ -25,6 +26,7 @@
"github.com/synapsecns/sanguine/services/rfq/api/docs"
"github.com/synapsecns/sanguine/services/rfq/api/model"
"github.com/synapsecns/sanguine/services/rfq/contracts/fastbridge"
"github.com/synapsecns/sanguine/services/rfq/relayer/relapi"
)

// QuoterAPIServer is a struct that holds the configuration, database connection, gin engine, RPC client, metrics handler, and fast bridge contracts.
Expand All @@ -37,6 +39,11 @@
handler metrics.Handler
fastBridgeContracts map[uint32]*fastbridge.FastBridge
roleCache map[uint32]*ttlcache.Cache[string, bool]
// relayAckCache contains a set of transactionID values that reflect
// transactions that have been acked for relay
relayAckCache *ttlcache.Cache[string, bool]
// ackMux is a mutex used to ensure that only one transaction id can be acked at a time.
ackMux sync.Mutex
}

// NewAPI holds the configuration, database connection, gin engine, RPC client, metrics handler, and fast bridge contracts.
Expand Down Expand Up @@ -80,27 +87,40 @@
ttlcache.WithTTL[string, bool](cacheInterval),
)
roleCache := roles[chainID]

go roleCache.Start()
go func() {
<-ctx.Done()
roleCache.Stop()
}()
}

// create the relay ack cache
relayAckCache := ttlcache.New[string, bool](
dwasse marked this conversation as resolved.
Show resolved Hide resolved
ttlcache.WithTTL[string, bool](cfg.GetRelayAckTimeout()),
ttlcache.WithDisableTouchOnHit[string, bool](),
)
go relayAckCache.Start()
go func() {
<-ctx.Done()
relayAckCache.Stop()
}()

return &QuoterAPIServer{
cfg: cfg,
db: store,
omnirpcClient: omniRPCClient,
handler: handler,
fastBridgeContracts: bridges,
roleCache: roles,
relayAckCache: relayAckCache,
ackMux: sync.Mutex{},
}, nil
}

// QuoteRoute is the API endpoint for handling quote related requests.
const (
QuoteRoute = "/quotes"
AckRoute = "/ack"
cacheInterval = time.Minute
)

Expand All @@ -120,6 +140,7 @@
// GET routes without the AuthMiddleware
// engine.PUT("/quotes", h.ModifyQuote)
engine.GET(QuoteRoute, h.GetQuotes)
engine.GET(AckRoute, r.GetRelayAck)
dwasse marked this conversation as resolved.
Show resolved Hide resolved

r.engine = engine

Expand Down Expand Up @@ -188,3 +209,28 @@
c.Next()
}
}

// GetRelayAck checks if a relay is pending or not.
func (r *QuoterAPIServer) GetRelayAck(c *gin.Context) {
Copy link
Contributor

Choose a reason for hiding this comment

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

Can we update this w/ swagger docs? You should see this in handler.go and be able to regenerate. Just consists of copying the comments over from anything in handler.go and rerunning go generate followed by a yarn docusaurus gen-api-docs all from docs/bridge

Would also be good to add the endpoint here

and document the step here

transactionID := c.Query("id")
if transactionID == "" {
c.JSON(http.StatusBadRequest, gin.H{"error": "Must specify 'id'"})
return
}

Check warning on line 219 in services/rfq/api/rest/server.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/api/rest/server.go#L217-L219

Added lines #L217 - L219 were not covered by tests

// If the tx id is already in the cache, it should not be relayed.
// Otherwise, insert into the cache.
r.ackMux.Lock()
ack := r.relayAckCache.Get(transactionID)
shouldRelay := ack == nil
if shouldRelay {
r.relayAckCache.Set(transactionID, true, ttlcache.DefaultTTL)
}
r.ackMux.Unlock()

resp := relapi.GetRelayAckResponse{
TxID: transactionID,
ShouldRelay: shouldRelay,
}
c.JSON(http.StatusOK, resp)
}
45 changes: 45 additions & 0 deletions services/rfq/api/rest/server_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ import (
"github.com/ethereum/go-ethereum/crypto"
"github.com/synapsecns/sanguine/ethergo/signer/wallet"
"github.com/synapsecns/sanguine/services/rfq/api/model"
"github.com/synapsecns/sanguine/services/rfq/relayer/relapi"
)

func (c *ServerSuite) TestNewQuoterAPIServer() {
Expand Down Expand Up @@ -203,6 +204,50 @@ func (c *ServerSuite) TestPutAndGetQuoteByRelayer() {
c.Assert().True(found, "Newly added quote not found")
}

func (c *ServerSuite) TestGetAck() {
c.startQuoterAPIServer()

// Send GET request
client := &http.Client{}
testTxID := "0x123"
req, err := http.NewRequestWithContext(c.GetTestContext(), http.MethodGet, fmt.Sprintf("http://localhost:%d/ack?id=%s", c.port, testTxID), nil)
c.Require().NoError(err)
resp, err := client.Do(req)
c.Require().NoError(err)
c.Equal(http.StatusOK, resp.StatusCode)

// Expect ack with shouldRelay=true
var result relapi.GetRelayAckResponse
err = json.NewDecoder(resp.Body).Decode(&result)
c.Require().NoError(err)
expectedResult := relapi.GetRelayAckResponse{
TxID: testTxID,
ShouldRelay: true,
}
c.Equal(expectedResult, result)
err = resp.Body.Close()
c.Require().NoError(err)

// Send another request with same txID
req, err = http.NewRequestWithContext(c.GetTestContext(), http.MethodGet, fmt.Sprintf("http://localhost:%d/ack?id=%s", c.port, testTxID), nil)
c.Require().NoError(err)
resp, err = client.Do(req)
c.Require().NoError(err)
c.Equal(http.StatusOK, resp.StatusCode)

// Expect ack with shouldRelay=false
err = json.NewDecoder(resp.Body).Decode(&result)
c.Require().NoError(err)
expectedResult = relapi.GetRelayAckResponse{
TxID: testTxID,
ShouldRelay: false,
}
c.Equal(expectedResult, result)
err = resp.Body.Close()
c.Require().NoError(err)
c.GetTestContext().Done()
}

// startQuoterAPIServer starts the API server and waits for it to initialize.
func (c *ServerSuite) startQuoterAPIServer() {
go func() {
Expand Down
2 changes: 1 addition & 1 deletion services/rfq/relayer/relapi/handler.go
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,7 @@
func (h *Handler) GetQuoteRequestStatusByTxID(c *gin.Context) {
txIDStr := c.Query("id")
if txIDStr == "" {
c.JSON(http.StatusBadRequest, gin.H{"error": "Must specify 'txID'"})
c.JSON(http.StatusBadRequest, gin.H{"error": "Must specify 'id'"})

Check warning on line 63 in services/rfq/relayer/relapi/handler.go

View check run for this annotation

Codecov / codecov/patch

services/rfq/relayer/relapi/handler.go#L63

Added line #L63 was not covered by tests
return
}

Expand Down
6 changes: 6 additions & 0 deletions services/rfq/relayer/relapi/model.go
Original file line number Diff line number Diff line change
Expand Up @@ -15,3 +15,9 @@ type GetTxRetryResponse struct {
Nonce uint64 `json:"nonce"`
GasAmount string `json:"gas_amount"`
}

// GetRelayAckResponse contains the schema for a POST /relay/ack response.
type GetRelayAckResponse struct {
TxID string `json:"tx_id"`
ShouldRelay bool `json:"should_relay"`
}
Loading