Files
nats.go/jetstream/jetstream_test.go
Piotr Piotrowski e7ab93ecb8 Add ordered consumer, FetchBytes and Next, rework options
Signed-off-by: Piotr Piotrowski <piotr@synadia.com>
2023-05-23 12:03:02 +02:00

928 lines
23 KiB
Go

// Copyright 2020-2023 The NATS 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 jetstream
import (
"context"
"errors"
"fmt"
"os"
"testing"
"time"
"github.com/nats-io/nats.go"
)
func TestNewWithAPIPrefix(t *testing.T) {
t.Run("import subject from another account", func(t *testing.T) {
conf := createConfFile(t, []byte(`
listen: 127.0.0.1:-1
no_auth_user: test_user
jetstream: {max_mem_store: 64GB, max_file_store: 10TB}
accounts: {
JS: {
jetstream: enabled
users: [ {user: main, password: foo} ]
exports [ { service: "$JS.API.>" }, { service: "foo" }]
},
U: {
users: [ {user: test_user, password: bar} ]
imports [
{ service: { subject: "$JS.API.>", account: JS } , to: "main.>" }
{ service: { subject: "foo", account: JS } }
]
},
}
`))
defer os.Remove(conf)
srv, _ := RunServerWithConfig(conf)
defer shutdownJSServerAndRemoveStorage(t, srv)
ncMain, err := nats.Connect(srv.ClientURL(), nats.UserInfo("main", "foo"))
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
defer ncMain.Close()
ctx, cancel := context.WithTimeout(context.Background(), 20*time.Second)
defer cancel()
jsMain, err := New(ncMain)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
_, err = jsMain.CreateStream(ctx, StreamConfig{
Name: "TEST",
Subjects: []string{"foo"},
})
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
ncTest, err := nats.Connect(srv.ClientURL())
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
defer ncTest.Close()
jsTest, err := NewWithAPIPrefix(ncTest, "main")
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
_, err = jsTest.Publish(ctx, "foo", []byte("msg"))
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
})
t.Run("empty API prefix", func(t *testing.T) {
srv := RunBasicJetStreamServer()
defer shutdownJSServerAndRemoveStorage(t, srv)
nc, err := nats.Connect(srv.ClientURL())
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
_, err = NewWithAPIPrefix(nc, "")
if err == nil || err.Error() != "API prefix cannot be empty" {
t.Fatalf(`Expected error: "API prefix cannot be empty"; got: %v`, err)
}
})
}
func TestNewWithDomain(t *testing.T) {
t.Run("jetstream account with domain", func(t *testing.T) {
conf := createConfFile(t, []byte(`
listen: 127.0.0.1:-1
jetstream: { domain: ABC }
`))
defer os.Remove(conf)
srv, _ := RunServerWithConfig(conf)
defer shutdownJSServerAndRemoveStorage(t, srv)
nc, err := nats.Connect(srv.ClientURL(), nats.UserInfo("main", "foo"))
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
defer nc.Close()
ctx, cancel := context.WithTimeout(context.Background(), 20*time.Second)
defer cancel()
js, err := NewWithDomain(nc, "ABC")
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
accInfo, err := js.AccountInfo(ctx)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
if accInfo.Domain != "ABC" {
t.Errorf("Invalid domain; want %v, got: %v", "ABC", accInfo.Domain)
}
_, err = js.CreateStream(ctx, StreamConfig{
Name: "TEST",
Subjects: []string{"foo"},
})
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
_, err = js.Publish(ctx, "foo", []byte("msg"))
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
})
t.Run("empty domain", func(t *testing.T) {
srv := RunBasicJetStreamServer()
defer shutdownJSServerAndRemoveStorage(t, srv)
nc, err := nats.Connect(srv.ClientURL())
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
_, err = NewWithDomain(nc, "")
if err == nil || err.Error() != "domain cannot be empty" {
t.Fatalf(`Expected error: "domain cannot be empty"; got: %v`, err)
}
})
}
func TestWithClientTrace(t *testing.T) {
srv := RunBasicJetStreamServer()
defer shutdownJSServerAndRemoveStorage(t, srv)
nc, err := nats.Connect(srv.ClientURL())
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
ctx, cancel := context.WithTimeout(context.Background(), 20*time.Second)
defer cancel()
var sent, received string
js, err := New(nc, WithClientTrace(&ClientTrace{
RequestSent: func(subj string, _ []byte) {
sent = fmt.Sprintf("Request sent: %s", subj)
},
ResponseReceived: func(subj string, _ []byte, _ nats.Header) {
received = fmt.Sprintf("Response received: %s", subj)
},
}))
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
_, err = js.CreateStream(ctx, StreamConfig{Name: "foo", Subjects: []string{"FOO.123"}})
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
if sent != "Request sent: $JS.API.STREAM.CREATE.foo" {
t.Fatalf(`Invalid value on sent request trace; want: "Request sent: $JS.API.STREAM.CREATE.foo"; got: %s`, sent)
}
if received != "Response received: $JS.API.STREAM.CREATE.foo" {
t.Fatalf(`Invalid value on response receive trace; want: "Response received: $JS.API.STREAM.CREATE.foo"; got: %s`, sent)
}
defer nc.Close()
}
func TestCreateStream(t *testing.T) {
tests := []struct {
name string
stream string
subject string
withError error
}{
{
name: "create stream, ok",
stream: "foo",
subject: "FOO.123",
},
{
name: "invalid stream name",
stream: "foo.123",
subject: "FOO.123",
withError: ErrInvalidStreamName,
},
{
name: "stream name required",
stream: "",
subject: "FOO.123",
withError: ErrStreamNameRequired,
},
{
name: "stream name already in use",
stream: "foo",
subject: "BAR.123",
withError: ErrStreamNameAlreadyInUse,
},
}
srv := RunBasicJetStreamServer()
defer shutdownJSServerAndRemoveStorage(t, srv)
nc, err := nats.Connect(srv.ClientURL())
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
ctx, cancel := context.WithTimeout(context.Background(), 20*time.Second)
defer cancel()
js, err := New(nc)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
defer nc.Close()
for _, test := range tests {
t.Run(test.name, func(t *testing.T) {
_, err = js.CreateStream(ctx, StreamConfig{Name: test.stream, Subjects: []string{test.subject}})
if test.withError != nil {
if !errors.Is(err, test.withError) {
t.Fatalf("Expected error: %v; got: %v", test.withError, err)
}
return
}
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
})
}
}
func TestUpdateStream(t *testing.T) {
tests := []struct {
name string
stream string
subject string
withError error
}{
{
name: "update existing stream",
stream: "foo",
subject: "BAR.123",
},
{
name: "invalid stream name",
stream: "foo.123",
subject: "FOO.123",
withError: ErrInvalidStreamName,
},
{
name: "stream name required",
stream: "",
subject: "FOO.123",
withError: ErrStreamNameRequired,
},
{
name: "stream not found",
stream: "bar",
subject: "FOO.123",
withError: ErrStreamNotFound,
},
}
srv := RunBasicJetStreamServer()
defer shutdownJSServerAndRemoveStorage(t, srv)
nc, err := nats.Connect(srv.ClientURL())
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
ctx, cancel := context.WithTimeout(context.Background(), 20*time.Second)
defer cancel()
js, err := New(nc)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
defer nc.Close()
_, err = js.CreateStream(ctx, StreamConfig{Name: "foo", Subjects: []string{"FOO.123"}})
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
for _, test := range tests {
t.Run(test.name, func(t *testing.T) {
s, err := js.UpdateStream(ctx, StreamConfig{Name: test.stream, Subjects: []string{test.subject}})
if test.withError != nil {
if !errors.Is(err, test.withError) {
t.Fatalf("Expected error: %v; got: %v", test.withError, err)
}
return
}
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
info, err := s.Info(ctx)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
if len(info.Config.Subjects) != 1 || info.Config.Subjects[0] != test.subject {
t.Fatalf("Invalid stream subjects after update: %v", info.Config.Subjects)
}
})
}
}
func TestStream(t *testing.T) {
tests := []struct {
name string
stream string
subject string
withError error
}{
{
name: "get existing stream",
stream: "foo",
},
{
name: "invalid stream name",
stream: "foo.123",
withError: ErrInvalidStreamName,
},
{
name: "stream name required",
stream: "",
withError: ErrStreamNameRequired,
},
{
name: "stream not found",
stream: "bar",
withError: ErrStreamNotFound,
},
}
srv := RunBasicJetStreamServer()
defer shutdownJSServerAndRemoveStorage(t, srv)
nc, err := nats.Connect(srv.ClientURL())
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
ctx, cancel := context.WithTimeout(context.Background(), 20*time.Second)
defer cancel()
js, err := New(nc)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
defer nc.Close()
_, err = js.CreateStream(ctx, StreamConfig{Name: "foo", Subjects: []string{"FOO.123"}})
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
for _, test := range tests {
t.Run(test.name, func(t *testing.T) {
s, err := js.Stream(ctx, test.stream)
if test.withError != nil {
if !errors.Is(err, test.withError) {
t.Fatalf("Expected error: %v; got: %v", test.withError, err)
}
return
}
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
if s.CachedInfo().Config.Name != test.stream {
t.Fatalf("Invalid stream fetched; want: foo; got: %s", s.CachedInfo().Config.Name)
}
})
}
}
func TestDeleteStream(t *testing.T) {
tests := []struct {
name string
stream string
subject string
withError error
}{
{
name: "delete existing stream",
stream: "foo",
},
{
name: "invalid stream name",
stream: "foo.123",
withError: ErrInvalidStreamName,
},
{
name: "stream name required",
stream: "",
withError: ErrStreamNameRequired,
},
{
name: "stream not found",
stream: "foo",
withError: ErrStreamNotFound,
},
}
srv := RunBasicJetStreamServer()
defer shutdownJSServerAndRemoveStorage(t, srv)
nc, err := nats.Connect(srv.ClientURL())
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
ctx, cancel := context.WithTimeout(context.Background(), 20*time.Second)
defer cancel()
js, err := New(nc)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
defer nc.Close()
_, err = js.CreateStream(ctx, StreamConfig{Name: "foo", Subjects: []string{"FOO.123"}})
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
for _, test := range tests {
t.Run(test.name, func(t *testing.T) {
err := js.DeleteStream(ctx, test.stream)
if test.withError != nil {
if !errors.Is(err, test.withError) {
t.Fatalf("Expected error: %v; got: %v", test.withError, err)
}
return
}
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
})
}
}
func TestAccountInfo(t *testing.T) {
t.Run("fetch account info", func(t *testing.T) {
srv := RunBasicJetStreamServer()
defer shutdownJSServerAndRemoveStorage(t, srv)
nc, err := nats.Connect(srv.ClientURL())
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
ctx, cancel := context.WithTimeout(context.Background(), 20*time.Second)
defer cancel()
js, err := New(nc)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
defer nc.Close()
_, err = js.CreateStream(ctx, StreamConfig{Name: "foo", Subjects: []string{"FOO.123"}})
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
info, err := js.AccountInfo(ctx)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
if info.Streams != 1 {
t.Fatalf("Invalid number of streams; want: 1; got: %d", info.Streams)
}
})
t.Run("jetstream not enabled on server", func(t *testing.T) {
srv := RunDefaultServer()
defer shutdownJSServerAndRemoveStorage(t, srv)
nc, err := nats.Connect(srv.ClientURL())
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
ctx, cancel := context.WithTimeout(context.Background(), 20*time.Second)
defer cancel()
js, err := New(nc)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
defer nc.Close()
_, err = js.AccountInfo(ctx)
if err == nil || !errors.Is(err, ErrJetStreamNotEnabled) {
t.Fatalf(": %v; got: %v", ErrJetStreamNotEnabled, err)
}
})
t.Run("jetstream not enabled for account", func(t *testing.T) {
conf := createConfFile(t, []byte(`
listen: 127.0.0.1:-1
jetstream: enabled
no_auth_user: foo
accounts: {
JS: {
jetstream: disabled
users: [ {user: foo, password: bar} ]
},
}
`))
defer os.Remove(conf)
srv, _ := RunServerWithConfig(conf)
defer shutdownJSServerAndRemoveStorage(t, srv)
nc, err := nats.Connect(srv.ClientURL())
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
ctx, cancel := context.WithTimeout(context.Background(), 20*time.Second)
defer cancel()
js, err := New(nc)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
defer nc.Close()
_, err = js.AccountInfo(ctx)
if err == nil || !errors.Is(err, ErrJetStreamNotEnabledForAccount) {
t.Fatalf(": %v; got: %v", ErrJetStreamNotEnabledForAccount, err)
}
})
}
func TestListStreams(t *testing.T) {
tests := []struct {
name string
streamsNum int
withError error
}{
{
name: "list streams",
streamsNum: 260,
},
{
name: "no stream available",
streamsNum: 0,
},
}
for _, test := range tests {
t.Run(test.name, func(t *testing.T) {
srv := RunBasicJetStreamServer()
defer shutdownJSServerAndRemoveStorage(t, srv)
nc, err := nats.Connect(srv.ClientURL())
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer cancel()
js, err := New(nc)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
defer nc.Close()
for i := 0; i < test.streamsNum; i++ {
_, err = js.CreateStream(ctx, StreamConfig{Name: fmt.Sprintf("foo%d", i), Subjects: []string{fmt.Sprintf("FOO.%d", i)}})
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
}
streamsList := js.ListStreams(ctx)
streams := make([]*StreamInfo, 0)
Loop:
for {
select {
case s := <-streamsList.Info():
streams = append(streams, s)
case err := <-streamsList.Err():
if !errors.Is(err, ErrEndOfData) {
t.Fatalf("Unexpected error: %v", err)
}
break Loop
}
}
if len(streams) != test.streamsNum {
t.Fatalf("Wrong number of streams; want: %d; got: %d", test.streamsNum, len(streams))
}
})
}
}
func TestStreamNames(t *testing.T) {
tests := []struct {
name string
streamsNum int
withError error
}{
{
name: "list streams",
streamsNum: 500,
},
{
name: "no stream available",
streamsNum: 0,
},
}
for _, test := range tests {
t.Run(test.name, func(t *testing.T) {
srv := RunBasicJetStreamServer()
defer shutdownJSServerAndRemoveStorage(t, srv)
nc, err := nats.Connect(srv.ClientURL())
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer cancel()
js, err := New(nc)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
defer nc.Close()
for i := 0; i < test.streamsNum; i++ {
_, err = js.CreateStream(ctx, StreamConfig{Name: fmt.Sprintf("foo%d", i), Subjects: []string{fmt.Sprintf("FOO.%d", i)}})
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
}
streamsList := js.StreamNames(ctx)
streams := make([]string, 0)
Loop:
for {
select {
case s := <-streamsList.Name():
streams = append(streams, s)
case err := <-streamsList.Err():
if !errors.Is(err, ErrEndOfData) {
t.Fatalf("Unexpected error: %v", err)
}
break Loop
}
}
if len(streams) != test.streamsNum {
t.Fatalf("Wrong number of streams; want: %d; got: %d", test.streamsNum, len(streams))
}
})
}
}
func TestJetStream_AddConsumer(t *testing.T) {
tests := []struct {
name string
stream string
consumerConfig ConsumerConfig
shouldCreate bool
withError error
}{
{
name: "create durable pull consumer",
stream: "foo",
consumerConfig: ConsumerConfig{Durable: "dur", AckPolicy: AckExplicitPolicy},
shouldCreate: true,
},
{
name: "create ephemeral pull consumer",
stream: "foo",
consumerConfig: ConsumerConfig{AckPolicy: AckExplicitPolicy},
shouldCreate: true,
},
{
name: "consumer already exists, update",
stream: "foo",
consumerConfig: ConsumerConfig{Durable: "dur", AckPolicy: AckExplicitPolicy, Description: "test consumer"},
},
{
name: "consumer already exists, illegal update",
stream: "foo",
consumerConfig: ConsumerConfig{Durable: "dur", AckPolicy: AckNonePolicy, Description: "test consumer"},
withError: ErrConsumerCreate,
},
{
name: "stream does not exist",
stream: "abc",
withError: ErrStreamNotFound,
},
{
name: "invalid stream name",
stream: "foo.1",
withError: ErrInvalidStreamName,
},
{
name: "invalid durable name",
stream: "foo",
consumerConfig: ConsumerConfig{Durable: "dur.123", AckPolicy: AckExplicitPolicy},
withError: ErrInvalidConsumerName,
},
}
srv := RunBasicJetStreamServer()
defer shutdownJSServerAndRemoveStorage(t, srv)
nc, err := nats.Connect(srv.ClientURL())
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
js, err := New(nc)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
defer nc.Close()
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer cancel()
_, err = js.CreateStream(ctx, StreamConfig{Name: "foo", Subjects: []string{"FOO.*"}})
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
for _, test := range tests {
t.Run(test.name, func(t *testing.T) {
var sub *nats.Subscription
if test.consumerConfig.FilterSubject != "" {
sub, err = nc.SubscribeSync(fmt.Sprintf("$JS.API.CONSUMER.CREATE.foo.*.%s", test.consumerConfig.FilterSubject))
} else {
sub, err = nc.SubscribeSync("$JS.API.CONSUMER.CREATE.foo.*")
}
c, err := js.AddConsumer(ctx, test.stream, test.consumerConfig)
if test.withError != nil {
if err == nil || !errors.Is(err, test.withError) {
t.Fatalf("Expected error: %v; got: %v", test.withError, err)
}
return
}
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
if test.shouldCreate {
if _, err := sub.NextMsgWithContext(ctx); err != nil {
t.Fatalf("Expected request on %s; got %s", sub.Subject, err)
}
}
_, err = js.Consumer(ctx, test.stream, c.CachedInfo().Name)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
})
}
}
func TestJetStream_Consumer(t *testing.T) {
tests := []struct {
name string
stream string
durable string
withError error
}{
{
name: "get existing consumer",
stream: "foo",
durable: "dur",
},
{
name: "consumer does not exist",
stream: "foo",
durable: "abc",
withError: ErrConsumerNotFound,
},
{
name: "invalid durable name",
stream: "foo",
durable: "dur.123",
withError: ErrInvalidConsumerName,
},
{
name: "stream does not exist",
stream: "abc",
durable: "dur",
withError: ErrStreamNotFound,
},
{
name: "invalid stream name",
stream: "foo.1",
durable: "dur",
withError: ErrInvalidStreamName,
},
}
srv := RunBasicJetStreamServer()
defer shutdownJSServerAndRemoveStorage(t, srv)
nc, err := nats.Connect(srv.ClientURL())
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
js, err := New(nc)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
defer nc.Close()
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer cancel()
s, err := js.CreateStream(ctx, StreamConfig{Name: "foo", Subjects: []string{"FOO.*"}})
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
_, err = s.AddConsumer(ctx, ConsumerConfig{Durable: "dur", AckPolicy: AckAllPolicy, Description: "desc"})
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
for _, test := range tests {
t.Run(test.name, func(t *testing.T) {
c, err := js.Consumer(ctx, test.stream, test.durable)
if test.withError != nil {
if err == nil || !errors.Is(err, test.withError) {
t.Fatalf("Expected error: %v; got: %v", test.withError, err)
}
return
}
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
if c.CachedInfo().Name != test.durable {
t.Fatalf("Unexpected consumer fetched; want: %s; got: %s", test.durable, c.CachedInfo().Name)
}
})
}
}
func TestJetStream_DeleteConsumer(t *testing.T) {
tests := []struct {
name string
stream string
durable string
withError error
}{
{
name: "delete existing consumer",
stream: "foo",
durable: "dur",
},
{
name: "consumer does not exist",
stream: "foo",
durable: "dur",
withError: ErrConsumerNotFound,
},
{
name: "invalid durable name",
stream: "foo",
durable: "dur.123",
withError: ErrInvalidConsumerName,
},
{
name: "stream not found",
stream: "abc",
durable: "dur",
withError: ErrStreamNotFound,
},
{
name: "invalid stream name",
stream: "foo.1",
durable: "dur",
withError: ErrInvalidStreamName,
},
}
srv := RunBasicJetStreamServer()
defer shutdownJSServerAndRemoveStorage(t, srv)
nc, err := nats.Connect(srv.ClientURL())
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
js, err := New(nc)
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
defer nc.Close()
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer cancel()
s, err := js.CreateStream(ctx, StreamConfig{Name: "foo", Subjects: []string{"FOO.*"}})
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
_, err = s.AddConsumer(ctx, ConsumerConfig{Durable: "dur", AckPolicy: AckAllPolicy, Description: "desc"})
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
for _, test := range tests {
t.Run(test.name, func(t *testing.T) {
err := js.DeleteConsumer(ctx, test.stream, test.durable)
if test.withError != nil {
if err == nil || !errors.Is(err, test.withError) {
t.Fatalf("Expected error: %v; got: %v", test.withError, err)
}
return
}
if err != nil {
t.Fatalf("Unexpected error: %v", err)
}
_, err = s.Consumer(ctx, test.durable)
if err == nil || !errors.Is(err, ErrConsumerNotFound) {
t.Fatalf("Expected error: %v; got: %v", ErrConsumerNotFound, err)
}
})
}
}