155 lines
3.0 KiB
Go
155 lines
3.0 KiB
Go
package sysevent
|
|
|
|
import (
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/streadway/amqp"
|
|
"gitlink.org.cn/cloudream/common/pkgs/async"
|
|
"gitlink.org.cn/cloudream/common/utils/serder"
|
|
"gitlink.org.cn/cloudream/jcs-pub/common/types/datamap"
|
|
)
|
|
|
|
type PublisherEventChan = async.UnboundChannel[PublisherEvent]
|
|
|
|
type PublisherEvent interface {
|
|
IsPublisherEvent() bool
|
|
}
|
|
|
|
type PublisherExited struct {
|
|
PublisherEvent
|
|
Err error
|
|
}
|
|
|
|
type PublishError struct {
|
|
PublisherEvent
|
|
Err error
|
|
}
|
|
|
|
type OtherError struct {
|
|
PublisherEvent
|
|
Err error
|
|
}
|
|
|
|
type Publisher struct {
|
|
cfg Config
|
|
connection *amqp.Connection
|
|
channel *amqp.Channel
|
|
eventChan *async.UnboundChannel[SysEvent]
|
|
thisSource Source
|
|
}
|
|
|
|
func NewPublisher(cfg Config, thisSource Source) (*Publisher, error) {
|
|
if !cfg.Enabled {
|
|
return &Publisher{
|
|
cfg: cfg,
|
|
thisSource: thisSource,
|
|
}, nil
|
|
}
|
|
|
|
config := amqp.Config{
|
|
Vhost: cfg.VHost,
|
|
}
|
|
|
|
url := fmt.Sprintf("amqp://%s:%s@%s", cfg.Account, cfg.Password, cfg.Address)
|
|
connection, err := amqp.DialConfig(url, config)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
channel, err := connection.Channel()
|
|
if err != nil {
|
|
connection.Close()
|
|
return nil, fmt.Errorf("openning channel on connection: %w", err)
|
|
}
|
|
|
|
err = channel.ExchangeDeclare(cfg.Exchange, "fanout", false, true, false, false, nil)
|
|
if err != nil {
|
|
connection.Close()
|
|
return nil, fmt.Errorf("declare exchange: %w", err)
|
|
}
|
|
|
|
pub := &Publisher{
|
|
connection: connection,
|
|
channel: channel,
|
|
eventChan: async.NewUnboundChannel[SysEvent](),
|
|
thisSource: thisSource,
|
|
}
|
|
|
|
return pub, nil
|
|
}
|
|
|
|
func (p *Publisher) Start() *PublisherEventChan {
|
|
ch := async.NewUnboundChannel[PublisherEvent]()
|
|
go func() {
|
|
if !p.cfg.Enabled {
|
|
return
|
|
}
|
|
|
|
defer ch.Close()
|
|
defer p.channel.Close()
|
|
defer p.connection.Close()
|
|
|
|
for {
|
|
event := <-p.eventChan.Receive().Chan()
|
|
if event.Err != nil {
|
|
if event.Err == async.ErrChannelClosed {
|
|
ch.Send(PublisherExited{Err: nil})
|
|
} else {
|
|
ch.Send(PublisherExited{Err: event.Err})
|
|
}
|
|
return
|
|
}
|
|
|
|
eventData, err := serder.ObjectToJSONEx(event.Value)
|
|
if err != nil {
|
|
ch.Send(OtherError{Err: fmt.Errorf("serialize event data: %w", err)})
|
|
continue
|
|
}
|
|
|
|
err = p.channel.Publish(p.cfg.Exchange, "", false, false, amqp.Publishing{
|
|
ContentType: "text/plain",
|
|
Body: eventData,
|
|
Expiration: "60000", // 消息超时时间默认1分钟
|
|
})
|
|
if err != nil {
|
|
ch.Send(PublishError{Err: err})
|
|
continue
|
|
}
|
|
}
|
|
}()
|
|
|
|
return ch
|
|
}
|
|
|
|
func (p *Publisher) Stop() {
|
|
if !p.cfg.Enabled {
|
|
return
|
|
}
|
|
|
|
p.channel.Close()
|
|
p.connection.Close()
|
|
}
|
|
|
|
// Publish 发布事件,会自动补齐必要信息
|
|
func (p *Publisher) Publish(eventBody datamap.SysEventBody) {
|
|
if !p.cfg.Enabled {
|
|
return
|
|
}
|
|
|
|
p.eventChan.Send(datamap.SysEvent{
|
|
Timestamp: time.Now(),
|
|
Source: p.thisSource,
|
|
Body: eventBody,
|
|
})
|
|
}
|
|
|
|
// PublishRaw 完全原样发布事件,不补齐任何信息
|
|
func (p *Publisher) PublishRaw(evt SysEvent) {
|
|
if !p.cfg.Enabled {
|
|
return
|
|
}
|
|
|
|
p.eventChan.Send(evt)
|
|
}
|