speedy/main.go
Denis-Cosmin Nutiu 64256f57cc initial commit
2021-10-04 14:03:36 +03:00

156 lines
4.1 KiB
Go

package main
import (
"github.com/confluentinc/confluent-kafka-go/kafka"
"github.com/getsentry/sentry-go"
"github.com/goccy/go-json"
"os"
"os/signal"
"speedy/pkg"
"sync"
)
func main() {
configurator, err := pkg.NewViperConfigurator()
if err != nil {
panic(err)
}
// Grab hostName
hostName, err := os.Hostname()
if err != nil {
pkg.SugaredLogger.Warn("Failed to get hostname! Using generic name.")
hostName = "Speedy"
}
pkg.SugaredLogger.Infof("Using hostname %s", hostName)
// Init logging
config := configurator.GetConfig()
pkg.InitLoggingWithParams(config.LoggingLevel, "console", "")
// Init sentry
err = sentry.Init(sentry.ClientOptions{
// Either set your DSN here or set the SENTRY_DSN environment variable.
Dsn: config.SentryDSN,
// Either set environment and release here or set the SENTRY_ENVIRONMENT
// and SENTRY_RELEASE environment variables.
Environment: hostName,
Release: "speedy-gonzales",
})
if err != nil {
pkg.SugaredLogger.Error("failed to init Sentry.")
}
// Init kafka
pkg.SugaredLogger.Infof("Using config:\n %s", config.ToPrettyJson())
kafkaConsumer, err := kafka.NewConsumer(&kafka.ConfigMap{
"bootstrap.servers": config.KafkaBoostrapServers,
"group.id": config.KafkaGroupId,
"auto.offset.reset": config.KafkaOffsetReset,
"socket.timeout.ms": "300000",
})
if err != nil {
panic(err)
}
err = kafkaConsumer.SubscribeTopics(config.SubscribeTopics, nil)
if err != nil {
pkg.SugaredLogger.Errorf("failed to subscribe: %s", err)
sentry.CaptureException(err)
panic(err)
}
defer func(c *kafka.Consumer) {
err := c.Close()
if err != nil {
panic(err)
}
}(kafkaConsumer)
pkg.SugaredLogger.Info("Initializing")
// Init Sink & Pusher
var lokiClient = pkg.LokiClientFactoryCreate(config.LokiPushMode, config.LokiPushUrl)
var speedyPusher = pkg.NewPusher(lokiClient, config.BufferMaxBatchSize, config.BufferMaxBytesSize)
go speedyPusher.RunForever()
isRunning := true
var waitGroup sync.WaitGroup
for i := 0; i < config.KafkaPollingGoroutines; i++ {
waitGroup.Add(1)
go func() {
defer waitGroup.Done()
for isRunning {
ev := kafkaConsumer.Poll(config.KafkaPollingTimeoutMs)
switch event := ev.(type) {
case kafka.AssignedPartitions:
err := kafkaConsumer.Assign(event.Partitions)
if err != nil {
pkg.SugaredLogger.Error(err)
sentry.CaptureException(err)
return
}
case kafka.RevokedPartitions:
err := kafkaConsumer.Unassign()
if err != nil {
pkg.SugaredLogger.Error(err)
sentry.CaptureException(err)
return
}
case *kafka.Message:
var messageMap = make(map[string]interface{})
err := json.Unmarshal(event.Value, &messageMap)
if err != nil {
pkg.SugaredLogger.Error(err)
continue
}
flattenMap := pkg.FlattenMap(messageMap)
flattenMapString, err := json.Marshal(flattenMap)
if err != nil {
pkg.SugaredLogger.Error(err)
continue
}
labelsMap := map[string]string{
"key": *event.TopicPartition.Topic,
}
clientId := (*flattenMap)["clientID"]
clientIdStr, ok := clientId.(string)
if ok {
labelsMap["clientId"] = clientIdStr
}
// Size of clientId key and val
labelsSize := 8 + len(clientIdStr)
speedyPusher.DataChannel <- pkg.LokiStream{
Labels: labelsMap,
Values: [][]string{{"", string(flattenMapString)}},
Size: len(event.Value) + labelsSize,
}
case kafka.PartitionEOF:
pkg.SugaredLogger.Info()
case kafka.Error:
if event.Code() == kafka.ErrTimedOut {
pkg.SugaredLogger.Debugf("Consumer error: %v\n", err)
} else {
// The client will automatically try to recover from all errors.
pkg.SugaredLogger.Warnf("Consumer error: %v\n", err)
sentry.CaptureException(err)
}
default:
}
}
}()
}
go func() {
// Handle SIGINT
c := make(chan os.Signal, 1)
signal.Notify(c, os.Interrupt)
// Block until a signal is received.
<-c
pkg.SugaredLogger.Info("Received SIGINT, shutting down.")
isRunning = false
}()
waitGroup.Wait()
pkg.SugaredLogger.Info("Exiting.")
}