Compare commits
5 Commits
7df46e03e0
...
5d7180cd94
Author | SHA1 | Date | |
---|---|---|---|
5d7180cd94 | |||
05c87ac7b3 | |||
d202cdfee8 | |||
e1f5fbcd6c | |||
1def1ff67a |
@ -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)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
@ -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
|
||||||
|
@ -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
|
||||||
|
@ -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
4
app.go
@ -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
|
||||||
|
@ -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
|
||||||
|
}
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
|
@ -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
|
||||||
|
}
|
||||||
|
@ -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,29 +52,48 @@ 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,
|
||||||
claironTime: settings.GongInterval * time.Minute,
|
weatherTime: settings.WeatherDelay * time.Minute,
|
||||||
claironFile: reveil.CurrentGongPath(cfg),
|
weatherAction: wact,
|
||||||
reverseOrder: int(time.Now().Unix()/86400)%2 == 0,
|
claironTime: settings.GongInterval * time.Minute,
|
||||||
|
claironFile: reveil.CurrentGongPath(cfg),
|
||||||
|
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
|
||||||
@ -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() {
|
||||||
|
@ -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
|
||||||
|
@ -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();
|
||||||
|
@ -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}
|
||||||
|
@ -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',
|
||||||
|
@ -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}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user