Compare commits

...

5 Commits

Author SHA1 Message Date
5d7180cd94 Can delete routine
All checks were successful
continuous-integration/drone/push Build is passing
continuous-integration/drone/tag Build is passing
2022-12-08 20:15:10 +01:00
05c87ac7b3 New route to launch action individually 2022-12-08 20:15:00 +01:00
d202cdfee8 New route to test routine
All checks were successful
continuous-integration/drone/tag Build is passing
2022-12-08 19:33:51 +01:00
e1f5fbcd6c Start routine at wakeup end 2022-12-08 19:33:34 +01:00
1def1ff67a Weather action 2022-12-08 19:33:16 +01:00
14 changed files with 276 additions and 24 deletions

View File

@ -2,6 +2,7 @@ package api
import ( import (
"fmt" "fmt"
"log"
"net/http" "net/http"
"github.com/gin-gonic/gin" "github.com/gin-gonic/gin"
@ -90,4 +91,23 @@ func declareActionsRoutes(cfg *config.Config, router *gin.RouterGroup) {
c.JSON(http.StatusOK, nil) c.JSON(http.StatusOK, nil)
}) })
actionsRoutes.POST("/run", func(c *gin.Context) {
action := c.MustGet("action").(*reveil.Action)
cmd, err := action.Launch()
if err != nil {
c.AbortWithStatusJSON(http.StatusInternalServerError, gin.H{"errmsg": fmt.Sprintf("Unable to run the action: %s", err.Error())})
return
}
go func() {
err := cmd.Wait()
if err != nil {
log.Printf("%q: %s", action.Name, err.Error())
}
}()
c.JSON(http.StatusOK, true)
})
} }

View File

@ -20,7 +20,7 @@ func declareAlarmRoutes(cfg *config.Config, router *gin.RouterGroup) {
router.POST("/alarm/run", func(c *gin.Context) { router.POST("/alarm/run", func(c *gin.Context) {
if player.CommonPlayer == nil { if player.CommonPlayer == nil {
err := player.WakeUp(cfg) err := player.WakeUp(cfg, nil)
if err != nil { if err != nil {
c.AbortWithStatusJSON(http.StatusInternalServerError, gin.H{"errmsg": err.Error()}) c.AbortWithStatusJSON(http.StatusInternalServerError, gin.H{"errmsg": err.Error()})
return return

View File

@ -13,7 +13,7 @@ import (
func declareAlarmsRoutes(cfg *config.Config, db *reveil.LevelDBStorage, resetTimer func(), router *gin.RouterGroup) { func declareAlarmsRoutes(cfg *config.Config, db *reveil.LevelDBStorage, resetTimer func(), router *gin.RouterGroup) {
router.GET("/alarms/next", func(c *gin.Context) { router.GET("/alarms/next", func(c *gin.Context) {
alarm, err := reveil.GetNextAlarm(cfg, db) alarm, _, err := reveil.GetNextAlarm(cfg, db)
if err != nil { if err != nil {
c.AbortWithStatusJSON(http.StatusInternalServerError, gin.H{"errmsg": err.Error()}) c.AbortWithStatusJSON(http.StatusInternalServerError, gin.H{"errmsg": err.Error()})
return return

View File

@ -78,4 +78,12 @@ func declareRoutinesRoutes(cfg *config.Config, router *gin.RouterGroup) {
c.JSON(http.StatusOK, nil) c.JSON(http.StatusOK, nil)
}) })
routinesRoutes.POST("/run", func(c *gin.Context) {
routine := c.MustGet("routine").(*reveil.Routine)
go routine.Launch(cfg)
c.JSON(http.StatusOK, true)
})
} }

4
app.go
View File

@ -80,11 +80,11 @@ func (app *App) ResetTimer() {
app.nextAlarm = nil app.nextAlarm = nil
} }
if na, err := reveil.GetNextAlarm(app.cfg, app.db); err == nil && na != nil { if na, routines, err := reveil.GetNextAlarm(app.cfg, app.db); err == nil && na != nil {
app.nextAlarm = time.AfterFunc(time.Until(*na), func() { app.nextAlarm = time.AfterFunc(time.Until(*na), func() {
app.nextAlarm = nil app.nextAlarm = nil
reveil.RemoveOldAlarmsSingle(app.db) reveil.RemoveOldAlarmsSingle(app.db)
err := player.WakeUp(app.cfg) err := player.WakeUp(app.cfg, routines)
if err != nil { if err != nil {
log.Println(err.Error()) log.Println(err.Error())
return return

View File

@ -4,9 +4,11 @@ import (
"bufio" "bufio"
"crypto/sha512" "crypto/sha512"
"errors" "errors"
"fmt"
"io/fs" "io/fs"
"log" "log"
"os" "os"
"os/exec"
"path/filepath" "path/filepath"
"strings" "strings"
@ -19,9 +21,10 @@ type Action struct {
Description string `json:"description,omitempty"` Description string `json:"description,omitempty"`
Path string `json:"path"` Path string `json:"path"`
Enabled bool `json:"enabled"` Enabled bool `json:"enabled"`
fullPath string
} }
func LoadAction(path string) (string, string, error) { func loadAction(path string) (string, string, error) {
fd, err := os.Open(path) fd, err := os.Open(path)
if err != nil { if err != nil {
return "", "", err return "", "", err
@ -64,6 +67,45 @@ func LoadAction(path string) (string, string, error) {
return name, description, nil return name, description, nil
} }
func LoadAction(cfg *config.Config, path string) (*Action, error) {
actionsDir, err := filepath.Abs(cfg.ActionsDir)
if err != nil {
return nil, err
}
path = filepath.Join(actionsDir, path)
d, err := os.Stat(path)
if err != nil {
return nil, err
}
if !d.Mode().IsRegular() {
return nil, fmt.Errorf("%q is not a file, it cannot be an action.", path)
}
hash := sha512.Sum512([]byte(path))
// Parse content
name, description, err := loadAction(path)
if err != nil {
return nil, fmt.Errorf("Invalid action file (trying to parse %s): %s", path, err.Error())
}
if apath, err := filepath.Abs(path); err == nil {
path = apath
}
return &Action{
Id: hash[:],
Name: name,
Description: description,
Path: strings.TrimPrefix(path, actionsDir+"/"),
Enabled: d.Mode().Perm()&0111 != 0,
fullPath: path,
}, nil
}
func LoadActions(cfg *config.Config) (actions []*Action, err error) { func LoadActions(cfg *config.Config) (actions []*Action, err error) {
actionsDir, err := filepath.Abs(cfg.ActionsDir) actionsDir, err := filepath.Abs(cfg.ActionsDir)
if err != nil { if err != nil {
@ -75,7 +117,7 @@ func LoadActions(cfg *config.Config) (actions []*Action, err error) {
hash := sha512.Sum512([]byte(path)) hash := sha512.Sum512([]byte(path))
// Parse content // Parse content
name, description, err := LoadAction(path) name, description, err := loadAction(path)
if err != nil { if err != nil {
log.Printf("Invalid action file (trying to parse %s): %s", path, err.Error()) log.Printf("Invalid action file (trying to parse %s): %s", path, err.Error())
// Ignore invalid files // Ignore invalid files
@ -96,6 +138,7 @@ func LoadActions(cfg *config.Config) (actions []*Action, err error) {
Description: description, Description: description,
Path: strings.TrimPrefix(path, actionsDir+"/"), Path: strings.TrimPrefix(path, actionsDir+"/"),
Enabled: d.Mode().Perm()&0111 != 0, Enabled: d.Mode().Perm()&0111 != 0,
fullPath: path,
}) })
} }
@ -130,3 +173,9 @@ func (a *Action) Disable() error {
func (a *Action) Remove() error { func (a *Action) Remove() error {
return os.Remove(a.Path) return os.Remove(a.Path)
} }
func (a *Action) Launch() (cmd *exec.Cmd, err error) {
cmd = exec.Command(a.fullPath)
err = cmd.Start()
return
}

View File

@ -40,37 +40,57 @@ func (h *Hour) UnmarshalJSON(src []byte) error {
return nil return nil
} }
func GetNextAlarm(cfg *config.Config, db *LevelDBStorage) (*time.Time, error) { func GetNextAlarm(cfg *config.Config, db *LevelDBStorage) (*time.Time, []Identifier, error) {
alarmsRepeated, err := GetAlarmsRepeated(db) alarmsRepeated, err := GetAlarmsRepeated(db)
if err != nil { if err != nil {
return nil, err return nil, nil, err
} }
var closestAlarm *time.Time var closestAlarm *time.Time
var closestAlarmRoutines []Identifier
for _, alarm := range alarmsRepeated { for _, alarm := range alarmsRepeated {
next := alarm.GetNextOccurence(cfg, db) next := alarm.GetNextOccurence(cfg, db)
if next != nil && (closestAlarm == nil || closestAlarm.After(*next)) { if next != nil && (closestAlarm == nil || closestAlarm.After(*next)) {
closestAlarm = next closestAlarm = next
closestAlarmRoutines = alarm.FollowingRoutines
} }
} }
alarmsSingle, err := GetAlarmsSingle(db) alarmsSingle, err := GetAlarmsSingle(db)
if err != nil { if err != nil {
return nil, err return nil, nil, err
} }
now := time.Now() now := time.Now()
for _, alarm := range alarmsSingle { for _, alarm := range alarmsSingle {
if closestAlarm == nil || (closestAlarm.After(alarm.Time) && alarm.Time.After(now)) { if closestAlarm == nil || (closestAlarm.After(alarm.Time) && alarm.Time.After(now)) {
closestAlarm = &alarm.Time closestAlarm = &alarm.Time
closestAlarmRoutines = alarm.FollowingRoutines
} }
} }
return closestAlarm, nil return closestAlarm, closestAlarmRoutines, nil
}
func GetNextException(cfg *config.Config, db *LevelDBStorage) (*time.Time, error) {
alarmsExceptions, err := GetAlarmExceptions(db)
if err != nil {
return nil, err
}
var closestException *time.Time
for _, except := range alarmsExceptions {
if except != nil && time.Time(*except.End).After(time.Now()) && (closestException == nil || closestException.After(time.Time(*except.Start))) {
tmp := time.Time(*except.Start)
closestException = &tmp
}
}
return closestException, nil
} }
func DropNextAlarm(cfg *config.Config, db *LevelDBStorage) error { func DropNextAlarm(cfg *config.Config, db *LevelDBStorage) error {
timenext, err := GetNextAlarm(cfg, db) timenext, _, err := GetNextAlarm(cfg, db)
if err != nil { if err != nil {
return err return err
} }

View File

@ -1,7 +1,9 @@
package reveil package reveil
import ( import (
"bytes"
"crypto/sha512" "crypto/sha512"
"fmt"
"io/fs" "io/fs"
"io/ioutil" "io/ioutil"
"log" "log"
@ -9,6 +11,7 @@ import (
"path/filepath" "path/filepath"
"strconv" "strconv"
"strings" "strings"
"time"
"git.nemunai.re/nemunaire/reveil/config" "git.nemunai.re/nemunaire/reveil/config"
) )
@ -19,6 +22,10 @@ type RoutineStep struct {
Args []string `json:"args,omitempty"` Args []string `json:"args,omitempty"`
} }
func (s *RoutineStep) GetAction(cfg *config.Config) (*Action, error) {
return LoadAction(cfg, s.Action)
}
type Routine struct { type Routine struct {
Id Identifier `json:"id"` Id Identifier `json:"id"`
Name string `json:"name"` Name string `json:"name"`
@ -72,6 +79,21 @@ func LoadRoutine(path string, cfg *config.Config) ([]RoutineStep, error) {
return steps, nil return steps, nil
} }
func LoadRoutineFromId(id Identifier, cfg *config.Config) (*Routine, error) {
routines, err := LoadRoutines(cfg)
if err != nil {
return nil, err
}
for _, routine := range routines {
if bytes.Equal(routine.Id, id) {
return routine, nil
}
}
return nil, fmt.Errorf("Unable to find routine %x", id)
}
func LoadRoutines(cfg *config.Config) (routines []*Routine, err error) { func LoadRoutines(cfg *config.Config) (routines []*Routine, err error) {
err = filepath.Walk(cfg.RoutinesDir, func(path string, d fs.FileInfo, err error) error { err = filepath.Walk(cfg.RoutinesDir, func(path string, d fs.FileInfo, err error) error {
if d.IsDir() && path != cfg.RoutinesDir { if d.IsDir() && path != cfg.RoutinesDir {
@ -117,3 +139,28 @@ func (r *Routine) Rename(newName string) error {
func (a *Routine) Remove() error { func (a *Routine) Remove() error {
return os.Remove(a.Path) return os.Remove(a.Path)
} }
func (a *Routine) Launch(cfg *config.Config) error {
for _, s := range a.Steps {
act, err := s.GetAction(cfg)
if err != nil {
log.Printf("Unable to get action: %s: %s", s.Action, err.Error())
continue
}
time.Sleep(time.Duration(s.Delay) * time.Second)
cmd, err := act.Launch()
if err != nil {
log.Printf("Unable to launch the action %q: %s", s.Action, err.Error())
continue
}
err = cmd.Wait()
if err != nil {
log.Printf("Something goes wrong when waiting for the action %q's end: %s", s.Action, err.Error())
}
}
return nil
}

View File

@ -25,11 +25,17 @@ type Player struct {
currentCmd *exec.Cmd currentCmd *exec.Cmd
currentCmdCh chan bool currentCmdCh chan bool
weatherTime time.Duration
weatherAction *reveil.Action
claironTime time.Duration claironTime time.Duration
claironFile string claironFile string
endRoutines []*reveil.Routine
ntick int64 ntick int64
hasClaironed bool hasClaironed bool
hasSpokeWeather bool
launched time.Time launched time.Time
volume uint16 volume uint16
dontUpdateVolume bool dontUpdateVolume bool
@ -37,7 +43,7 @@ type Player struct {
playedItem int playedItem int
} }
func WakeUp(cfg *config.Config) (err error) { func WakeUp(cfg *config.Config, routine []reveil.Identifier) (err error) {
if CommonPlayer != nil { if CommonPlayer != nil {
return fmt.Errorf("Unable to start the player: a player is already running") return fmt.Errorf("Unable to start the player: a player is already running")
} }
@ -46,31 +52,50 @@ func WakeUp(cfg *config.Config) (err error) {
seed -= seed % 172800 seed -= seed % 172800
rand.Seed(seed) rand.Seed(seed)
CommonPlayer, err = NewPlayer(cfg) CommonPlayer, err = NewPlayer(cfg, routine)
if err != nil { if err != nil {
return err return err
} }
go CommonPlayer.WakeUp() go CommonPlayer.WakeUp(cfg)
return nil return nil
} }
func NewPlayer(cfg *config.Config) (*Player, error) { func NewPlayer(cfg *config.Config, routines []reveil.Identifier) (*Player, error) {
// Load our settings // Load our settings
settings, err := reveil.ReadSettings(cfg.SettingsFile) settings, err := reveil.ReadSettings(cfg.SettingsFile)
if err != nil { if err != nil {
return nil, fmt.Errorf("Unable to read settings: %w", err) return nil, fmt.Errorf("Unable to read settings: %w", err)
} }
// Load weather action
wact, err := reveil.LoadAction(cfg, settings.WeatherAction)
if err != nil {
log.Println("Unable to load weather action:", err.Error())
}
p := Player{ p := Player{
Stopper: make(chan bool, 1), Stopper: make(chan bool, 1),
currentCmdCh: make(chan bool, 1), currentCmdCh: make(chan bool, 1),
MaxRunTime: settings.MaxRunTime * time.Minute, MaxRunTime: settings.MaxRunTime * time.Minute,
weatherTime: settings.WeatherDelay * time.Minute,
weatherAction: wact,
claironTime: settings.GongInterval * time.Minute, claironTime: settings.GongInterval * time.Minute,
claironFile: reveil.CurrentGongPath(cfg), claironFile: reveil.CurrentGongPath(cfg),
reverseOrder: int(time.Now().Unix()/86400)%2 == 0, reverseOrder: int(time.Now().Unix()/86400)%2 == 0,
} }
// Load routines
for _, routine := range routines {
r, err := reveil.LoadRoutineFromId(routine, cfg)
if err != nil {
log.Printf("Unable to load routine %x: %s", routine, err.Error())
continue
}
p.endRoutines = append(p.endRoutines, r)
}
// Load our track list // Load our track list
tracks, err := reveil.LoadTracks(cfg) tracks, err := reveil.LoadTracks(cfg)
if err != nil { if err != nil {
@ -98,6 +123,16 @@ func NewPlayer(cfg *config.Config) (*Player, error) {
return &p, nil return &p, nil
} }
func (p *Player) launchAction(a *reveil.Action) (err error) {
p.currentCmd, err = a.Launch()
log.Println("Running action ", a.Name)
err = p.currentCmd.Wait()
p.currentCmdCh <- true
return
}
func (p *Player) playFile(filepath string) (err error) { func (p *Player) playFile(filepath string) (err error) {
p.currentCmd = exec.Command("paplay", filepath) p.currentCmd = exec.Command("paplay", filepath)
if err = p.currentCmd.Start(); err != nil { if err = p.currentCmd.Start(); err != nil {
@ -114,7 +149,7 @@ func (p *Player) playFile(filepath string) (err error) {
return return
} }
func (p *Player) WakeUp() { func (p *Player) WakeUp(cfg *config.Config) {
log.Println("Playlist in use:", strings.Join(p.Playlist, " ; ")) log.Println("Playlist in use:", strings.Join(p.Playlist, " ; "))
// Prepare sound player // Prepare sound player
@ -141,6 +176,11 @@ loop:
p.SetVolume(65535) p.SetVolume(65535)
p.dontUpdateVolume = true p.dontUpdateVolume = true
go p.playFile(p.claironFile) go p.playFile(p.claironFile)
} else if p.weatherAction != nil && !p.hasSpokeWeather && time.Since(p.launched) >= p.weatherTime {
log.Println("weather time!")
p.SetVolume(65535)
p.dontUpdateVolume = true
go p.launchAction(p.weatherAction)
} else { } else {
p.dontUpdateVolume = false p.dontUpdateVolume = false
p.volume = 3500 + uint16(math.Log(1+float64(p.ntick)/8)*9500) p.volume = 3500 + uint16(math.Log(1+float64(p.ntick)/8)*9500)
@ -214,6 +254,11 @@ loopcalm:
CommonPlayer = nil CommonPlayer = nil
} }
// TODO: Start Routine if any
for _, r := range p.endRoutines {
go r.Launch(cfg)
}
} }
func (p *Player) NextTrack() { func (p *Player) NextTrack() {

View File

@ -24,7 +24,7 @@ func DeclareNoJSRoutes(router *gin.Engine, cfg *config.Config, db *reveil.LevelD
router.SetHTMLTemplate(templ) router.SetHTMLTemplate(templ)
router.GET("/nojs.html", func(c *gin.Context) { router.GET("/nojs.html", func(c *gin.Context) {
alarm, err := reveil.GetNextAlarm(cfg, db) alarm, _, err := reveil.GetNextAlarm(cfg, db)
if err != nil { if err != nil {
c.HTML(http.StatusInternalServerError, "error.tmpl", gin.H{"errmsg": err.Error()}) c.HTML(http.StatusInternalServerError, "error.tmpl", gin.H{"errmsg": err.Error()})
return return
@ -109,7 +109,7 @@ func DeclareNoJSRoutes(router *gin.Engine, cfg *config.Config, db *reveil.LevelD
case "start": case "start":
if player.CommonPlayer == nil { if player.CommonPlayer == nil {
err := player.WakeUp(cfg) err := player.WakeUp(cfg, nil)
if err != nil { if err != nil {
c.HTML(http.StatusInternalServerError, "error.tmpl", gin.H{"errmsg": err.Error()}) c.HTML(http.StatusInternalServerError, "error.tmpl", gin.H{"errmsg": err.Error()})
return return

View File

@ -25,6 +25,18 @@ export class Action {
} }
} }
async launch() {
const res = await fetch(`api/actions/${this.id}/run`, {
method: 'POST',
headers: {'Accept': 'application/json'}
});
if (res.status == 200) {
return true;
} else {
throw new Error((await res.json()).errmsg);
}
}
toggleEnable() { toggleEnable() {
this.enabled = !this.enabled; this.enabled = !this.enabled;
this.save(); this.save();

View File

@ -27,6 +27,7 @@
color="outline-danger" color="outline-danger"
size="sm" size="sm"
class="float-end ms-1" class="float-end ms-1"
on:click={() => routine.delete()}
> >
<Icon name="trash" /> <Icon name="trash" />
</Button> </Button>
@ -37,6 +38,14 @@
> >
<Icon name="pencil" /> <Icon name="pencil" />
</Button> </Button>
<Button
color="outline-success"
size="sm"
class="float-end ms-1"
on:click={() => routine.launch()}
>
<Icon name="play-fill" />
</Button>
{routine.name} {routine.name}
</CardHeader> </CardHeader>
{#if routine.steps} {#if routine.steps}

View File

@ -26,6 +26,18 @@ export class Routine {
} }
} }
async launch() {
const res = await fetch(`api/routines/${this.id}/run`, {
method: 'POST',
headers: {'Accept': 'application/json'}
});
if (res.status == 200) {
return true;
} else {
throw new Error((await res.json()).errmsg);
}
}
async save() { async save() {
const res = await fetch(this.id?`api/routines/${this.id}`:'api/routines', { const res = await fetch(this.id?`api/routines/${this.id}`:'api/routines', {
method: this.id?'PUT':'POST', method: this.id?'PUT':'POST',

View File

@ -3,6 +3,7 @@
import { import {
Container, Container,
Icon,
Input, Input,
ListGroup, ListGroup,
ListGroupItem, ListGroupItem,
@ -10,6 +11,14 @@
} from 'sveltestrap'; } from 'sveltestrap';
import { getAction } from '$lib/action'; import { getAction } from '$lib/action';
import { actions } from '$lib/stores/actions';
function deleteThis(action) {
action.delete().then(() => {
actions.refresh();
goto('routines/actions/');
})
}
</script> </script>
{#await getAction($page.params.aid)} {#await getAction($page.params.aid)}
@ -34,5 +43,26 @@
<Input type="switch" on:change={() => action.toggleEnable()} checked={action.enabled} /> <Input type="switch" on:change={() => action.toggleEnable()} checked={action.enabled} />
</ListGroupItem> </ListGroupItem>
</ListGroup> </ListGroup>
<ListGroup class="my-2 text-center">
<ListGroupItem
action
tag="button"
class="text-success fw-bold"
on:click={() => action.launch()}
>
<Icon name="play-fill" />
Lancer cette action
</ListGroupItem>
<ListGroupItem
action
tag="button"
class="text-danger fw-bold"
on:click={() => deleteThis(action)}
>
<Icon name="trash" />
Supprimer cette action
</ListGroupItem>
</ListGroup>
</Container> </Container>
{/await} {/await}