blob: 5f4f04eafc76cf958d75a7cfe293b548128ef00b [file] [log] [blame]
package poller
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
import (
"os"
"time"
"github.com/apache/trafficcontrol/lib/go-log"
"github.com/apache/trafficcontrol/lib/go-tc"
"github.com/apache/trafficcontrol/traffic_monitor/handler"
"github.com/apache/trafficcontrol/traffic_monitor/towrap" // TODO move to common
)
type MonitorCfg struct {
CDN string
Cfg tc.TrafficMonitorConfigMap
}
type MonitorConfigPoller struct {
Session towrap.TrafficOpsSessionThreadsafe
SessionChannel chan towrap.TrafficOpsSessionThreadsafe
ConfigChannel chan MonitorCfg
OpsConfigChannel chan handler.OpsConfig
Interval time.Duration
IntervalChan chan time.Duration
OpsConfig handler.OpsConfig
}
// NewMonitorConfig Creates and returns a new MonitorConfigPoller.
func NewMonitorConfig(interval time.Duration) MonitorConfigPoller {
return MonitorConfigPoller{
Interval: interval,
SessionChannel: make(chan towrap.TrafficOpsSessionThreadsafe),
// ConfigChannel MUST have a buffer size 1, to make the nonblocking writeConfig work
ConfigChannel: make(chan MonitorCfg, 1),
OpsConfigChannel: make(chan handler.OpsConfig),
IntervalChan: make(chan time.Duration),
}
}
// writeConfig writes the given config to the Config chan. This is nonblocking, and immediately returns.
// Because readers only ever want the latest config, if nobody has read the previous write, we remove it. Since the config chan is buffered size 1, this function is therefore asynchronous.
func (p MonitorConfigPoller) writeConfig(cfg MonitorCfg) {
for {
select {
case p.ConfigChannel <- cfg:
return // return after successfully writing.
case <-p.ConfigChannel:
// if the channel buffer was full, read, then loop and try to write again
}
}
}
func (p MonitorConfigPoller) Poll() {
tick := time.NewTicker(p.Interval)
defer tick.Stop()
defer func() {
if err := recover(); err != nil {
log.Errorf("MonitorConfigPoller panic: %v\n", err)
} else {
log.Errorf("MonitorConfigPoller failed without panic\n")
}
log.Errorf("%s\n", stacktrace())
os.Exit(1) // The Monitor can't run without a MonitorConfigPoller
}()
for {
// Every case MUST be asynchronous and non-blocking, to prevent livelocks. If a chan must be written to, it must either be buffered AND remove existing values, or be written to in a goroutine.
select {
case opsConfig := <-p.OpsConfigChannel:
log.Infof("MonitorConfigPoller: received new opsConfig: %v\n", opsConfig)
p.OpsConfig = opsConfig
case session := <-p.SessionChannel:
log.Infof("MonitorConfigPoller: received new session: %v\n", session)
p.Session = session
case i := <-p.IntervalChan:
if i == p.Interval {
continue
}
log.Infof("MonitorConfigPoller: received new interval: %v\n", i)
if i < 0 {
log.Errorf("MonitorConfigPoller: received negative interval: %v; ignoring\n", i)
continue
}
p.Interval = i
tick.Stop()
tick = time.NewTicker(p.Interval)
case <-tick.C:
if !p.Session.Initialized() || p.OpsConfig.CdnName == "" {
log.Warnln("MonitorConfigPoller: skipping this iteration, Session is nil")
continue
}
monitorConfig, err := p.Session.TrafficMonitorConfigMap(p.OpsConfig.CdnName)
if err != nil {
log.Errorf("MonitorConfigPoller: %s\n %v\n", err, monitorConfig)
continue
}
// poll the CRConfig so that it is synchronized with the TMConfig
if _, err := p.Session.CRConfigRaw(p.OpsConfig.CdnName); err != nil {
log.Errorf("MonitorConfigPoller: error getting CRConfig: %v", err)
continue
}
p.writeConfig(MonitorCfg{CDN: p.OpsConfig.CdnName, Cfg: *monitorConfig})
}
}
}