kappanhang/main.go

164 lines
3 KiB
Go
Raw Normal View History

2020-10-16 17:13:46 +02:00
package main
import (
2020-10-16 19:25:02 +02:00
"os"
"os/signal"
"runtime/debug"
"strings"
2020-10-16 19:25:02 +02:00
"syscall"
"time"
2020-10-16 17:13:46 +02:00
)
2020-11-04 18:56:10 +01:00
const waitBetweenRetries = time.Second
const retryCount = 5
const waitOnRetryFailure = 65 * time.Second
var gotErrChan = make(chan bool)
2020-11-01 12:53:21 +01:00
var quitChan = make(chan bool)
2020-10-16 17:13:46 +02:00
2020-10-26 08:39:12 +01:00
func getAboutStr() string {
2020-10-26 09:44:12 +01:00
var v string
bi, ok := debug.ReadBuildInfo()
if ok {
v = bi.Main.Version
2020-10-26 09:44:12 +01:00
} else {
v = "(devel)"
2020-10-26 09:44:12 +01:00
}
return "kappanhang " + v + " by Norbert Varga HA2NON and Akos Marton ES1AKOS https://github.com/nonoo/kappanhang"
2020-10-26 08:39:12 +01:00
}
2020-11-04 18:56:10 +01:00
func wait(d time.Duration, osSignal chan os.Signal) (shouldExit bool) {
for sec := d.Seconds(); sec > 0; sec-- {
log.Print("waiting ", sec, " seconds...")
select {
case <-time.After(time.Second):
case <-osSignal:
log.Print("sigterm received")
return true
case <-quitChan:
return true
}
}
return false
}
func runControlStream(osSignal chan os.Signal) (requireWait, shouldExit bool, exitCode int) {
// Depleting gotErrChan.
var finished bool
for !finished {
select {
case <-gotErrChan:
default:
finished = true
}
2020-10-18 13:19:52 +02:00
}
2020-10-28 18:03:35 +01:00
ctrl := &controlStream{}
2020-10-20 08:47:42 +02:00
2020-10-28 10:15:13 +01:00
if err := ctrl.init(); err != nil {
log.Error(err)
2020-10-28 10:15:13 +01:00
ctrl.deinit()
2020-11-04 18:56:10 +01:00
return
}
select {
2020-11-04 18:56:10 +01:00
// Need to wait before reinit because the IC-705 will disconnect our audio stream eventually if we relogin
// in a too short interval without a deauth...
case requireWait = <-gotErrChan:
2020-10-28 10:15:13 +01:00
ctrl.deinit()
return
case <-osSignal:
log.Print("sigterm received")
2020-10-28 10:15:13 +01:00
ctrl.deinit()
2020-11-04 18:56:10 +01:00
return false, true, 0
2020-11-01 12:53:21 +01:00
case <-quitChan:
ctrl.deinit()
2020-11-04 18:56:10 +01:00
return false, true, 0
}
}
func reportError(err error) {
if !strings.Contains(err.Error(), "use of closed network connection") {
log.ErrorC(log.GetCallerFileName(true), ": ", err)
}
requireWait := true
if strings.Contains(err.Error(), "got radio disconnected") {
requireWait = false
}
// Non-blocking notify.
select {
case gotErrChan <- requireWait:
default:
}
2020-10-16 19:25:02 +02:00
}
2020-10-16 17:13:46 +02:00
func main() {
parseArgs()
2020-10-25 21:25:15 +01:00
log.Init()
2020-10-26 08:39:12 +01:00
log.Print(getAboutStr())
2020-10-20 08:47:42 +02:00
osSignal := make(chan os.Signal, 1)
signal.Notify(osSignal, os.Interrupt, syscall.SIGTERM)
if statusLog.isRealtimeInternal() {
keyboard.init()
}
2020-11-04 18:56:10 +01:00
var retries int
var requireWait bool
var shouldExit bool
var exitCode int
2020-11-04 18:56:10 +01:00
exit:
for {
requireWait, shouldExit, exitCode = runControlStream(osSignal)
if shouldExit {
break
}
2020-10-25 20:07:26 +01:00
select {
case <-osSignal:
log.Print("sigterm received")
2020-11-04 18:56:10 +01:00
break exit
2020-11-01 12:53:21 +01:00
case <-quitChan:
2020-11-04 18:56:10 +01:00
break exit
2020-10-25 20:07:26 +01:00
default:
}
2020-11-04 18:56:10 +01:00
if requireWait {
if retries < retryCount {
retries++
shouldExit = wait(waitBetweenRetries, osSignal)
} else {
retries = 0
shouldExit = wait(waitOnRetryFailure, osSignal)
}
} else {
retries = 0
shouldExit = wait(time.Second, osSignal)
}
if shouldExit {
break
}
2020-11-04 18:56:10 +01:00
log.Print("restarting control stream...")
}
2020-10-20 08:49:17 +02:00
2020-11-03 22:26:34 +01:00
serialTCPSrv.deinit()
2020-11-03 14:08:14 +01:00
runCmdRunner.stop()
serialCmdRunner.stop()
rigctldRunner.stop()
audio.deinit()
serialPort.deinit()
if statusLog.isRealtimeInternal() {
keyboard.deinit()
}
log.Print("exiting")
os.Exit(exitCode)
2020-10-16 17:13:46 +02:00
}