Skip to content

Commit

Permalink
http resolver
Browse files Browse the repository at this point in the history
  • Loading branch information
saolyn committed Jan 9, 2024
1 parent 75bbeb6 commit 2c2d9bd
Show file tree
Hide file tree
Showing 2 changed files with 74 additions and 0 deletions.
1 change: 1 addition & 0 deletions validator/client/BUILD.bazel
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ go_library(
"log.go",
"metrics.go",
"multiple_endpoints_grpc_resolver.go",
"multiple_endpoints_http_resolver.go",
"propose.go",
"propose_protect.go",
"registration.go",
Expand Down
73 changes: 73 additions & 0 deletions validator/client/multiple_endpoints_http_resolver.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,73 @@
package client

import (
"fmt"
"net/http"

"github.com/prysmaticlabs/prysm/v4/network/httputil"
)

// MultipleEndpointsHTTPResolver is a custom resolver for HTTP clients that supports multiple addresses.
type MultipleEndpointsHTTPResolver struct {
addresses []string
currentIdx int
client *http.Client
}

// NewMultipleEndpointsHTTPResolver creates a new instance of MultipleEndpointsHTTPResolver.
func NewMultipleEndpointsHTTPResolver(addresses []string) *MultipleEndpointsHTTPResolver {
return &MultipleEndpointsHTTPResolver{
addresses: addresses,
currentIdx: 0,
client: &http.Client{},
}
}

func (r *MultipleEndpointsHTTPResolver) HttpMiddleware(next http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) {
// Attempt to send the request to the current endpoint
err := r.sendRequest(req, w)

// Retry logic: switch to the next endpoint and retry if there is an error
for i := 0; i < len(r.addresses)-1 && err != nil; i++ {
r.switchEndpoint()
err = r.sendRequest(req, w)
}

// Handle the error or continue processing the response
if err != nil {
// Handle the error (e.g., log it, return an error response, etc.)
httputil.HandleError(w, fmt.Sprintf("failed to send request: %v", err), http.StatusInternalServerError)
return
}

// Continue processing the response if there is no error
next.ServeHTTP(w, req)
})
}

// sendRequest sends the HTTP request to the current endpoint.
func (r *MultipleEndpointsHTTPResolver) sendRequest(req *http.Request, w http.ResponseWriter) error {
// Update the request URL with the current endpoint
req.URL.Host = r.resolveEndpoint()

// Send the HTTP request using the client
resp, err := r.client.Do(req)
if err != nil {
// Optionally handle specific errors or log them
httputil.HandleError(w, fmt.Sprintf("error sending request to %s: %v\n", r.resolveEndpoint(), err), http.StatusInternalServerError)
return err
}
defer resp.Body.Close()
return nil
}

// resolveEndpoint returns the current endpoint based on the resolver's state.
func (r *MultipleEndpointsHTTPResolver) resolveEndpoint() string {
return r.addresses[r.currentIdx]
}

// switchToNextEndpoint switches to the next available endpoint.
func (r *MultipleEndpointsHTTPResolver) switchEndpoint() {
r.currentIdx = (r.currentIdx + 1) % len(r.addresses)
}

0 comments on commit 2c2d9bd

Please sign in to comment.