196 lines
4.2 KiB
Go
196 lines
4.2 KiB
Go
package controller
|
|
|
|
import (
|
|
"fmt"
|
|
"github.com/golang/glog"
|
|
api "github.com/osrg/gobgp/api"
|
|
"net"
|
|
"os/exec"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
func portMonitor(protocol, port string) bool {
|
|
switch protocol {
|
|
case "tcp":
|
|
conn, err := net.Listen(protocol, ":"+port)
|
|
if err != nil {
|
|
glog.V(4).Infof("Monitor tcp port up")
|
|
return true
|
|
}
|
|
defer conn.Close()
|
|
case "udp":
|
|
conn, err := net.ListenPacket(protocol, ":"+port)
|
|
if err != nil {
|
|
glog.V(4).Infof("Monitor udp port up")
|
|
return true
|
|
}
|
|
defer conn.Close()
|
|
}
|
|
return false
|
|
}
|
|
|
|
func execMonitor(cmd string) bool {
|
|
out := exec.Command("bash", "-c", cmd)
|
|
if err := out.Start(); err != nil {
|
|
glog.V(2).Infof("Cannot exec cmd: %s : %v", cmd, err)
|
|
return false
|
|
}
|
|
if err := out.Wait(); err != nil {
|
|
if _, ok := err.(*exec.ExitError); ok {
|
|
glog.V(4).Infof("Monitor cmd failed")
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
type appMon struct {
|
|
app *App
|
|
done chan bool
|
|
announced bool
|
|
checkOn bool
|
|
}
|
|
|
|
type MonitorMgr struct {
|
|
monitors map[string]*appMon
|
|
cleanups map[string]chan bool
|
|
c *Controller
|
|
monitorInterval time.Duration
|
|
cleanupTimer time.Duration
|
|
sync.Mutex
|
|
}
|
|
|
|
func NewMonitor(localAS, peerAS int, monitorInterval time.Duration, peerIP string, cleanup time.Duration) *MonitorMgr {
|
|
c, err := NewController(localAS, peerAS, peerIP)
|
|
if err != nil {
|
|
glog.Exitf("Failed to start BGP controller: %v", err)
|
|
}
|
|
return &MonitorMgr{
|
|
c: c,
|
|
monitors: make(map[string]*appMon),
|
|
cleanups: make(map[string]chan bool),
|
|
monitorInterval: monitorInterval,
|
|
cleanupTimer: cleanup,
|
|
}
|
|
}
|
|
|
|
func (m *MonitorMgr) Add(app *App) {
|
|
// stop and start a new one if one already running
|
|
m.Remove(app.Name)
|
|
appMon := &appMon{app: app, done: make(chan bool)}
|
|
m.Lock()
|
|
m.monitors[app.Name] = appMon
|
|
m.Unlock()
|
|
go m.runLoop(appMon)
|
|
glog.Infof("Registered a new app: %v", app)
|
|
}
|
|
|
|
func (m *MonitorMgr) Remove(appName string) {
|
|
m.Lock()
|
|
defer m.Unlock()
|
|
if a, ok := m.monitors[appName]; ok {
|
|
if a.checkOn {
|
|
a.done <- true
|
|
}
|
|
if a.announced {
|
|
if err := m.c.Withdraw(a.app.Vip); err != nil {
|
|
glog.Errorf("Failed to withdraw route: %v", err)
|
|
}
|
|
}
|
|
deleteLoopback(appName)
|
|
}
|
|
delete(m.monitors, appName)
|
|
}
|
|
|
|
func (m *MonitorMgr) checkCond(am *appMon) error {
|
|
var cond bool
|
|
app := am.app
|
|
switch app.Monitor.Type {
|
|
case Monitor_PORT:
|
|
cond = portMonitor(app.Monitor.Protocol, app.Monitor.Port)
|
|
case Monitor_EXEC:
|
|
cond = execMonitor(app.Monitor.Cmd)
|
|
}
|
|
m.Lock()
|
|
defer m.Unlock()
|
|
if cond {
|
|
glog.V(2).Infof("%s Monitor for app: %s succeeded", app.Monitor.Type.String(), app.Name)
|
|
if !am.announced {
|
|
if err := addLoopback(app.Name, app.Vip); err != nil {
|
|
return err
|
|
}
|
|
if err := m.c.Announce(app.Vip); err != nil {
|
|
return fmt.Errorf("Failed to announce route: %v", err)
|
|
}
|
|
am.announced = true
|
|
if exit, ok := m.cleanups[app.Name]; ok {
|
|
exit <- true
|
|
}
|
|
}
|
|
} else {
|
|
glog.V(2).Infof("%s Monitor for app: %s Failed", app.Monitor.Type.String(), app.Name)
|
|
if am.announced {
|
|
if err := m.c.Withdraw(app.Vip); err != nil {
|
|
return fmt.Errorf("Failed to withdraw route: %v", err)
|
|
}
|
|
am.announced = false
|
|
exit := make(chan bool)
|
|
go m.Cleanup(app.Name, exit)
|
|
m.cleanups[app.Name] = exit
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (m *MonitorMgr) runLoop(am *appMon) {
|
|
am.checkOn = true
|
|
if err := m.checkCond(am); err != nil {
|
|
glog.Errorln(err)
|
|
}
|
|
t := time.NewTicker(m.monitorInterval)
|
|
defer t.Stop()
|
|
for {
|
|
select {
|
|
case <-t.C:
|
|
if err := m.checkCond(am); err != nil {
|
|
glog.Errorln(err)
|
|
}
|
|
case <-am.done:
|
|
glog.V(2).Infof("Exit run-loop for app: %s", am.app.Name)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func (m *MonitorMgr) CloseAll() {
|
|
glog.Infof("Shutting down all open bgp sessions")
|
|
if err := m.c.Shutdown(); err != nil {
|
|
glog.Errorf("Failed to shut-down BGP: %v", err)
|
|
}
|
|
for name, am := range m.monitors {
|
|
if am.checkOn {
|
|
am.done <- true
|
|
}
|
|
deleteLoopback(name)
|
|
}
|
|
}
|
|
|
|
func (m *MonitorMgr) Cleanup(app string, exit chan bool) {
|
|
t := time.NewTimer(m.cleanupTimer)
|
|
defer t.Stop()
|
|
for {
|
|
select {
|
|
case <-t.C:
|
|
glog.Infof("Cleaning up app %s", app)
|
|
m.Remove(app)
|
|
case <-exit:
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func (m *MonitorMgr) GetInfo() (*api.Peer, error) {
|
|
return m.c.PeerInfo()
|
|
}
|