From 4632ac21337ca71fbe68018c3a7272b9a866dc9e Mon Sep 17 00:00:00 2001 From: Rohit Nayak Date: Wed, 9 Oct 2024 22:19:52 +0200 Subject: [PATCH 1/3] Use default timeout if buffering is not enabled and prevent a panic Signed-off-by: Rohit Nayak --- go/vt/vtgate/plan_execute.go | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/go/vt/vtgate/plan_execute.go b/go/vt/vtgate/plan_execute.go index 34abe5a2aa3..de5c0b16d03 100644 --- a/go/vt/vtgate/plan_execute.go +++ b/go/vt/vtgate/plan_execute.go @@ -104,7 +104,10 @@ func (e *Executor) newExecute( // based on the buffering configuration. This way we should be able to perform the max retries // within the given window of time for most queries and we should not end up waiting too long // after the traffic switch fails or the buffer window has ended, retrying old queries. - timeout := e.resolver.scatterConn.gateway.buffer.GetConfig().MaxFailoverDuration / (MaxBufferingRetries - 1) + timeout := 30 * time.Second + if e.resolver.scatterConn.gateway.buffer != nil { + timeout = e.resolver.scatterConn.gateway.buffer.GetConfig().MaxFailoverDuration / (MaxBufferingRetries - 1) + } if waitForNewerVSchema(ctx, e, lastVSchemaCreated, timeout) { vs = e.VSchema() lastVSchemaCreated = vs.GetCreated() From cd0a3924f17273b0c6b0071e2c4a7eb30fda9639 Mon Sep 17 00:00:00 2001 From: Rohit Nayak Date: Thu, 10 Oct 2024 21:10:10 +0200 Subject: [PATCH 2/3] Address code review Signed-off-by: Rohit Nayak --- go/vt/vtgate/plan_execute.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/go/vt/vtgate/plan_execute.go b/go/vt/vtgate/plan_execute.go index de5c0b16d03..0041aca895d 100644 --- a/go/vt/vtgate/plan_execute.go +++ b/go/vt/vtgate/plan_execute.go @@ -105,7 +105,7 @@ func (e *Executor) newExecute( // within the given window of time for most queries and we should not end up waiting too long // after the traffic switch fails or the buffer window has ended, retrying old queries. timeout := 30 * time.Second - if e.resolver.scatterConn.gateway.buffer != nil { + if e.resolver.scatterConn.gateway.buffer != nil && e.resolver.scatterConn.gateway.buffer.GetConfig() != nil { timeout = e.resolver.scatterConn.gateway.buffer.GetConfig().MaxFailoverDuration / (MaxBufferingRetries - 1) } if waitForNewerVSchema(ctx, e, lastVSchemaCreated, timeout) { From 67ee01460b6ba56e4c81a067958e54874cd604bb Mon Sep 17 00:00:00 2001 From: Harshit Gangal Date: Fri, 11 Oct 2024 12:10:04 +0530 Subject: [PATCH 3/3] test: added unit test Signed-off-by: Harshit Gangal --- go/vt/vtgate/executor_test.go | 14 ++++++++++++++ go/vt/vtgate/plan_execute.go | 4 +++- 2 files changed, 17 insertions(+), 1 deletion(-) diff --git a/go/vt/vtgate/executor_test.go b/go/vt/vtgate/executor_test.go index 1cd58b31cc3..72cb6e4b4e7 100644 --- a/go/vt/vtgate/executor_test.go +++ b/go/vt/vtgate/executor_test.go @@ -1520,6 +1520,20 @@ func TestExecutorUnrecognized(t *testing.T) { require.Error(t, err, "unrecognized statement: invalid statement'") } +func TestExecutorDeniedErrorNoBuffer(t *testing.T) { + executor, sbc1, _, _, ctx := createExecutorEnv(t) + sbc1.EphemeralShardErr = errors.New("enforce denied tables") + + vschemaWaitTimeout = 500 * time.Millisecond + + session := NewAutocommitSession(&vtgatepb.Session{TargetString: "@primary"}) + startExec := time.Now() + _, err := executor.Execute(ctx, nil, "TestExecutorDeniedErrorNoBuffer", session, "select * from user", nil) + require.NoError(t, err, "enforce denied tables not buffered") + endExec := time.Now() + require.GreaterOrEqual(t, endExec.Sub(startExec).Milliseconds(), int64(500)) +} + // TestVSchemaStats makes sure the building and displaying of the // VSchemaStats works. func TestVSchemaStats(t *testing.T) { diff --git a/go/vt/vtgate/plan_execute.go b/go/vt/vtgate/plan_execute.go index 0041aca895d..1c0915470ef 100644 --- a/go/vt/vtgate/plan_execute.go +++ b/go/vt/vtgate/plan_execute.go @@ -36,6 +36,8 @@ import ( type planExec func(ctx context.Context, plan *engine.Plan, vc *vcursorImpl, bindVars map[string]*querypb.BindVariable, startTime time.Time) error type txResult func(sqlparser.StatementType, *sqltypes.Result) error +var vschemaWaitTimeout = 30 * time.Second + func waitForNewerVSchema(ctx context.Context, e *Executor, lastVSchemaCreated time.Time, timeout time.Duration) bool { pollingInterval := 10 * time.Millisecond waitCtx, cancel := context.WithTimeout(ctx, timeout) @@ -104,7 +106,7 @@ func (e *Executor) newExecute( // based on the buffering configuration. This way we should be able to perform the max retries // within the given window of time for most queries and we should not end up waiting too long // after the traffic switch fails or the buffer window has ended, retrying old queries. - timeout := 30 * time.Second + timeout := vschemaWaitTimeout if e.resolver.scatterConn.gateway.buffer != nil && e.resolver.scatterConn.gateway.buffer.GetConfig() != nil { timeout = e.resolver.scatterConn.gateway.buffer.GetConfig().MaxFailoverDuration / (MaxBufferingRetries - 1) }