-
Notifications
You must be signed in to change notification settings - Fork 21
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Outstanding Todos: - Include smart producer public api calls when producer is merged - Ensure the number of messages consumed is the same as the ones produced
- Loading branch information
Showing
2 changed files
with
164 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,3 +1,128 @@ | ||
//go:build rabbitmq.stream.system_test | ||
|
||
package stream_test | ||
|
||
import ( | ||
"fmt" | ||
. "github.com/onsi/ginkgo/v2" | ||
. "github.com/onsi/gomega" | ||
"github.com/onsi/gomega/gmeasure" | ||
"github.com/rabbitmq/rabbitmq-stream-go-client/v2/pkg/codecs/amqp" | ||
"github.com/rabbitmq/rabbitmq-stream-go-client/v2/pkg/common" | ||
"github.com/rabbitmq/rabbitmq-stream-go-client/v2/pkg/raw" | ||
"github.com/rabbitmq/rabbitmq-stream-go-client/v2/pkg/stream" | ||
"golang.org/x/exp/slog" | ||
"sync" | ||
"time" | ||
) | ||
|
||
var _ = Describe("stream package", func() { | ||
const ( | ||
streamName = "stream-system-test" | ||
// 100 byte message | ||
messageBody = "Rabbitmq-is-awesomeRabbitmq-is-awesomeRabbitmq-is-awesomeRabbitmq-is-awesomeRabbitmq-is-awesome!!!!!" | ||
defaultRabbitmqUri = "rabbitmq-stream://guest:guest@localhost/%2F" | ||
) | ||
It("can create and connect to a stream, publish and receive messages", func(ctx SpecContext) { | ||
h := slog.HandlerOptions{Level: slog.LevelDebug}.NewTextHandler(GinkgoWriter) | ||
debugLogger := slog.New(h) | ||
itCtx := raw.NewContextWithLogger(ctx, *debugLogger) | ||
|
||
By("creating a new environment") | ||
envConfig := stream.NewEnvironmentConfiguration( | ||
stream.WithUri(defaultRabbitmqUri)) | ||
env, err := stream.NewEnvironment(itCtx, envConfig) | ||
Expect(err).NotTo(HaveOccurred()) | ||
|
||
By("creating a stream") | ||
csOpts := stream.CreateStreamOptions{time.Second * 60, stream.Kilobyte, stream.Megabyte} | ||
Expect(env.CreateStream(itCtx, streamName, csOpts)).To(Succeed()) | ||
|
||
// See https://github.com/rabbitmq/rabbitmq-stream-go-client/pull/255 | ||
// TODO Change this By when stream pkg producer is megered | ||
By("creating a raw client producer") | ||
throughputExp := gmeasure.NewExperiment("100-byte message throughput") | ||
AddReportEntry(throughputExp.Name, throughputExp) | ||
stopWatch := throughputExp.NewStopwatch() | ||
|
||
streamClientConfiguration, err := raw.NewClientConfiguration(defaultRabbitmqUri) | ||
Expect(err).ToNot(HaveOccurred()) | ||
streamClient, err := raw.DialConfig(itCtx, streamClientConfiguration) | ||
Expect(err).ToNot(HaveOccurred()) | ||
Expect(streamClient.IsOpen()).To(BeTrue(), "expected stream client to be open") | ||
|
||
const publisherId = 1 | ||
Expect( | ||
streamClient.DeclarePublisher(itCtx, publisherId, "e2e-publisher", streamName), | ||
).To(Succeed()) | ||
|
||
// See https://github.com/rabbitmq/rabbitmq-stream-go-client/pull/255 | ||
// TODO Change this By when stream pkg producer is megered | ||
By("receiving confirmations") | ||
c := streamClient.NotifyPublish(make(chan *raw.PublishConfirm, 100)) | ||
var wg sync.WaitGroup | ||
wg.Add(1) | ||
|
||
const numMessages = 100_000 | ||
go func() { | ||
defer GinkgoRecover() | ||
defer wg.Done() | ||
confirmStopwatch := throughputExp.NewStopwatch() | ||
defer confirmStopwatch.Record("publish confirmation") | ||
|
||
confirmedIds := make(map[uint64]struct{}, numMessages) | ||
for { | ||
select { | ||
case <-ctx.Done(): | ||
Fail(fmt.Sprintf("context timed out: expected to receive 100_000 confirmations: received %d", len(confirmedIds))) | ||
case confirm, ok := <-c: | ||
if !ok { | ||
return | ||
} | ||
ids := confirm.PublishingIds() | ||
for i := 0; i < len(ids); i++ { | ||
confirmedIds[ids[i]] = struct{}{} | ||
} | ||
|
||
if len(confirmedIds) == numMessages { | ||
return | ||
} | ||
} | ||
} | ||
}() | ||
|
||
// See https://github.com/rabbitmq/rabbitmq-stream-go-client/pull/255 | ||
// TODO Change this By when stream pkg producer is megered | ||
By("sending messages") | ||
for i := uint64(0); i < numMessages; i++ { | ||
messageContainer := raw.NewPublishingMessage(i, &plainTextMessage{messageBody}) | ||
Expect( | ||
streamClient.Send(itCtx, publisherId, wrap[common.PublishingMessager](messageContainer)), | ||
).To(Succeed()) | ||
} | ||
stopWatch.Record("Send").Reset() | ||
|
||
wg.Wait() | ||
|
||
By("creating a consumer") | ||
handleMessages := func(consumerContext stream.ConsumerContext, message *amqp.Message) { | ||
fmt.Printf("messages: %s\n", message.Data) | ||
} | ||
opts := &stream.ConsumerOptions{} | ||
mutex := &sync.Mutex{} | ||
consumer, err := stream.NewConsumer(streamName, streamClient, handleMessages, opts, mutex) | ||
Expect(err).NotTo(HaveOccurred()) | ||
err = consumer.Subscribe(itCtx) | ||
Expect(err).NotTo(HaveOccurred()) | ||
// TODO ensure all messages are consumed | ||
|
||
By("deleting the stream") | ||
Expect(env.DeleteStream(itCtx, streamName)).To(Succeed()) | ||
}) | ||
}) | ||
|
||
func wrap[T any](v T) []T { | ||
r := make([]T, 1) | ||
r[0] = v | ||
return r | ||
} |