mirror of
https://github.com/mainflux/mainflux.git
synced 2025-04-28 13:48:49 +08:00

* NOISSUEE - Create broker package for NATS Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Create funcs to return NATS connection Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * mv os.exit to main Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Fix Reviews Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Fix tests and typos Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Fix CI Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Fix reviews Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Unify Publisher and Subscriber interfaces Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Rename Nats interface Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * typo Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Mv message.pb.go, messsage.proto and topics.go to broker directory Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Fix go.mod Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Use mainflux broker for writers and twins services Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Fix go.mod Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Fix twins tests Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Fix make proto Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Fix message.proto Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Fix golangcibot Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * regenerate message.pb.go Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Fix comment Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Fix comment Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Fix make proto Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com> * Add NATS errors Signed-off-by: Manuel Imperiale <manuel.imperiale@gmail.com>
102 lines
2.6 KiB
Go
102 lines
2.6 KiB
Go
// Copyright (c) Mainflux
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
package writers
|
|
|
|
import (
|
|
"fmt"
|
|
"io/ioutil"
|
|
|
|
"github.com/BurntSushi/toml"
|
|
"github.com/gogo/protobuf/proto"
|
|
"github.com/mainflux/mainflux/broker"
|
|
"github.com/mainflux/mainflux/errors"
|
|
"github.com/mainflux/mainflux/logger"
|
|
"github.com/mainflux/mainflux/transformers"
|
|
"github.com/mainflux/mainflux/transformers/senml"
|
|
"github.com/nats-io/nats.go"
|
|
)
|
|
|
|
var (
|
|
errOpenConfFile = errors.New("Unable to open configuration file")
|
|
errParseConfFile = errors.New("Unable to parse configuration file")
|
|
)
|
|
|
|
type consumer struct {
|
|
broker broker.Nats
|
|
repo MessageRepository
|
|
transformer transformers.Transformer
|
|
logger logger.Logger
|
|
}
|
|
|
|
// Start method starts consuming messages received from NATS.
|
|
// This method transforms messages to SenML format before
|
|
// using MessageRepository to store them.
|
|
func Start(broker broker.Nats, repo MessageRepository, transformer transformers.Transformer, queue string, subjectsCfgPath string, logger logger.Logger) error {
|
|
c := consumer{
|
|
broker: broker,
|
|
repo: repo,
|
|
transformer: transformer,
|
|
logger: logger,
|
|
}
|
|
|
|
subjects, err := loadSubjectsConfig(subjectsCfgPath)
|
|
if err != nil {
|
|
logger.Warn(fmt.Sprintf("Failed to load subjects: %s", err))
|
|
}
|
|
|
|
for _, subject := range subjects {
|
|
_, err := broker.QueueSubscribe(subject, queue, c.consume)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (c *consumer) consume(m *nats.Msg) {
|
|
var msg broker.Message
|
|
if err := proto.Unmarshal(m.Data, &msg); err != nil {
|
|
c.logger.Warn(fmt.Sprintf("Failed to unmarshal received message: %s", err))
|
|
return
|
|
}
|
|
|
|
t, err := c.transformer.Transform(msg)
|
|
if err != nil {
|
|
c.logger.Warn(fmt.Sprintf("Failed to tranform received message: %s", err))
|
|
return
|
|
}
|
|
msgs, ok := t.([]senml.Message)
|
|
if !ok {
|
|
c.logger.Warn("Invalid message format from the Transformer output.")
|
|
return
|
|
}
|
|
|
|
if err := c.repo.Save(msgs...); err != nil {
|
|
c.logger.Warn(fmt.Sprintf("Failed to save message: %s", err))
|
|
return
|
|
}
|
|
}
|
|
|
|
type filterConfig struct {
|
|
List []string `toml:"filter"`
|
|
}
|
|
|
|
type subjectsConfig struct {
|
|
Subjects filterConfig `toml:"subjects"`
|
|
}
|
|
|
|
func loadSubjectsConfig(subjectsConfigPath string) ([]string, error) {
|
|
data, err := ioutil.ReadFile(subjectsConfigPath)
|
|
if err != nil {
|
|
return []string{broker.SubjectAllChannels}, errors.Wrap(errOpenConfFile, err)
|
|
}
|
|
|
|
var subjectsCfg subjectsConfig
|
|
if err := toml.Unmarshal(data, &subjectsCfg); err != nil {
|
|
return []string{broker.SubjectAllChannels}, errors.Wrap(errParseConfFile, err)
|
|
}
|
|
|
|
return subjectsCfg.Subjects.List, nil
|
|
}
|