You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

108 lines
2.9 KiB

This file contains ambiguous Unicode characters!

This file contains ambiguous Unicode characters that may be confused with others in your current locale. If your use case is intentional and legitimate, you can safely ignore this warning. Use the Escape button to highlight these characters.

package topics
import (
amqp "github.com/rabbitmq/amqp091-go"
"goskeleton/app/global/variable"
"goskeleton/app/utils/rabbitmq/error_record"
)
// CreateProducer 创建一个生产者
func CreateProducer(options ...OptionsProd) (*producer, error) {
// 获取配置信息
conn, err := amqp.Dial(variable.ConfigYml.GetString("RabbitMq.Topics.Addr"))
exchangeType := variable.ConfigYml.GetString("RabbitMq.Topics.ExchangeType")
exchangeName := variable.ConfigYml.GetString("RabbitMq.Topics.ExchangeName")
queueName := variable.ConfigYml.GetString("RabbitMq.Topics.QueueName")
durable := variable.ConfigYml.GetBool("RabbitMq.Topics.Durable")
if err != nil {
variable.ZapLog.Error(err.Error())
return nil, err
}
prod := &producer{
connect: conn,
exchangeType: exchangeType,
exchangeName: exchangeName,
queueName: queueName,
durable: durable,
}
// 加载用户设置的参数
for _, val := range options {
val.apply(prod)
}
return prod, nil
}
// 定义一个消息队列结构体Topics 模型
type producer struct {
connect *amqp.Connection
exchangeType string
exchangeName string
queueName string
durable bool
occurError error
enableDelayMsgPlugin bool // 是否使用延迟队列模式
args amqp.Table
}
// Send 发送消息
// 参数:
// routeKey 路由键、
// data 发送的数据、
// delayMillisecond 延迟时间(毫秒),只有启用了消息延迟插件才有效果
func (p *producer) Send(routeKey, data string, delayMillisecond int) bool {
// 获取一个频道
ch, err := p.connect.Channel()
p.occurError = error_record.ErrorDeal(err)
defer func() {
_ = ch.Close()
}()
// 声明交换机,该模式生产者只负责将消息投递到交换机即可
err = ch.ExchangeDeclare(
p.exchangeName, //交换器名称
p.exchangeType, //topic模式
p.durable, //交换机是否持久化
!p.durable, //交换器是否自动删除
false,
false,
p.args,
)
p.occurError = error_record.ErrorDeal(err)
// 如果交换机是持久化的,那么消息也设置为持久化
msgPersistent := amqp.Transient
if p.durable {
msgPersistent = amqp.Persistent
}
// 投递消息
if err == nil {
err = ch.Publish(
p.exchangeName, // 交换机名称
routeKey, // topics 模式默认为空即可
false,
false,
amqp.Publishing{
DeliveryMode: msgPersistent, //消息是否持久化,这里与保持保持一致即可
ContentType: "text/plain",
Body: []byte(data),
Headers: amqp.Table{
"x-delay": delayMillisecond, // 延迟时间: 毫秒
},
})
}
p.occurError = error_record.ErrorDeal(err)
if p.occurError != nil { // 发生错误,返回 false
return false
} else {
return true
}
}
// Close 发送完毕手动关闭这样不影响send多次发送数据
func (p *producer) Close() {
_ = p.connect.Close()
}