@@ -2,14 +2,20 @@ package kafka
2
2
3
3
import (
4
4
"context"
5
+ "errors"
6
+ "fmt"
7
+ "math/rand"
8
+ "strings"
9
+ "sync"
5
10
6
11
"github.com/IBM/sarama"
7
12
"github.com/openimsdk/tools/log"
13
+ "github.com/openimsdk/tools/mcontext"
8
14
"github.com/openimsdk/tools/mq"
9
15
)
10
16
11
17
func NewMConsumerGroupV2 (ctx context.Context , conf * Config , groupID string , topics []string , autoCommitEnable bool ) (mq.Consumer , error ) {
12
- config , err := BuildProducerConfig ( * conf )
18
+ config , err := BuildConsumerGroupConfig ( conf , sarama . OffsetNewest , autoCommitEnable )
13
19
if err != nil {
14
20
return nil , err
15
21
}
@@ -21,27 +27,26 @@ func NewMConsumerGroupV2(ctx context.Context, conf *Config, groupID string, topi
21
27
topics : topics ,
22
28
groupID : groupID ,
23
29
consumer : group ,
24
- session : make (chan * sessionClaim , 1 ),
25
- idle : make (chan struct {}, 1 ),
30
+ msg : make (chan * consumerMessage , 64 ),
26
31
}
27
- mcg .idle <- struct {}{}
32
+ mcg .ctx , mcg . cancel = context . WithCancel ( ctx )
28
33
mcg .loopConsume ()
29
34
return mcg , nil
30
35
}
31
36
32
- type sessionClaim struct {
33
- session sarama.ConsumerGroupSession
34
- claim sarama.ConsumerGroupClaim
37
+ type consumerMessage struct {
38
+ Msg * sarama.ConsumerMessage
39
+ Session sarama.ConsumerGroupSession
35
40
}
36
41
37
42
type mqConsumerGroup struct {
38
43
topics []string
39
44
groupID string
40
45
consumer sarama.ConsumerGroup
46
+ ctx context.Context
41
47
cancel context.CancelFunc
42
- idle chan struct {}
43
- session chan * sessionClaim
44
- curr * sessionClaim
48
+ msg chan * consumerMessage
49
+ lock sync.Mutex
45
50
}
46
51
47
52
func (* mqConsumerGroup ) Setup (sarama.ConsumerGroupSession ) error { return nil }
@@ -50,65 +55,66 @@ func (*mqConsumerGroup) Cleanup(sarama.ConsumerGroupSession) error {
50
55
return nil
51
56
}
52
57
58
+ func (x * mqConsumerGroup ) closeMsgChan () {
59
+ select {
60
+ case <- x .ctx .Done ():
61
+ if x .lock .TryLock () {
62
+ close (x .msg )
63
+ x .lock .Unlock ()
64
+ }
65
+ default :
66
+ }
67
+ }
68
+
53
69
func (x * mqConsumerGroup ) loopConsume () {
54
- var ctx context.Context
55
- ctx , x .cancel = context .WithCancel (context .Background ())
56
70
go func () {
57
- defer func () {
58
- close (x .session )
59
- }()
71
+ defer x .closeMsgChan ()
72
+ ctx := mcontext .SetOperationID (x .ctx , fmt .Sprintf ("consumer_group_%s_%s_%d" , strings .Join (x .topics , "_" ), x .groupID , rand .Uint32 ()))
60
73
for {
61
- select {
62
- case <- ctx .Done ():
63
- return
64
- case x .idle <- struct {}{}:
65
- }
66
- if err := x .consumer .Consume (ctx , x .topics , x ); err != nil {
74
+ if err := x .consumer .Consume (x .ctx , x .topics , x ); err != nil {
75
+ switch {
76
+ case errors .Is (err , context .Canceled ):
77
+ return
78
+ case errors .Is (err , sarama .ErrClosedConsumerGroup ):
79
+ return
80
+ }
67
81
log .ZWarn (ctx , "consume err" , err , "topic" , x .topics , "groupID" , x .groupID )
68
82
}
69
83
}
70
84
}()
71
85
}
72
86
73
87
func (x * mqConsumerGroup ) ConsumeClaim (session sarama.ConsumerGroupSession , claim sarama.ConsumerGroupClaim ) error {
74
- x .session <- & sessionClaim {session , claim }
75
- return nil
76
- }
77
-
78
- func (x * mqConsumerGroup ) Subscribe (ctx context.Context , fn mq.Handler ) error {
88
+ defer x .closeMsgChan ()
89
+ msg := claim .Messages ()
79
90
for {
80
- curr := x .curr
81
- if curr == nil {
82
- select {
83
- case <- ctx .Done ():
84
- return ctx .Err ()
85
- case val , ok := <- x .session :
86
- if ! ok {
87
- return sarama .ErrClosedConsumerGroup
88
- }
89
- curr = val
90
- x .curr = val
91
- }
92
- }
93
91
select {
94
- case <- ctx .Done ():
95
- return ctx . Err ()
96
- case val , ok := <- curr . claim . Messages () :
92
+ case <- x . ctx .Done ():
93
+ return context . Canceled
94
+ case val , ok := <- msg :
97
95
if ! ok {
98
- x .curr = nil
99
- select {
100
- case <- x .idle :
101
- default :
102
- }
103
- continue
96
+ return nil
104
97
}
105
- ctx := GetContextWithMQHeader (val .Headers )
106
- if err := fn (ctx , string (val .Key ), val .Value ); err != nil {
107
- return err
108
- }
109
- curr .session .MarkMessage (val , "" )
110
- curr .session .Commit ()
98
+ x .msg <- & consumerMessage {Msg : val , Session : session }
99
+ }
100
+ }
101
+ }
102
+
103
+ func (x * mqConsumerGroup ) Subscribe (ctx context.Context , fn mq.Handler ) error {
104
+ select {
105
+ case <- ctx .Done ():
106
+ return context .Cause (ctx )
107
+ case msg , ok := <- x .msg :
108
+ if ! ok {
109
+ return sarama .ErrClosedConsumerGroup
110
+ }
111
+ ctx := GetContextWithMQHeader (msg .Msg .Headers )
112
+ if err := fn (ctx , string (msg .Msg .Key ), msg .Msg .Value ); err != nil {
113
+ return err
111
114
}
115
+ msg .Session .MarkMessage (msg .Msg , "" )
116
+ msg .Session .Commit ()
117
+ return nil
112
118
}
113
119
}
114
120
0 commit comments