kubernetes/test/utils/ktesting/withcontext.go
Patrick Ohly 3b63fe83a0 ktesting: support cancellation after WithoutCancel
Not canceling the parent context made sense, but the new context should
be cancelable like any other TContext. Found when passing tCtx.WithoutCancel()
to StartTestServer and the tear-down function got stuck because it couldn't
cancel the context.
2026-03-17 07:50:18 +01:00

73 lines
2.1 KiB
Go

/*
Copyright 2023 The Kubernetes 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 ktesting
import (
"context"
"time"
"k8s.io/klog/v2"
)
// WithCancel sets up cancellation in a [TContext.Cleanup] callback and
// constructs a new TContext where [TContext.Cancel] cancels only the new
// context.
func (tCtx TContext) WithCancel() TContext {
ctx, cancel := context.WithCancelCause(tCtx)
tCtx.Context = ctx
tCtx.cancel = func(cause string) {
var cancelCause error
if cause != "" {
cancelCause = canceledError(cause)
}
cancel(cancelCause)
}
return tCtx
}
// WithoutCancel causes the returned context to ignore cancellation of its parent.
// Calling Cancel will only cancel the new context.
// This matches [context.WithoutCancel].
func (tCtx TContext) WithoutCancel() TContext {
ctx := context.WithoutCancel(tCtx)
tCtx.Context = ctx
tCtx = tCtx.WithCancel() // Re-create a cancelable TContext.
return tCtx
}
// WithTimeout sets up new context with a timeout. Canceling the timeout gets
// registered in a cleanup callback. [TContext.Cancel] cancels only
// the new context. The cause is used as reason why the context is canceled
// once the timeout is reached. It may be empty, in which case the usual
// "context canceled" error is used.
func (tCtx TContext) WithTimeout(timeout time.Duration, timeoutCause string) TContext {
ctx, cancel := withTimeout(tCtx, tCtx.TB(), timeout, timeoutCause)
tCtx.Context = ctx
tCtx.cancel = cancel
return tCtx
}
// WithLogger constructs a new context with a different logger.
func (tCtx TContext) WithLogger(logger klog.Logger) TContext {
ctx := klog.NewContext(tCtx, logger)
tCtx.Context = ctx
return tCtx
}