rework device interface, remove libgpiod, add command device
This commit is contained in:
parent
e660d5937a
commit
2e0e50dca5
6 changed files with 125 additions and 118 deletions
|
@ -7,7 +7,13 @@ Configuration is done using environment variables.
|
||||||
- `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`.
|
- `command`: Runs a subprocess to aquire the info. See the example below for more info.
|
||||||
|
|
||||||
|
### Example: Raspberry pi using the command device
|
||||||
|
- `WOLBODGE_DEVICE_TYPE=command`
|
||||||
|
- `WOLBODGE_DEVICE_SHELL=bash -c $COMMAND`: Configure which shell we should use.
|
||||||
|
- `WOLBODGE_DEVICE_STATUS_COMMAND=gpioget --bias pull-down gpiochip0 17 > $WOLBODGE_STATUS_FILE`: Which command should be used to aquire the device's status. It should echo out a "0" for off and a "1" for on into the file created at "$WOLBODGE_STATUS_FILE".
|
||||||
|
- `WOLBODGE_DEVICE_TOGGLE_STATUS_COMMAND=gpioset gpiochip0 27=1 && sleep 2 && gpioset gpiochip0 27=0`: Which command should be used to toggle the device's status. This usually involves pressing the power button virtually.
|
||||||
|
|
||||||
## Usage example
|
## Usage example
|
||||||
Usage example for a backup script that backs up to a NAS that isn't online 24/7.
|
Usage example for a backup script that backs up to a NAS that isn't online 24/7.
|
||||||
|
@ -39,4 +45,3 @@ fi
|
||||||
|
|
||||||
## TODOs
|
## TODOs
|
||||||
- Alerts when a session has existed for a long time
|
- Alerts when a session has existed for a long time
|
||||||
- Command device that can use commands to turn on/off a device
|
|
||||||
|
|
94
devices/command.go
Normal file
94
devices/command.go
Normal file
|
@ -0,0 +1,94 @@
|
||||||
|
package devices
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
)
|
||||||
|
|
||||||
|
// TODO: A command device
|
||||||
|
// You will be able to use shell commands to push the power button and get the status.
|
||||||
|
//
|
||||||
|
// Status will give an env variable: WOLBODGE_STATUS_FILE, you will be expected to write a "0" or a "1" there, depending on the state.
|
||||||
|
|
||||||
|
func NewCommand(shell []string, statusCmd string, toggleStatusCmd string) Device {
|
||||||
|
device := &command{
|
||||||
|
shell: shell,
|
||||||
|
statusCmd: statusCmd,
|
||||||
|
toggleStatusCmd: toggleStatusCmd,
|
||||||
|
lock: sync.Mutex{},
|
||||||
|
}
|
||||||
|
return device
|
||||||
|
}
|
||||||
|
|
||||||
|
type command struct {
|
||||||
|
shell []string
|
||||||
|
statusCmd string
|
||||||
|
toggleStatusCmd string
|
||||||
|
lock sync.Mutex
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *command) Status() (bool, error) {
|
||||||
|
d.lock.Lock()
|
||||||
|
defer d.lock.Unlock()
|
||||||
|
|
||||||
|
file, err := os.CreateTemp("", "wolbodge-status-*")
|
||||||
|
if err != nil {
|
||||||
|
return false, fmt.Errorf("failed to create temporary status file: %w", err)
|
||||||
|
}
|
||||||
|
defer os.Remove(file.Name())
|
||||||
|
|
||||||
|
cmd := d.makeCommand(d.statusCmd)
|
||||||
|
cmd.Env = append(
|
||||||
|
cmd.Environ(),
|
||||||
|
fmt.Sprintf("WOLBODGE_STATUS_FILE=%s", file.Name()),
|
||||||
|
)
|
||||||
|
|
||||||
|
err = cmd.Run()
|
||||||
|
if err != nil {
|
||||||
|
return false, fmt.Errorf("failed to run command: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := os.ReadFile(file.Name())
|
||||||
|
if err != nil {
|
||||||
|
return false, fmt.Errorf("failed to read status file: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
content := string(data)
|
||||||
|
content = strings.TrimSpace(content)
|
||||||
|
|
||||||
|
switch content {
|
||||||
|
case "0":
|
||||||
|
return false, nil
|
||||||
|
case "1":
|
||||||
|
return true, nil
|
||||||
|
default:
|
||||||
|
return false, fmt.Errorf("invalid status file content, expected \"0\" or \"1\", got \"%s\"", content)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *command) ToggleStatus() error {
|
||||||
|
d.lock.Lock()
|
||||||
|
defer d.lock.Unlock()
|
||||||
|
|
||||||
|
cmd := d.makeCommand(d.toggleStatusCmd)
|
||||||
|
err := cmd.Run()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to run command: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *command) makeCommand(command string) *exec.Cmd {
|
||||||
|
args := make([]string, len(d.shell))
|
||||||
|
copy(args, d.shell)
|
||||||
|
|
||||||
|
for i := range args {
|
||||||
|
args[i] = strings.ReplaceAll(args[i], "$COMMAND", command)
|
||||||
|
}
|
||||||
|
|
||||||
|
return exec.Command(args[0], args[1:]...)
|
||||||
|
}
|
|
@ -6,8 +6,13 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
type Device interface {
|
type Device interface {
|
||||||
|
// Retreive the status of the device.
|
||||||
|
// This is usually acheived by taking a reading from the power LED.
|
||||||
Status() (bool, error)
|
Status() (bool, error)
|
||||||
PushPowerButton() error
|
|
||||||
|
// Toggle the status of the device.
|
||||||
|
// This is usually achieved by pressing the power button.
|
||||||
|
ToggleStatus() error
|
||||||
}
|
}
|
||||||
|
|
||||||
func KeepDeviceInSync(logger *log.Logger, device Device, wantStatus func() (bool, error), exit <-chan bool) {
|
func KeepDeviceInSync(logger *log.Logger, device Device, wantStatus func() (bool, error), exit <-chan bool) {
|
||||||
|
@ -36,15 +41,15 @@ func KeepDeviceInSync(logger *log.Logger, device Device, wantStatus func() (bool
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
err = device.PushPowerButton()
|
err = device.ToggleStatus()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Printf("Failed to push power button, waiting 2 minutes until next attempt: %s", err)
|
logger.Printf("Failed to toggle device status, waiting 2 minutes until next attempt: %s", err)
|
||||||
// Cooldown to not overwhelm the device with start/stop requests
|
// Cooldown to not overwhelm the device with start/stop requests
|
||||||
time.Sleep(2 * time.Minute)
|
time.Sleep(2 * time.Minute)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.Printf("Pushed power button")
|
logger.Printf("Toggled device status")
|
||||||
time.Sleep(2 * time.Minute)
|
time.Sleep(2 * time.Minute)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,101 +0,0 @@
|
||||||
package devices
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"os/exec"
|
|
||||||
"sync"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
// 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 {
|
|
||||||
device := &libgpiod{
|
|
||||||
gpiochip: gpiochip,
|
|
||||||
powerButtonPin: powerButtonPin,
|
|
||||||
powerLEDPin: powerLEDPin,
|
|
||||||
lock: sync.Mutex{},
|
|
||||||
}
|
|
||||||
|
|
||||||
return device
|
|
||||||
}
|
|
||||||
|
|
||||||
type libgpiod struct {
|
|
||||||
gpiochip string
|
|
||||||
powerButtonPin int
|
|
||||||
powerLEDPin int
|
|
||||||
lock sync.Mutex
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *libgpiod) Status() (bool, error) {
|
|
||||||
d.lock.Lock()
|
|
||||||
defer d.lock.Unlock()
|
|
||||||
|
|
||||||
status, err := d.read(d.powerLEDPin)
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return status, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *libgpiod) PushPowerButton() error {
|
|
||||||
d.lock.Lock()
|
|
||||||
defer d.lock.Unlock()
|
|
||||||
|
|
||||||
err := d.write(d.powerButtonPin, 1)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
time.Sleep(2 * time.Second)
|
|
||||||
|
|
||||||
err = d.write(d.powerButtonPin, 0)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *libgpiod) write(pin int, value int) error {
|
|
||||||
cmd := exec.Command(
|
|
||||||
"env", "sh", "-c",
|
|
||||||
fmt.Sprintf("gpioset %s %d=%d", d.gpiochip, pin, value),
|
|
||||||
)
|
|
||||||
|
|
||||||
outRaw, err := cmd.Output()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
out := string(outRaw)
|
|
||||||
if out == "" {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
return fmt.Errorf("unexpected command output: %s", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *libgpiod) read(pin int) (bool, error) {
|
|
||||||
cmd := exec.Command(
|
|
||||||
"env", "sh", "-c",
|
|
||||||
fmt.Sprintf("gpioget --bias pull-down %s %d", d.gpiochip, pin),
|
|
||||||
)
|
|
||||||
|
|
||||||
outRaw, err := cmd.Output()
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
|
|
||||||
out := string(outRaw)
|
|
||||||
|
|
||||||
switch out {
|
|
||||||
case "0\n":
|
|
||||||
return false, nil
|
|
||||||
case "1\n":
|
|
||||||
return true, nil
|
|
||||||
default:
|
|
||||||
return false, fmt.Errorf("unexpected command output: %s", out)
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -2,10 +2,10 @@ package devices
|
||||||
|
|
||||||
import "sync"
|
import "sync"
|
||||||
|
|
||||||
func NewTest(initialStatus bool) Device {
|
func NewTest(status bool) Device {
|
||||||
device := &test{
|
device := &test{
|
||||||
lock: sync.Mutex{},
|
lock: sync.Mutex{},
|
||||||
status: initialStatus,
|
status: status,
|
||||||
}
|
}
|
||||||
return device
|
return device
|
||||||
}
|
}
|
||||||
|
@ -22,7 +22,7 @@ func (d *test) Status() (bool, error) {
|
||||||
return d.status, nil
|
return d.status, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *test) PushPowerButton() error {
|
func (d *test) ToggleStatus() error {
|
||||||
d.lock.Lock()
|
d.lock.Lock()
|
||||||
defer d.lock.Unlock()
|
defer d.lock.Unlock()
|
||||||
|
|
||||||
|
|
20
env.go
20
env.go
|
@ -5,6 +5,7 @@ import (
|
||||||
"log"
|
"log"
|
||||||
"os"
|
"os"
|
||||||
"strconv"
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
"git.1e99.eu/1e99/wolbodge/devices"
|
"git.1e99.eu/1e99/wolbodge/devices"
|
||||||
"git.1e99.eu/1e99/wolbodge/sessions"
|
"git.1e99.eu/1e99/wolbodge/sessions"
|
||||||
|
@ -31,15 +32,15 @@ func NewDevice() (devices.Device, error) {
|
||||||
case "test":
|
case "test":
|
||||||
return devices.NewTest(false), nil
|
return devices.NewTest(false), nil
|
||||||
|
|
||||||
case "libgpiod":
|
case "command":
|
||||||
var gpiochip string
|
var shell []string
|
||||||
var powerButtonPin int
|
var statusCmd string
|
||||||
var powerLEDPin int
|
var toggleStatusCmd string
|
||||||
Env("WOLBODGE_DEVICE_GPIOCHIP", &gpiochip, "gpiochip0")
|
Env("WOLBODGE_DEVICE_SHELL", &shell, "")
|
||||||
Env("WOLBODGE_DEVICE_POWER_BUTTON_PIN", &powerButtonPin, "27")
|
Env("WOLBODGE_DEVICE_STATUS_COMMAND", &statusCmd, "")
|
||||||
Env("WOLBODGE_DEVICE_POWER_LED_PIN", &powerLEDPin, "17")
|
Env("WOLBODGE_DEVICE_TOGGLE_STATUS_COMMAND", &toggleStatusCmd, "")
|
||||||
|
|
||||||
return devices.NewLibgpiod(gpiochip, powerButtonPin, powerLEDPin), nil
|
return devices.NewCommand(shell, statusCmd, toggleStatusCmd), nil
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return nil, fmt.Errorf("unknown device type: %s", deviceType)
|
return nil, fmt.Errorf("unknown device type: %s", deviceType)
|
||||||
|
@ -73,5 +74,8 @@ func Env(name string, out any, def string) {
|
||||||
|
|
||||||
case *string:
|
case *string:
|
||||||
*value = raw
|
*value = raw
|
||||||
|
|
||||||
|
case *[]string:
|
||||||
|
*value = strings.Fields(raw)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue