kratos/internal/context/context_test.go

280 lines
6.7 KiB

package context
import (
"context"
"errors"
"reflect"
"testing"
"time"
)
func TestContext(t *testing.T) {
type ctxKey1 struct{}
type ctxKey2 struct{}
ctx1 := context.WithValue(context.Background(), ctxKey1{}, "https://github.com/go-kratos/")
ctx2 := context.WithValue(context.Background(), ctxKey2{}, "https://go-kratos.dev/")
ctx, cancel := Merge(ctx1, ctx2)
defer cancel()
got := ctx.Value(ctxKey1{})
value1, ok := got.(string)
if !ok {
t.Errorf("expect %v, got %v", true, ok)
}
if !reflect.DeepEqual(value1, "https://github.com/go-kratos/") {
t.Errorf("expect %v, got %v", "https://github.com/go-kratos/", value1)
}
got2 := ctx.Value(ctxKey2{})
value2, ok := got2.(string)
if !ok {
t.Errorf("expect %v, got %v", true, ok)
}
if !reflect.DeepEqual("https://go-kratos.dev/", value2) {
t.Errorf("expect %v, got %v", "https://go-kratos.dev/", value2)
}
t.Log(value1)
t.Log(value2)
}
func TestMerge(t *testing.T) {
type ctxKey1 struct{}
type ctxKey2 struct{}
ctx, cancel := context.WithCancel(context.Background())
cancel()
ctx1 := context.WithValue(context.Background(), ctxKey1{}, "https://github.com/go-kratos/")
ctx2 := context.WithValue(ctx, ctxKey2{}, "https://go-kratos.dev/")
ctx, cancel = Merge(ctx1, ctx2)
defer cancel()
got := ctx.Value(ctxKey1{})
value1, ok := got.(string)
if !ok {
t.Errorf("expect %v, got %v", true, ok)
}
if !reflect.DeepEqual(value1, "https://github.com/go-kratos/") {
t.Errorf("expect %v, got %v", "https://github.com/go-kratos/", value1)
}
got2 := ctx.Value(ctxKey2{})
value2, ok := got2.(string)
if !ok {
t.Errorf("expect %v, got %v", true, ok)
}
if !reflect.DeepEqual(value2, "https://go-kratos.dev/") {
t.Errorf("expect %v, got %v", " https://go-kratos.dev/", value2)
}
t.Log(ctx)
}
func TestErr(t *testing.T) {
ctx1, cancel := context.WithTimeout(context.Background(), time.Microsecond)
defer cancel()
time.Sleep(time.Millisecond)
ctx, cancel := Merge(ctx1, context.Background())
defer cancel()
if !errors.Is(ctx.Err(), context.DeadlineExceeded) {
t.Errorf("expect %v, got %v", context.DeadlineExceeded, ctx.Err())
}
}
func TestDone(t *testing.T) {
ctx1, cancel := context.WithCancel(context.Background())
defer cancel()
ctx, cancel := Merge(ctx1, context.Background())
go func() {
time.Sleep(time.Millisecond * 50)
cancel()
}()
if <-ctx.Done() != struct{}{} {
t.Errorf("expect %v, got %v", struct{}{}, <-ctx.Done())
}
}
func TestFinish(t *testing.T) {
mc := &mergeCtx{
parent1: context.Background(),
parent2: context.Background(),
done: make(chan struct{}),
cancelCh: make(chan struct{}),
}
err := mc.finish(context.DeadlineExceeded)
if !errors.Is(err, context.DeadlineExceeded) {
t.Errorf("expect %v, got %v", context.DeadlineExceeded, err)
}
if !reflect.DeepEqual(mc.doneMark, uint32(1)) {
t.Errorf("expect %v, got %v", 1, mc.doneMark)
}
if <-mc.done != struct{}{} {
t.Errorf("expect %v, got %v", struct{}{}, <-mc.done)
}
}
func TestWait(t *testing.T) {
ctx1, cancel := context.WithCancel(context.Background())
mc := &mergeCtx{
parent1: ctx1,
parent2: context.Background(),
done: make(chan struct{}),
cancelCh: make(chan struct{}),
}
go func() {
time.Sleep(time.Millisecond * 50)
cancel()
}()
mc.wait()
t.Log(mc.doneErr)
if !errors.Is(mc.doneErr, context.Canceled) {
t.Errorf("expect %v, got %v", context.Canceled, mc.doneErr)
}
ctx2, cancel2 := context.WithCancel(context.Background())
mc = &mergeCtx{
parent1: ctx2,
parent2: context.Background(),
done: make(chan struct{}),
cancelCh: make(chan struct{}),
}
go func() {
time.Sleep(time.Millisecond * 50)
cancel2()
}()
mc.wait()
t.Log(mc.doneErr)
if !errors.Is(mc.doneErr, context.Canceled) {
t.Errorf("expect %v, got %v", context.Canceled, mc.doneErr)
}
}
func TestCancel(t *testing.T) {
mc := &mergeCtx{
parent1: context.Background(),
parent2: context.Background(),
done: make(chan struct{}),
cancelCh: make(chan struct{}),
}
mc.cancel()
if <-mc.cancelCh != struct{}{} {
t.Errorf("expect %v, got %v", struct{}{}, <-mc.cancelCh)
}
}
func Test_mergeCtx_Deadline(t *testing.T) {
type fields struct {
parent1Timeout time.Time
parent2Timeout time.Time
}
tests := []struct {
name string
fields fields
want1 bool
}{
{
name: "parent1 not deadline",
fields: fields{time.Time{}, time.Now().Add(time.Second * 100)},
want1: true,
},
{
name: "parent2 not deadline",
fields: fields{time.Now().Add(time.Second * 100), time.Time{}},
want1: true,
},
{
name: " parent1 parent2 not deadline",
fields: fields{time.Time{}, time.Time{}},
want1: false,
},
{
name: " parent1 < parent2",
fields: fields{time.Now().Add(time.Second * 100), time.Now().Add(time.Second * 200)},
want1: true,
},
{
name: " parent1 > parent2",
fields: fields{time.Now().Add(time.Second * 100), time.Now().Add(time.Second * 50)},
want1: true,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
var parent1, parent2 context.Context
var cancel1, cancel2 context.CancelFunc
if reflect.DeepEqual(tt.fields.parent1Timeout, time.Time{}) {
parent1 = context.Background()
} else {
parent1, cancel1 = context.WithDeadline(context.Background(), tt.fields.parent1Timeout)
defer cancel1()
}
if reflect.DeepEqual(tt.fields.parent2Timeout, time.Time{}) {
parent2 = context.Background()
} else {
parent2, cancel2 = context.WithDeadline(context.Background(), tt.fields.parent2Timeout)
defer cancel2()
}
mc := &mergeCtx{
parent1: parent1,
parent2: parent2,
}
got, got1 := mc.Deadline()
t.Log(got)
if got1 != tt.want1 {
t.Errorf("Deadline() got1 = %v, want %v", got1, tt.want1)
}
})
}
}
func Test_Err2(t *testing.T) {
ctx1, cancel := context.WithCancel(context.Background())
defer cancel()
time.Sleep(time.Millisecond)
ctx, cancel := Merge(ctx1, context.Background())
defer cancel()
if ctx.Err() != nil {
t.Errorf("expect %v, got %v", nil, ctx.Err())
}
ctx1, cancel1 := context.WithCancel(context.Background())
time.Sleep(time.Millisecond)
ctx, cancel = Merge(ctx1, context.Background())
defer cancel()
cancel1()
if !errors.Is(ctx.Err(), context.Canceled) {
t.Errorf("expect %v, got %v", context.Canceled, ctx.Err())
}
ctx1, cancel1 = context.WithCancel(context.Background())
time.Sleep(time.Millisecond)
ctx, cancel = Merge(context.Background(), ctx1)
defer cancel()
cancel1()
if !errors.Is(ctx.Err(), context.Canceled) {
t.Errorf("expect %v, got %v", context.Canceled, ctx.Err())
}
ctx, cancel = Merge(context.Background(), context.Background())
cancel()
if !errors.Is(ctx.Err(), context.Canceled) {
t.Errorf("expect %v, got %v", context.Canceled, ctx.Err())
}
}