Skip to content

Commit

Permalink
fix: subroute blocking if all client data was already read (#194)
Browse files Browse the repository at this point in the history
  • Loading branch information
ydylla authored May 25, 2024
1 parent f049165 commit 83ccc7e
Showing 1 changed file with 17 additions and 8 deletions.
25 changes: 17 additions & 8 deletions layer4/routes.go
Original file line number Diff line number Diff line change
Expand Up @@ -107,17 +107,26 @@ func (routes RouteList) Compile(logger *zap.Logger, matchingTimeout time.Duratio
if err != nil {
return err
}
isFirstPrefetch := true
for { // retry prefetching and matching routes until timeout
err = cx.prefetch()
if err != nil {
logFunc := logger.Error
if errors.Is(err, os.ErrDeadlineExceeded) {
err = ErrMatchingTimeout
logFunc = logger.Warn

// Do not call prefetch if this is the first loop iteration and there already is some data available,
// since this means we are at the start of a subroute handler and previous prefetch calls likely already fetched all bytes available from the client.
// Which means it would block the subroute handler. In the second iteration (if no subroute routes match) blocking is the correct behaviour.
if !isFirstPrefetch || cx.buf == nil || len(cx.buf[cx.offset:]) == 0 {
err = cx.prefetch()
isFirstPrefetch = false
if err != nil {
logFunc := logger.Error
if errors.Is(err, os.ErrDeadlineExceeded) {
err = ErrMatchingTimeout
logFunc = logger.Warn
}
logFunc("matching connection", zap.String("remote", cx.RemoteAddr().String()), zap.Error(err))
return nil // return nil so the error does not get logged again
}
logFunc("matching connection", zap.String("remote", cx.RemoteAddr().String()), zap.Error(err))
return nil // return nil so the error does not get logged again
}

for _, route := range routes {
// A route must match at least one of the matcher sets
matched, err := route.matcherSets.AnyMatch(cx)
Expand Down

0 comments on commit 83ccc7e

Please sign in to comment.