1
0
mirror of https://github.com/mainflux/mainflux.git synced 2025-05-06 19:29:15 +08:00
__touk__ f8ce94e9bb
NOISSUE - Refactor MQTT subscriber (#1561)
* correct suscriber interface validator + refactore token error handling

Signed-off-by: tzzed <zerouali.t@gmail.com>

* apply review suggestion

Signed-off-by: tzzed <zerouali.t@gmail.com>

Co-authored-by: Dušan Borovčanin <dusan.borovcanin@mainflux.com>
2022-02-09 10:19:09 +01:00

85 lines
1.9 KiB
Go

// Copyright (c) Mainflux
// SPDX-License-Identifier: Apache-2.0
package mqtt
import (
"fmt"
"time"
mqtt "github.com/eclipse/paho.mqtt.golang"
"github.com/gogo/protobuf/proto"
log "github.com/mainflux/mainflux/logger"
"github.com/mainflux/mainflux/pkg/errors"
"github.com/mainflux/mainflux/pkg/messaging"
)
var (
errSubscribeTimeout = errors.New("failed to subscribe due to timeout reached")
errUnsubscribeTimeout = errors.New("failed to unsubscribe due to timeout reached")
)
var _ messaging.Subscriber = (*subscriber)(nil)
type subscriber struct {
client mqtt.Client
timeout time.Duration
logger log.Logger
}
// NewSubscriber returns a new MQTT message subscriber.
func NewSubscriber(address string, timeout time.Duration, logger log.Logger) (messaging.Subscriber, error) {
client, err := newClient(address, timeout)
if err != nil {
return nil, err
}
ret := subscriber{
client: client,
timeout: timeout,
logger: logger,
}
return ret, nil
}
func (sub subscriber) Subscribe(topic string, handler messaging.MessageHandler) error {
token := sub.client.Subscribe(topic, qos, sub.mqttHandler(handler))
if token.Error() != nil {
return token.Error()
}
ok := token.WaitTimeout(sub.timeout)
if !ok {
return errSubscribeTimeout
}
return token.Error()
}
func (sub subscriber) Unsubscribe(topic string) error {
token := sub.client.Unsubscribe(topic)
if token.Error() != nil {
return token.Error()
}
ok := token.WaitTimeout(sub.timeout)
if !ok {
return errUnsubscribeTimeout
}
return token.Error()
}
func (sub subscriber) mqttHandler(h messaging.MessageHandler) mqtt.MessageHandler {
return func(c mqtt.Client, m mqtt.Message) {
var msg messaging.Message
if err := proto.Unmarshal(m.Payload(), &msg); err != nil {
sub.logger.Warn(fmt.Sprintf("Failed to unmarshal received message: %s", err))
return
}
if err := h(msg); err != nil {
sub.logger.Warn(fmt.Sprintf("Failed to handle Mainflux message: %s", err))
}
}
}