Compare commits

...

2 commits

Author SHA1 Message Date
1e99
e660d5937a cleanup code 2025-03-22 14:31:17 +01:00
1e99
b58e786943 add todos to readme 2025-03-22 14:31:02 +01:00
5 changed files with 53 additions and 70 deletions

View file

@ -5,7 +5,7 @@ Configuration is done using environment variables.
- `WOLBODGE_ADDRESS`: The address that WoLBodge should listen on, defaults to `:3000`. - `WOLBODGE_ADDRESS`: The address that WoLBodge should listen on, defaults to `:3000`.
- `WOLBODGE_STORAGE_TYPE`: Storage type to pick, defaults to `ram`. - `WOLBODGE_STORAGE_TYPE`: Storage type to pick, defaults to `ram`.
- `ram`: Sessions are stored in RAM. - `ram`: Sessions are stored in RAM.
- `WOLBODGE_DEVICE_TYPE`: Device type to pick, defaults to `test`: - `WOLBODGE_DEVICE_TYPE`: Device type to pick, defaults to `test`.
- `test`: A dummy device, used for testing. - `test`: A dummy device, used for testing.
- `libgpiod`: Uses the libgpiod commands under the hood. The gpio chip is specified using `WOLBODGE_DEVICE_GPIOCHIP`, the power button pin using `WOLBODGE_DEVICE_POWER_BUTTON_PIN` and the power LED pin using `WOLBODGE_DEVICE_POWER_LED_PIN`. - `libgpiod`: Uses the libgpiod commands under the hood. The gpio chip is specified using `WOLBODGE_DEVICE_GPIOCHIP`, the power button pin using `WOLBODGE_DEVICE_POWER_BUTTON_PIN` and the power LED pin using `WOLBODGE_DEVICE_POWER_LED_PIN`.
@ -36,3 +36,7 @@ if [ $exit_code != 0 ]; then
exit 1 exit 1
fi fi
``` ```
## TODOs
- Alerts when a session has existed for a long time
- Command device that can use commands to turn on/off a device

66
cron.go
View file

@ -1,66 +0,0 @@
package main
import (
"log"
"time"
"git.1e99.eu/1e99/wolbodge/devices"
"git.1e99.eu/1e99/wolbodge/sessions"
)
func DeviceCheck(logger *log.Logger, device devices.Device, wantStatus func() (bool, error), exit <-chan bool) {
for {
select {
case <-exit:
break
default:
status, err := device.Status()
if err != nil {
logger.Printf("Failed to fetch device status, waiting 1 minute until next attempt: %s", err)
time.Sleep(1 * time.Minute)
continue
}
want, err := wantStatus()
if err != nil {
logger.Printf("Failed to fetch want status, waiting 1 minute until next attempt: %s", err)
time.Sleep(1 * time.Minute)
continue
}
if status == want {
// Don't lock up our single-threaded session storage too much
time.Sleep(2 * time.Second)
continue
}
err = device.PushPowerButton()
if err != nil {
logger.Printf("Failed to push power button, waiting 2 minutes until next attempt: %s", err)
// Cooldown to not overwhelm the device with start/stop requests
time.Sleep(2 * time.Minute)
return
}
logger.Printf("Pushed power button")
time.Sleep(2 * time.Minute)
}
}
}
func PruneStorage(logger *log.Logger, storage sessions.Storage, interval time.Duration, exit <-chan bool) {
for {
select {
case <-exit:
break
default:
/*pruned, err := storage.PruneTickets()
if err != nil {
logger.Printf("Failed to prune tickets: %s", err)
}
logger.Printf("Pruned %d tickets", pruned)
time.Sleep(interval)*/
}
}
}

View file

@ -1,6 +1,51 @@
package devices package devices
import (
"log"
"time"
)
type Device interface { type Device interface {
Status() (bool, error) Status() (bool, error)
PushPowerButton() error PushPowerButton() error
} }
func KeepDeviceInSync(logger *log.Logger, device Device, wantStatus func() (bool, error), exit <-chan bool) {
for {
select {
case <-exit:
break
default:
status, err := device.Status()
if err != nil {
logger.Printf("Failed to fetch device status, waiting 1 minute until next attempt: %s", err)
time.Sleep(1 * time.Minute)
continue
}
want, err := wantStatus()
if err != nil {
logger.Printf("Failed to fetch want status, waiting 1 minute until next attempt: %s", err)
time.Sleep(1 * time.Minute)
continue
}
if status == want {
// Don't lock up our single-threaded session storage too much
time.Sleep(2 * time.Second)
continue
}
err = device.PushPowerButton()
if err != nil {
logger.Printf("Failed to push power button, waiting 2 minutes until next attempt: %s", err)
// Cooldown to not overwhelm the device with start/stop requests
time.Sleep(2 * time.Minute)
return
}
logger.Printf("Pushed power button")
time.Sleep(2 * time.Minute)
}
}
}

View file

@ -8,6 +8,7 @@ import (
) )
// TODO: Do not use subprocesses here, they are unreliable, provide hidden dependencies and are a common target for vulnerabilities. // TODO: Do not use subprocesses here, they are unreliable, provide hidden dependencies and are a common target for vulnerabilities.
// TODO: Improve pulldown, pullup
func NewLibgpiod(gpiochip string, powerButtonPin int, powerLEDPin int) Device { func NewLibgpiod(gpiochip string, powerButtonPin int, powerLEDPin int) Device {
device := &libgpiod{ device := &libgpiod{
gpiochip: gpiochip, gpiochip: gpiochip,

View file

@ -5,8 +5,8 @@ import (
"html/template" "html/template"
"log" "log"
"net/http" "net/http"
"time"
"git.1e99.eu/1e99/wolbodge/devices"
"git.1e99.eu/1e99/wolbodge/routes" "git.1e99.eu/1e99/wolbodge/routes"
_ "embed" _ "embed"
@ -46,8 +46,7 @@ func Run() error {
mux.Handle("DELETE /api/session/{session_id}", routes.APIEndSession(storage)) mux.Handle("DELETE /api/session/{session_id}", routes.APIEndSession(storage))
exit := make(chan bool) exit := make(chan bool)
go PruneStorage(logger, storage, 10*time.Second, exit) go devices.KeepDeviceInSync(logger, device, storage.HasSessions, exit)
go DeviceCheck(logger, device, storage.HasSessions, exit)
logger.Printf("Listening on %s", address) logger.Printf("Listening on %s", address)
err = http.ListenAndServe(address, mux) err = http.ListenAndServe(address, mux)