package throttle import ( "github.com/cyrilix/robocar-base/service" "github.com/cyrilix/robocar-protobuf/go/events" mqtt "github.com/eclipse/paho.mqtt.golang" "go.uber.org/zap" "google.golang.org/protobuf/proto" "sync" "time" ) func New(client mqtt.Client, throttleTopic, driveModeTopic, rcThrottleTopic string, minValue, maxValue float32, publishPilotFrequency int) *Controller { return &Controller{ client: client, throttleTopic: throttleTopic, driveModeTopic: driveModeTopic, rcThrottleTopic: rcThrottleTopic, minThrottle: minValue, maxThrottle: maxValue, driveMode: events.DriveMode_USER, publishPilotFrequency: publishPilotFrequency, } } type Controller struct { client mqtt.Client throttleTopic string minThrottle, maxThrottle float32 muDriveMode sync.RWMutex driveMode events.DriveMode cancel chan interface{} publishPilotFrequency int driveModeTopic, rcThrottleTopic string } func (c *Controller) Start() error { if err := registerCallbacks(c); err != nil { zap.S().Errorf("unable to register callbacks: %v", err) return err } c.cancel = make(chan interface{}) ticker := time.NewTicker(1 * time.Second / time.Duration(c.publishPilotFrequency)) for { select { case <-ticker.C: c.onPublishPilotValue() case <-c.cancel: break } } } func (c *Controller) onPublishPilotValue() { c.muDriveMode.RLock() defer c.muDriveMode.RUnlock() if c.driveMode != events.DriveMode_PILOT { return } throttleMsg := events.ThrottleMessage{ Throttle: c.minThrottle, Confidence: 1.0, } payload, err := proto.Marshal(&throttleMsg) if err != nil { zap.S().Errorf("unable to marshal %T protobuf content: %err", throttleMsg, err) return } publish(c.client, c.throttleTopic, &payload) } func (c *Controller) Stop() { close(c.cancel) service.StopService("throttle", c.client, c.driveModeTopic, c.rcThrottleTopic) } func (c *Controller) onDriveMode(_ mqtt.Client, message mqtt.Message) { var msg events.DriveModeMessage err := proto.Unmarshal(message.Payload(), &msg) if err != nil { zap.S().Errorf("unable to unmarshal protobuf %T message: %v", msg, err) return } c.muDriveMode.Lock() defer c.muDriveMode.Unlock() c.driveMode = msg.GetDriveMode() } func (c *Controller) onRCThrottle(_ mqtt.Client, message mqtt.Message) { c.muDriveMode.RLock() defer c.muDriveMode.RUnlock() if c.driveMode == events.DriveMode_USER { // Republish same content payload := message.Payload() var throttleMsg events.ThrottleMessage err := proto.Unmarshal(payload, &throttleMsg) if err != nil { zap.S().Errorf("unable to unmarshall throttle msg to check throttle value: %v", err) return } zap.S().Debugf("publish new throttle value from rc: %v", throttleMsg.GetThrottle()) if throttleMsg.GetThrottle() > c.maxThrottle { zap.S().Debugf("throttle upper that max value allowed, patch value from %v to %v", throttleMsg.GetThrottle(), c.maxThrottle) throttleMsg.Throttle = c.maxThrottle payloadPatched, err := proto.Marshal(&throttleMsg) if err != nil { zap.S().Errorf("unable to marshall throttle msg: %v", err) return } publish(c.client, c.throttleTopic, &payloadPatched) return } publish(c.client, c.throttleTopic, &payload) } } var registerCallbacks = func(p *Controller) error { err := service.RegisterCallback(p.client, p.driveModeTopic, p.onDriveMode) if err != nil { return err } err = service.RegisterCallback(p.client, p.rcThrottleTopic, p.onRCThrottle) if err != nil { return err } return nil } var publish = func(client mqtt.Client, topic string, payload *[]byte) { client.Publish(topic, 0, false, *payload) }