2020-10-16 17:13:46 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2020-10-16 19:25:02 +02:00
|
|
|
"os"
|
|
|
|
|
"os/signal"
|
2020-10-26 09:55:29 +01:00
|
|
|
"runtime/debug"
|
2020-10-23 14:00:59 +02:00
|
|
|
"strings"
|
2020-10-16 19:25:02 +02:00
|
|
|
"syscall"
|
2020-10-23 14:00:59 +02:00
|
|
|
"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
|
|
|
|
|
|
2020-10-23 14:00:59 +02:00
|
|
|
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
|
2020-10-26 09:55:29 +01:00
|
|
|
bi, ok := debug.ReadBuildInfo()
|
|
|
|
|
if ok {
|
|
|
|
|
v = bi.Main.Version
|
2020-10-26 09:44:12 +01:00
|
|
|
} else {
|
2020-10-26 09:55:29 +01:00
|
|
|
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) {
|
2020-10-23 14:00:59 +02:00
|
|
|
// 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 {
|
2020-10-23 14:00:59 +02:00
|
|
|
log.Error(err)
|
2020-10-28 10:15:13 +01:00
|
|
|
ctrl.deinit()
|
2020-11-05 23:13:46 +01:00
|
|
|
if strings.Contains(err.Error(), "invalid username/password") {
|
|
|
|
|
return false, true, 1
|
|
|
|
|
}
|
2020-11-04 18:56:10 +01:00
|
|
|
return
|
2020-10-23 14:00:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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()
|
2020-10-23 14:00:59 +02:00
|
|
|
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
|
2020-10-18 11:15:31 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-23 14:00:59 +02:00
|
|
|
func reportError(err error) {
|
|
|
|
|
if !strings.Contains(err.Error(), "use of closed network connection") {
|
|
|
|
|
log.ErrorC(log.GetCallerFileName(true), ": ", err)
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-28 22:11:57 +01:00
|
|
|
requireWait := true
|
|
|
|
|
if strings.Contains(err.Error(), "got radio disconnected") {
|
|
|
|
|
requireWait = false
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-23 14:00:59 +02:00
|
|
|
// Non-blocking notify.
|
|
|
|
|
select {
|
2020-10-28 22:11:57 +01:00
|
|
|
case gotErrChan <- requireWait:
|
2020-10-23 14:00:59 +02:00
|
|
|
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
|
|
|
|
2020-10-23 14:00:59 +02:00
|
|
|
osSignal := make(chan os.Signal, 1)
|
|
|
|
|
signal.Notify(osSignal, os.Interrupt, syscall.SIGTERM)
|
2020-10-18 18:34:22 +02:00
|
|
|
|
2020-10-30 22:12:30 +01:00
|
|
|
if statusLog.isRealtimeInternal() {
|
2020-10-30 20:26:33 +01:00
|
|
|
keyboard.init()
|
|
|
|
|
}
|
2020-10-30 15:57:33 +01:00
|
|
|
|
2020-11-04 18:56:10 +01:00
|
|
|
var retries int
|
|
|
|
|
var requireWait bool
|
2020-10-23 14:00:59 +02:00
|
|
|
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-10-23 14:00:59 +02:00
|
|
|
}
|
2020-11-04 18:56:10 +01:00
|
|
|
log.Print("restarting control stream...")
|
2020-10-23 14:00:59 +02:00
|
|
|
}
|
2020-10-20 08:49:17 +02:00
|
|
|
|
2020-11-08 20:48:39 +01:00
|
|
|
rigctld.deinit()
|
2020-11-03 22:26:34 +01:00
|
|
|
serialTCPSrv.deinit()
|
2020-11-03 14:08:14 +01:00
|
|
|
runCmdRunner.stop()
|
|
|
|
|
serialCmdRunner.stop()
|
2020-10-28 22:32:45 +01:00
|
|
|
audio.deinit()
|
|
|
|
|
serialPort.deinit()
|
2020-10-30 20:26:33 +01:00
|
|
|
|
2020-10-30 22:12:30 +01:00
|
|
|
if statusLog.isRealtimeInternal() {
|
2020-10-30 20:26:33 +01:00
|
|
|
keyboard.deinit()
|
|
|
|
|
}
|
2020-10-28 22:32:45 +01:00
|
|
|
|
2020-10-23 14:00:59 +02:00
|
|
|
log.Print("exiting")
|
|
|
|
|
os.Exit(exitCode)
|
2020-10-16 17:13:46 +02:00
|
|
|
}
|