mirror of
https://github.com/grafana/grafana.git
synced 2025-02-25 18:55:37 -06:00
Chore: Add context util that allow to provide cause of cancellation (#53918)
This commit is contained in:
parent
6804a8c9cc
commit
736d035c65
44
pkg/util/contextutil.go
Normal file
44
pkg/util/contextutil.go
Normal file
@ -0,0 +1,44 @@
|
||||
package util
|
||||
|
||||
import (
|
||||
"context"
|
||||
"sync"
|
||||
|
||||
"github.com/hashicorp/go-multierror"
|
||||
)
|
||||
|
||||
// this is a decorator for a regular context that contains a custom error and returns the
|
||||
type contextWithCancellableReason struct {
|
||||
context.Context
|
||||
mtx sync.Mutex
|
||||
err error
|
||||
}
|
||||
|
||||
func (c *contextWithCancellableReason) Err() error {
|
||||
c.mtx.Lock()
|
||||
defer c.mtx.Unlock()
|
||||
if c.err != nil {
|
||||
return multierror.Append(c.Context.Err(), c.err)
|
||||
}
|
||||
return c.Context.Err()
|
||||
}
|
||||
|
||||
type CancelCauseFunc func(error)
|
||||
|
||||
// WithCancelCause creates a cancellable context that can be cancelled with a custom reason
|
||||
func WithCancelCause(parent context.Context) (context.Context, CancelCauseFunc) {
|
||||
ctx, cancel := context.WithCancel(parent)
|
||||
errOnce := sync.Once{}
|
||||
result := &contextWithCancellableReason{
|
||||
Context: ctx,
|
||||
}
|
||||
cancelFn := func(reason error) {
|
||||
errOnce.Do(func() {
|
||||
result.mtx.Lock()
|
||||
result.err = reason
|
||||
result.mtx.Unlock()
|
||||
cancel()
|
||||
})
|
||||
}
|
||||
return result, cancelFn
|
||||
}
|
46
pkg/util/contextutil_test.go
Normal file
46
pkg/util/contextutil_test.go
Normal file
@ -0,0 +1,46 @@
|
||||
package util
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"testing"
|
||||
|
||||
"github.com/stretchr/testify/require"
|
||||
)
|
||||
|
||||
func TestWithCancelWithReason(t *testing.T) {
|
||||
t.Run("should add custom reason to the standard error", func(t *testing.T) {
|
||||
expected := errors.New("test-err")
|
||||
ctx, fn := WithCancelCause(context.Background())
|
||||
fn(expected)
|
||||
select {
|
||||
case <-ctx.Done():
|
||||
default:
|
||||
require.Fail(t, "the context was not cancelled")
|
||||
}
|
||||
require.ErrorIs(t, ctx.Err(), expected)
|
||||
require.ErrorIs(t, ctx.Err(), context.Canceled)
|
||||
})
|
||||
|
||||
t.Run("should return only the first reason if called multiple times", func(t *testing.T) {
|
||||
expected := errors.New("test-err")
|
||||
ctx, fn := WithCancelCause(context.Background())
|
||||
fn(expected)
|
||||
fn(errors.New("other error"))
|
||||
require.ErrorIs(t, ctx.Err(), expected)
|
||||
})
|
||||
|
||||
t.Run("should return only the first reason if called multiple times", func(t *testing.T) {
|
||||
expected := errors.New("test-err")
|
||||
ctx, fn := WithCancelCause(context.Background())
|
||||
fn(expected)
|
||||
fn(errors.New("other error"))
|
||||
require.ErrorIs(t, ctx.Err(), expected)
|
||||
})
|
||||
|
||||
t.Run("should return context.Canceled if no reason provided", func(t *testing.T) {
|
||||
ctx, fn := WithCancelCause(context.Background())
|
||||
fn(nil)
|
||||
require.Equal(t, ctx.Err(), context.Canceled)
|
||||
})
|
||||
}
|
Loading…
Reference in New Issue
Block a user