代码拉取完成,页面将自动刷新
package queuetest
import (
"sync"
"testing"
"github.com/elastic/beats/libbeat/common"
"github.com/elastic/beats/libbeat/publisher/queue"
)
type QueueFactory func() queue.Queue
type workerFactory func(*sync.WaitGroup, interface{}, *TestLogger, queue.Queue) func()
func TestSingleProducerConsumer(
t *testing.T,
events, batchSize int,
factory QueueFactory,
) {
tests := []struct {
name string
producers, consumers workerFactory
}{
{
"single producer, consumer without ack, complete batches",
makeProducer(events, false, countEvent),
makeConsumer(events, -1),
},
{
"single producer, consumer, without ack, limited batches",
makeProducer(events, false, countEvent),
makeConsumer(events, batchSize),
},
{
"single producer, consumer, with ack, complete batches",
makeProducer(events, true, countEvent),
makeConsumer(events, -1),
},
{
"single producer, consumer, with ack, limited batches",
makeProducer(events, true, countEvent),
makeConsumer(events, batchSize),
},
}
for _, test := range tests {
t.Run(test.name, withLogOutput(func(t *testing.T) {
log := NewTestLogger(t)
log.Debug("run test: ", test.name)
queue := factory()
defer func() {
err := queue.Close()
if err != nil {
t.Error(err)
}
}()
var wg sync.WaitGroup
go test.producers(&wg, nil, log, queue)()
go test.consumers(&wg, nil, log, queue)()
wg.Wait()
}))
}
}
func TestMultiProducerConsumer(
t *testing.T,
events, batchSize int,
factory QueueFactory,
) {
tests := []struct {
name string
producers, consumers workerFactory
}{
{
"2 producers, 1 consumer, without ack, complete batches",
multiple(
makeProducer(events, false, countEvent),
makeProducer(events, false, countEvent),
),
makeConsumer(events*2, -1),
},
{
"2 producers, 1 consumer, all ack, complete batches",
multiple(
makeProducer(events, true, countEvent),
makeProducer(events, true, countEvent),
),
makeConsumer(events*2, -1),
},
{
"2 producers, 1 consumer, 1 ack, complete batches",
multiple(
makeProducer(events, true, countEvent),
makeProducer(events, false, countEvent),
),
makeConsumer(events*2, -1),
},
{
"2 producers, 1 consumer, without ack, limited batches",
multiple(
makeProducer(events, false, countEvent),
makeProducer(events, false, countEvent),
),
makeConsumer(events*2, batchSize),
},
{
"2 producers, 1 consumer, all ack, limited batches",
multiple(
makeProducer(events, true, countEvent),
makeProducer(events, true, countEvent),
),
makeConsumer(events*2, batchSize),
},
{
"2 producers, 1 consumer, 1 ack, limited batches",
multiple(
makeProducer(events, true, countEvent),
makeProducer(events, false, countEvent),
),
makeConsumer(events*2, batchSize),
},
{
"1 producer, 2 consumers, without ack, complete batches",
makeProducer(events, true, countEvent),
multiConsumer(2, events, -1),
},
{
"1 producer, 2 consumers, without ack, limited batches",
makeProducer(events, true, countEvent),
multiConsumer(2, events, -1),
},
{
"2 producers, 2 consumer, without ack, complete batches",
multiple(
makeProducer(events, false, countEvent),
makeProducer(events, false, countEvent),
),
multiConsumer(2, events*2, -1),
},
{
"2 producers, 2 consumer, all ack, complete batches",
multiple(
makeProducer(events, true, countEvent),
makeProducer(events, true, countEvent),
),
multiConsumer(2, events*2, -1),
},
{
"2 producers, 2 consumer, 1 ack, complete batches",
multiple(
makeProducer(events, true, countEvent),
makeProducer(events, false, countEvent),
),
multiConsumer(2, events*2, -1),
},
{
"2 producers, 2 consumer, without ack, limited batches",
multiple(
makeProducer(events, false, countEvent),
makeProducer(events, false, countEvent),
),
multiConsumer(2, events*2, batchSize),
},
{
"2 producers, 2 consumer, all ack, limited batches",
multiple(
makeProducer(events, true, countEvent),
makeProducer(events, true, countEvent),
),
multiConsumer(2, events*2, batchSize),
},
{
"2 producers, 2 consumer, 1 ack, limited batches",
multiple(
makeProducer(events, true, countEvent),
makeProducer(events, false, countEvent),
),
multiConsumer(2, events*2, batchSize),
},
}
for _, test := range tests {
t.Run(test.name, withLogOutput(func(t *testing.T) {
log := NewTestLogger(t)
log.Debug("run test: ", test.name)
queue := factory()
defer func() {
err := queue.Close()
if err != nil {
t.Error(err)
}
}()
var wg sync.WaitGroup
go test.producers(&wg, nil, log, queue)()
go test.consumers(&wg, nil, log, queue)()
wg.Wait()
}))
}
}
func multiple(
fns ...workerFactory,
) workerFactory {
return func(wg *sync.WaitGroup, info interface{}, log *TestLogger, queue queue.Queue) func() {
runners := make([]func(), len(fns))
for i, gen := range fns {
runners[i] = gen(wg, info, log, queue)
}
return func() {
for _, r := range runners {
go r()
}
}
}
}
func makeProducer(
maxEvents int,
waitACK bool,
makeFields func(int) common.MapStr,
) func(*sync.WaitGroup, interface{}, *TestLogger, queue.Queue) func() {
return func(wg *sync.WaitGroup, info interface{}, log *TestLogger, b queue.Queue) func() {
wg.Add(1)
return func() {
defer wg.Done()
log.Debug("start producer")
defer log.Debug("stop producer")
var (
ackWG sync.WaitGroup
ackCB func(int)
)
if waitACK {
ackWG.Add(maxEvents)
total := 0
ackCB = func(N int) {
total += N
log.Debugf("producer ACK: N=%v, total=%v\n", N, total)
for i := 0; i < N; i++ {
ackWG.Done()
}
}
}
producer := b.Producer(queue.ProducerConfig{
ACK: ackCB,
})
for i := 0; i < maxEvents; i++ {
producer.Publish(makeEvent(makeFields(i)))
}
ackWG.Wait()
}
}
}
func makeConsumer(maxEvents, batchSize int) workerFactory {
return multiConsumer(1, maxEvents, batchSize)
}
func multiConsumer(numConsumers, maxEvents, batchSize int) workerFactory {
return func(wg *sync.WaitGroup, info interface{}, log *TestLogger, b queue.Queue) func() {
wg.Add(1)
return func() {
defer wg.Done()
var events sync.WaitGroup
consumers := make([]queue.Consumer, numConsumers)
for i := range consumers {
consumers[i] = b.Consumer()
}
events.Add(maxEvents)
for _, c := range consumers {
c := c
wg.Add(1)
go func() {
defer wg.Done()
for {
batch, err := c.Get(batchSize)
if err != nil {
return
}
for range batch.Events() {
events.Done()
}
batch.ACK()
}
}()
}
events.Wait()
// disconnect consumers
for _, c := range consumers {
c.Close()
}
}
}
}
func countEvent(i int) common.MapStr {
return common.MapStr{
"count": i,
}
}
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。