xds: add ConfigSelector to support RouteAction timeouts (#3991)
This commit is contained in:
93
internal/resolver/config_selector.go
Normal file
93
internal/resolver/config_selector.go
Normal file
@ -0,0 +1,93 @@
|
||||
/*
|
||||
*
|
||||
* Copyright 2020 gRPC authors.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*
|
||||
*/
|
||||
|
||||
// Package resolver provides internal resolver-related functionality.
|
||||
package resolver
|
||||
|
||||
import (
|
||||
"context"
|
||||
"sync"
|
||||
|
||||
"google.golang.org/grpc/internal/serviceconfig"
|
||||
"google.golang.org/grpc/resolver"
|
||||
)
|
||||
|
||||
// ConfigSelector controls what configuration to use for every RPC.
|
||||
type ConfigSelector interface {
|
||||
// Selects the configuration for the RPC.
|
||||
SelectConfig(RPCInfo) *RPCConfig
|
||||
}
|
||||
|
||||
// RPCInfo contains RPC information needed by a ConfigSelector.
|
||||
type RPCInfo struct {
|
||||
// Context is the user's context for the RPC and contains headers and
|
||||
// application timeout. It is passed for interception purposes and for
|
||||
// efficiency reasons. SelectConfig should not be blocking.
|
||||
Context context.Context
|
||||
Method string // i.e. "/Service/Method"
|
||||
}
|
||||
|
||||
// RPCConfig describes the configuration to use for each RPC.
|
||||
type RPCConfig struct {
|
||||
// The context to use for the remainder of the RPC; can pass info to LB
|
||||
// policy or affect timeout or metadata.
|
||||
Context context.Context
|
||||
MethodConfig serviceconfig.MethodConfig // configuration to use for this RPC
|
||||
OnCommitted func() // Called when the RPC has been committed (retries no longer possible)
|
||||
}
|
||||
|
||||
type csKeyType string
|
||||
|
||||
const csKey = csKeyType("grpc.internal.resolver.configSelector")
|
||||
|
||||
// SetConfigSelector sets the config selector in state and returns the new
|
||||
// state.
|
||||
func SetConfigSelector(state resolver.State, cs ConfigSelector) resolver.State {
|
||||
state.Attributes = state.Attributes.WithValues(csKey, cs)
|
||||
return state
|
||||
}
|
||||
|
||||
// GetConfigSelector retrieves the config selector from state, if present, and
|
||||
// returns it or nil if absent.
|
||||
func GetConfigSelector(state resolver.State) ConfigSelector {
|
||||
cs, _ := state.Attributes.Value(csKey).(ConfigSelector)
|
||||
return cs
|
||||
}
|
||||
|
||||
// SafeConfigSelector allows for safe switching of ConfigSelector
|
||||
// implementations such that previous values are guaranteed to not be in use
|
||||
// when UpdateConfigSelector returns.
|
||||
type SafeConfigSelector struct {
|
||||
mu sync.RWMutex
|
||||
cs ConfigSelector
|
||||
}
|
||||
|
||||
// UpdateConfigSelector swaps to the provided ConfigSelector and blocks until
|
||||
// all uses of the previous ConfigSelector have completed.
|
||||
func (scs *SafeConfigSelector) UpdateConfigSelector(cs ConfigSelector) {
|
||||
scs.mu.Lock()
|
||||
defer scs.mu.Unlock()
|
||||
scs.cs = cs
|
||||
}
|
||||
|
||||
// SelectConfig defers to the current ConfigSelector in scs.
|
||||
func (scs *SafeConfigSelector) SelectConfig(r RPCInfo) *RPCConfig {
|
||||
scs.mu.RLock()
|
||||
defer scs.mu.RUnlock()
|
||||
return scs.cs.SelectConfig(r)
|
||||
}
|
153
internal/resolver/config_selector_test.go
Normal file
153
internal/resolver/config_selector_test.go
Normal file
@ -0,0 +1,153 @@
|
||||
/*
|
||||
*
|
||||
* Copyright 2020 gRPC authors.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*
|
||||
*/
|
||||
|
||||
package resolver
|
||||
|
||||
import (
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
"github.com/google/go-cmp/cmp"
|
||||
"google.golang.org/grpc/internal/grpctest"
|
||||
"google.golang.org/grpc/internal/serviceconfig"
|
||||
)
|
||||
|
||||
type s struct {
|
||||
grpctest.Tester
|
||||
}
|
||||
|
||||
func Test(t *testing.T) {
|
||||
grpctest.RunSubTests(t, s{})
|
||||
}
|
||||
|
||||
type fakeConfigSelector struct {
|
||||
selectConfig func(RPCInfo) *RPCConfig
|
||||
}
|
||||
|
||||
func (f *fakeConfigSelector) SelectConfig(r RPCInfo) *RPCConfig {
|
||||
return f.selectConfig(r)
|
||||
}
|
||||
|
||||
func (s) TestSafeConfigSelector(t *testing.T) {
|
||||
testRPCInfo := RPCInfo{Method: "test method"}
|
||||
|
||||
retChan1 := make(chan *RPCConfig)
|
||||
retChan2 := make(chan *RPCConfig)
|
||||
|
||||
one := 1
|
||||
two := 2
|
||||
|
||||
resp1 := &RPCConfig{MethodConfig: serviceconfig.MethodConfig{MaxReqSize: &one}}
|
||||
resp2 := &RPCConfig{MethodConfig: serviceconfig.MethodConfig{MaxReqSize: &two}}
|
||||
|
||||
cs1Called := make(chan struct{})
|
||||
cs2Called := make(chan struct{})
|
||||
|
||||
cs1 := &fakeConfigSelector{
|
||||
selectConfig: func(r RPCInfo) *RPCConfig {
|
||||
cs1Called <- struct{}{}
|
||||
if diff := cmp.Diff(r, testRPCInfo); diff != "" {
|
||||
t.Errorf("SelectConfig(%v) called; want %v\n Diffs:\n%s", r, testRPCInfo, diff)
|
||||
}
|
||||
return <-retChan1
|
||||
},
|
||||
}
|
||||
cs2 := &fakeConfigSelector{
|
||||
selectConfig: func(r RPCInfo) *RPCConfig {
|
||||
cs2Called <- struct{}{}
|
||||
if diff := cmp.Diff(r, testRPCInfo); diff != "" {
|
||||
t.Errorf("SelectConfig(%v) called; want %v\n Diffs:\n%s", r, testRPCInfo, diff)
|
||||
}
|
||||
return <-retChan2
|
||||
},
|
||||
}
|
||||
|
||||
scs := &SafeConfigSelector{}
|
||||
scs.UpdateConfigSelector(cs1)
|
||||
|
||||
cs1Returned := make(chan struct{})
|
||||
go func() {
|
||||
got := scs.SelectConfig(testRPCInfo) // blocks until send to retChan1
|
||||
if got != resp1 {
|
||||
t.Errorf("SelectConfig(%v) = %v; want %v", testRPCInfo, got, resp1)
|
||||
}
|
||||
close(cs1Returned)
|
||||
}()
|
||||
|
||||
// cs1 is blocked but should be called
|
||||
select {
|
||||
case <-time.After(500 * time.Millisecond):
|
||||
t.Fatalf("timed out waiting for cs1 to be called")
|
||||
case <-cs1Called:
|
||||
}
|
||||
|
||||
// swap in cs2 now that cs1 is called
|
||||
csSwapped := make(chan struct{})
|
||||
go func() {
|
||||
// wait awhile first to ensure cs1 could be called below.
|
||||
time.Sleep(50 * time.Millisecond)
|
||||
scs.UpdateConfigSelector(cs2) // Blocks until cs1 done
|
||||
close(csSwapped)
|
||||
}()
|
||||
|
||||
// Allow cs1 to return and cs2 to eventually be swapped in.
|
||||
retChan1 <- resp1
|
||||
|
||||
cs1Done := false // set when cs2 is first called
|
||||
for dl := time.Now().Add(150 * time.Millisecond); !time.Now().After(dl); {
|
||||
gotConfigChan := make(chan *RPCConfig)
|
||||
go func() {
|
||||
gotConfigChan <- scs.SelectConfig(testRPCInfo)
|
||||
}()
|
||||
select {
|
||||
case <-time.After(500 * time.Millisecond):
|
||||
t.Fatalf("timed out waiting for cs1 or cs2 to be called")
|
||||
case <-cs1Called:
|
||||
// Initially, before swapping to cs2, cs1 should be called
|
||||
retChan1 <- resp1
|
||||
go func() { <-gotConfigChan }()
|
||||
if cs1Done {
|
||||
t.Fatalf("cs1 called after cs2")
|
||||
}
|
||||
case <-cs2Called:
|
||||
// Success! the new config selector is being called
|
||||
if !cs1Done {
|
||||
select {
|
||||
case <-csSwapped:
|
||||
case <-time.After(50 * time.Millisecond):
|
||||
t.Fatalf("timed out waiting for UpdateConfigSelector to return")
|
||||
}
|
||||
select {
|
||||
case <-cs1Returned:
|
||||
case <-time.After(50 * time.Millisecond):
|
||||
t.Fatalf("timed out waiting for cs1 to return")
|
||||
}
|
||||
cs1Done = true
|
||||
}
|
||||
retChan2 <- resp2
|
||||
got := <-gotConfigChan
|
||||
if diff := cmp.Diff(got, resp2); diff != "" {
|
||||
t.Fatalf("SelectConfig(%v) = %v; want %v\n Diffs:\n%s", testRPCInfo, got, resp2, diff)
|
||||
}
|
||||
}
|
||||
time.Sleep(10 * time.Millisecond)
|
||||
}
|
||||
if !cs1Done {
|
||||
t.Fatalf("timed out waiting for cs2 to be called")
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user