go实现的redis消息队列

go实现的消息队列 redis_queue

  • 支持并发队列
  • 支持Topic,Group注册队列监听消息
  • 支持消息异常监听重启队列

贴相关核心代码,最后附上git链接

//消息载体
type QueuePayload struct {
    ID       string      `json:"id"`
    IsFast   bool        `json:"is_fast"`
    Topic    string      `json:"topic"`
    Group    string      `json:"group"`
    Body     interface{} `json:"body"`
}

var instanceQueueManager *QueueManager
var onceQueueManager sync.Once

//队列管理器
type QueueManager struct {
    db             *redis.Client
    MaxRetry       int						//重试的最大次数
    RecoverCh      chan RecoverData			//队列恢复监听通道
    Handlers       map[string]interface{}	//注册队列Map
}

//初始化队列管理器
func NewQueueManager() *QueueManager {
    onceQueueManager.Do(func() {
        instanceQueueManager = &QueueManager{}
        instanceQueueManager.MaxRetry = 3
        instanceQueueManager.Handlers = make(map[string]interface{})
    })
    return instanceQueueManager
}

func (r *QueueManager) GetQueueName(topic string, group string) string {
    var name string
    if len(group) > 0 {
        name = fmt.Sprintf("Queue_%s::%s", topic, group)
    } else {
        name = fmt.Sprintf("Queue_%s", topic)
    }
    return name
}

//注册队列
func (r *QueueManager) RegisterQueue(topic string, group string, handler interface{}) error{
    name := r.GetQueueName(topic, group)
    if _, ok := r.Handlers[name]; ok {
        return errors.New("is exits")
    }else{
        r.Handlers[name] = handler
        go r.QueueConsume(topic, group)
    }
    return nil
}

//生产者执行入队列
func (r *QueueManager) QueuePublish(payload *QueuePayload) error {
    if len(payload.Topic) <= 0 {
        return errors.New("TopicId can not be empty")
    }
    id, err := uuid.NewUUID()
    if err != nil {
        return err
    }
    payload.ID = id.String()
    payloadStr, _ := json.Marshal(payload)

    r.db.LPush(r.GetQueueName(payload.Topic, payload.Group), payloadStr)
    return nil
}

//消费者执行出队列
func (r *QueueManager) QueueConsume(topic string, group string) {
    defer func() {
        if err := recover(); err != nil {
            var stacktrace string
            for i := 1; ; i++ {
                _, f, l, got := runtime.Caller(i)
                if !got {
                    break
                }
                stacktrace += fmt.Sprintf("%s:%d\n", f, l)
            }
            // when stack finishes
            logMessage := fmt.Sprintf("Trace: %s\n", err)
            logMessage += fmt.Sprintf("\n%s", stacktrace)
            log.Println(logMessage)
            //执行恢复函数
            r.handleRecover(topic, group)
        }
    }()
    for {
        //消费者执行出列
        var payload QueuePayload
        result := r.db.BRPop(0, r.GetQueueName(topic, group))
        if (len(result.Val()) > 0) {
            vals := result.Val()[1]
            err := json.Unmarshal([]byte(vals), &payload)
            if err != nil {
                log.Println("BRPOP json.Unmarshal Error:", err)
                continue
            }
            //执行回调函数
            r.handleCallBack(&payload)
        }
    }
}

//执行恢复函数
func (r *QueueManager) handleRecover(topic string, group string) {
    handleName := r.GetQueueName(topic, group)
    handler, ok := r.Handlers[handleName]
    if r.RecoverCh != nil && ok{
        r.RecoverCh <- RecoverData{topic, group, handler}
    }
}

//执行回调函数
func (r *QueueManager) handleCallBack (payload *QueuePayload){
    handleName := r.GetQueueName(payload.Topic, payload.Group)
    it := r.Handlers[handleName]
    if it != nil {
        if ob, ok := it.(Queueable); ok {
            //同步执行Max次,保证队列顺序,失败则丢弃消息,
            for i:=0; i< r.MaxRetry; i++ {
                rs := ob.Execute(payload)
                if rs.State{
                    break
                }
            }
        }else{
            log.Println("no ExecuteFunc,pop:", payload)
        }
    }
}

上述完整代码链接: git链接

参考的代码:https://github.com/bennya8/go_redis_queue_manager

你可能感兴趣的:(golang,redis,golang,队列,redis)