Handle routines
This commit is contained in:
parent
b125a3cd00
commit
392d0133f7
1
.gitignore
vendored
1
.gitignore
vendored
@ -1,6 +1,7 @@
|
|||||||
actions
|
actions
|
||||||
gongs
|
gongs
|
||||||
reveil
|
reveil
|
||||||
|
routines
|
||||||
tracks
|
tracks
|
||||||
vendor
|
vendor
|
||||||
ui/build
|
ui/build
|
||||||
|
@ -1,37 +1,81 @@
|
|||||||
package api
|
package api
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
"git.nemunai.re/nemunaire/reveil/config"
|
"git.nemunai.re/nemunaire/reveil/config"
|
||||||
|
"git.nemunai.re/nemunaire/reveil/model"
|
||||||
)
|
)
|
||||||
|
|
||||||
func declareRoutinesRoutes(cfg *config.Config, router *gin.RouterGroup) {
|
func declareRoutinesRoutes(cfg *config.Config, router *gin.RouterGroup) {
|
||||||
router.GET("/routines", func(c *gin.Context) {
|
router.GET("/routines", func(c *gin.Context) {
|
||||||
|
routines, err := reveil.LoadRoutines(cfg)
|
||||||
|
if err != nil {
|
||||||
|
c.AbortWithStatusJSON(http.StatusInternalServerError, gin.H{"errmsg": err.Error()})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
c.JSON(http.StatusOK, routines)
|
||||||
})
|
})
|
||||||
router.POST("/routines", func(c *gin.Context) {
|
router.POST("/routines", func(c *gin.Context) {
|
||||||
|
c.AbortWithStatusJSON(http.StatusNotImplemented, gin.H{"errmsg": "TODO"})
|
||||||
})
|
})
|
||||||
|
|
||||||
routinesRoutes := router.Group("/routines/:gid")
|
routinesRoutes := router.Group("/routines/:tid")
|
||||||
routinesRoutes.Use(routineHandler)
|
routinesRoutes.Use(func(c *gin.Context) {
|
||||||
|
routines, err := reveil.LoadRoutines(cfg)
|
||||||
|
if err != nil {
|
||||||
|
c.AbortWithStatusJSON(http.StatusInternalServerError, gin.H{"errmsg": err.Error()})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, t := range routines {
|
||||||
|
if t.Id.ToString() == c.Param("tid") {
|
||||||
|
c.Set("routine", t)
|
||||||
|
c.Next()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
c.AbortWithStatusJSON(http.StatusNotFound, gin.H{"errmsg": "Routine not found"})
|
||||||
|
})
|
||||||
|
|
||||||
routinesRoutes.GET("", func(c *gin.Context) {
|
routinesRoutes.GET("", func(c *gin.Context) {
|
||||||
c.JSON(http.StatusOK, c.MustGet("routine"))
|
c.JSON(http.StatusOK, c.MustGet("routine"))
|
||||||
})
|
})
|
||||||
routinesRoutes.PUT("", func(c *gin.Context) {
|
routinesRoutes.PUT("", func(c *gin.Context) {
|
||||||
c.JSON(http.StatusOK, c.MustGet("routine"))
|
oldroutine := c.MustGet("routine").(*reveil.Routine)
|
||||||
|
|
||||||
|
var routine reveil.Routine
|
||||||
|
if err := c.ShouldBindJSON(&routine); err != nil {
|
||||||
|
c.AbortWithStatusJSON(http.StatusBadRequest, gin.H{"errmsg": err.Error()})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if routine.Name != oldroutine.Name {
|
||||||
|
err := oldroutine.Rename(routine.Name)
|
||||||
|
if err != nil {
|
||||||
|
c.AbortWithStatusJSON(http.StatusInternalServerError, gin.H{"errmsg": fmt.Sprintf("Unable to rename the routine: %s", err.Error())})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: change actions
|
||||||
|
|
||||||
|
c.JSON(http.StatusOK, oldroutine)
|
||||||
})
|
})
|
||||||
routinesRoutes.DELETE("", func(c *gin.Context) {
|
routinesRoutes.DELETE("", func(c *gin.Context) {
|
||||||
c.JSON(http.StatusOK, c.MustGet("routine"))
|
routine := c.MustGet("routine").(*reveil.Routine)
|
||||||
|
|
||||||
|
err := routine.Remove()
|
||||||
|
if err != nil {
|
||||||
|
c.AbortWithStatusJSON(http.StatusInternalServerError, gin.H{"errmsg": fmt.Sprintf("Unable to remove the routine: %s", err.Error())})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
c.JSON(http.StatusOK, nil)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func routineHandler(c *gin.Context) {
|
|
||||||
c.Set("routine", nil)
|
|
||||||
|
|
||||||
c.Next()
|
|
||||||
}
|
|
||||||
|
119
model/routine.go
Normal file
119
model/routine.go
Normal file
@ -0,0 +1,119 @@
|
|||||||
|
package reveil
|
||||||
|
|
||||||
|
import (
|
||||||
|
"crypto/sha512"
|
||||||
|
"io/fs"
|
||||||
|
"io/ioutil"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"git.nemunai.re/nemunaire/reveil/config"
|
||||||
|
)
|
||||||
|
|
||||||
|
type RoutineStep struct {
|
||||||
|
Delay uint64 `json:"delay"`
|
||||||
|
Action string `json:"action"`
|
||||||
|
Args []string `json:"args,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Routine struct {
|
||||||
|
Id Identifier `json:"id"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Path string `json:"path"`
|
||||||
|
Steps []RoutineStep `json:"steps"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func LoadRoutine(path string, cfg *config.Config) ([]RoutineStep, error) {
|
||||||
|
ds, err := ioutil.ReadDir(path)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
actionsDir, err := filepath.Abs(cfg.ActionsDir)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var steps []RoutineStep
|
||||||
|
|
||||||
|
for _, f := range ds {
|
||||||
|
fullpath := filepath.Join(path, f.Name())
|
||||||
|
if f.Mode()&os.ModeSymlink == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
dst, err := os.Readlink(fullpath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if adst, err := filepath.Abs(filepath.Join(path, dst)); err == nil {
|
||||||
|
dst = adst
|
||||||
|
}
|
||||||
|
|
||||||
|
args := strings.Split(f.Name(), "_")
|
||||||
|
|
||||||
|
delay, err := strconv.ParseUint(args[0], 10, 64)
|
||||||
|
|
||||||
|
step := RoutineStep{
|
||||||
|
Delay: delay,
|
||||||
|
Action: strings.TrimPrefix(dst, actionsDir+"/"),
|
||||||
|
}
|
||||||
|
if len(args) > 1 {
|
||||||
|
step.Args = args[1:]
|
||||||
|
}
|
||||||
|
|
||||||
|
steps = append(steps, step)
|
||||||
|
}
|
||||||
|
|
||||||
|
return steps, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func LoadRoutines(cfg *config.Config) (routines []*Routine, err error) {
|
||||||
|
err = filepath.Walk(cfg.RoutinesDir, func(path string, d fs.FileInfo, err error) error {
|
||||||
|
if d.IsDir() && path != cfg.RoutinesDir {
|
||||||
|
hash := sha512.Sum512([]byte(path))
|
||||||
|
|
||||||
|
// Explore directory
|
||||||
|
steps, err := LoadRoutine(path, cfg)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("Invalid routine directory (trying to walk through %s): %s", path, err.Error())
|
||||||
|
// Ignore invalid routines
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
routines = append(routines, &Routine{
|
||||||
|
Id: hash[:],
|
||||||
|
Name: d.Name(),
|
||||||
|
Path: path,
|
||||||
|
Steps: steps,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Routine) Rename(newName string) error {
|
||||||
|
newPath := filepath.Join(filepath.Dir(r.Path), newName)
|
||||||
|
|
||||||
|
err := os.Rename(
|
||||||
|
r.Path,
|
||||||
|
newPath,
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
r.Path = newPath
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *Routine) Remove() error {
|
||||||
|
return os.Remove(a.Path)
|
||||||
|
}
|
@ -1,7 +1,9 @@
|
|||||||
<script>
|
<script>
|
||||||
import {
|
import {
|
||||||
|
Badge,
|
||||||
Button,
|
Button,
|
||||||
Card,
|
Card,
|
||||||
|
CardBody,
|
||||||
CardHeader,
|
CardHeader,
|
||||||
Col,
|
Col,
|
||||||
Container,
|
Container,
|
||||||
@ -11,25 +13,11 @@
|
|||||||
Icon,
|
Icon,
|
||||||
} from 'sveltestrap';
|
} from 'sveltestrap';
|
||||||
|
|
||||||
|
import { actions_idx } from '../stores/actions';
|
||||||
|
|
||||||
export let routine = {
|
export let routine = {
|
||||||
title: "Classique",
|
name: "Classique",
|
||||||
steps: [
|
steps: [],
|
||||||
{
|
|
||||||
id: 1,
|
|
||||||
name: "Salutation & heure",
|
|
||||||
start: 0,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 2,
|
|
||||||
name: "7 min workout",
|
|
||||||
start: 60,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 3,
|
|
||||||
name: "RATP traffic",
|
|
||||||
start: 540,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@ -49,11 +37,26 @@
|
|||||||
>
|
>
|
||||||
<Icon name="pencil" />
|
<Icon name="pencil" />
|
||||||
</Button>
|
</Button>
|
||||||
{routine.title}
|
{routine.name}
|
||||||
</CardHeader>
|
</CardHeader>
|
||||||
|
{#if routine.steps}
|
||||||
<ListGroup>
|
<ListGroup>
|
||||||
{#each routine.steps as step (step.id)}
|
{#each routine.steps as step}
|
||||||
<ListGroupItem action>{step.name}</ListGroupItem>
|
<ListGroupItem action>
|
||||||
|
{#if $actions_idx && $actions_idx[step.action]}
|
||||||
|
{$actions_idx[step.action].name}
|
||||||
|
{:else}
|
||||||
|
{step.action}
|
||||||
|
{/if}
|
||||||
|
<Badge class="float-end">
|
||||||
|
{step.delay/60} min
|
||||||
|
</Badge>
|
||||||
|
</ListGroupItem>
|
||||||
{/each}
|
{/each}
|
||||||
</ListGroup>
|
</ListGroup>
|
||||||
|
{:else}
|
||||||
|
<CardBody>
|
||||||
|
Aucune action définie.
|
||||||
|
</CardBody>
|
||||||
|
{/if}
|
||||||
</Card>
|
</Card>
|
||||||
|
61
ui/src/lib/routine.js
Normal file
61
ui/src/lib/routine.js
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
export class Routine {
|
||||||
|
constructor(res) {
|
||||||
|
if (res) {
|
||||||
|
this.update(res);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
update({ id, name, path, steps }) {
|
||||||
|
this.id = id;
|
||||||
|
this.name = name;
|
||||||
|
this.path = path;
|
||||||
|
|
||||||
|
steps.sort((a, b) => a.delay - b.delay);
|
||||||
|
this.steps = steps;
|
||||||
|
}
|
||||||
|
|
||||||
|
async delete() {
|
||||||
|
const res = await fetch(`api/routines/${this.id}`, {
|
||||||
|
method: 'DELETE',
|
||||||
|
headers: {'Accept': 'application/json'}
|
||||||
|
});
|
||||||
|
if (res.status == 200) {
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
throw new Error((await res.json()).errmsg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async save() {
|
||||||
|
const res = await fetch(this.id?`api/routines/${this.id}`:'api/routines', {
|
||||||
|
method: this.id?'PUT':'POST',
|
||||||
|
headers: {'Accept': 'application/json'},
|
||||||
|
body: JSON.stringify(this),
|
||||||
|
});
|
||||||
|
if (res.status == 200) {
|
||||||
|
const data = await res.json();
|
||||||
|
this.update(data);
|
||||||
|
return data;
|
||||||
|
} else {
|
||||||
|
throw new Error((await res.json()).errmsg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getRoutines() {
|
||||||
|
const res = await fetch(`api/routines`, {headers: {'Accept': 'application/json'}})
|
||||||
|
if (res.status == 200) {
|
||||||
|
return (await res.json()).map((r) => new Routine(r));
|
||||||
|
} else {
|
||||||
|
throw new Error((await res.json()).errmsg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getRoutine(rid) {
|
||||||
|
const res = await fetch(`api/routines/${rid}`, {headers: {'Accept': 'application/json'}})
|
||||||
|
if (res.status == 200) {
|
||||||
|
return new Routine(await res.json());
|
||||||
|
} else {
|
||||||
|
throw new Error((await res.json()).errmsg);
|
||||||
|
}
|
||||||
|
}
|
@ -3,10 +3,13 @@
|
|||||||
Card,
|
Card,
|
||||||
Col,
|
Col,
|
||||||
Container,
|
Container,
|
||||||
Row,
|
|
||||||
Icon,
|
Icon,
|
||||||
|
Row,
|
||||||
|
Spinner,
|
||||||
} from 'sveltestrap';
|
} from 'sveltestrap';
|
||||||
|
|
||||||
|
import { routines } from '../../stores/routines';
|
||||||
|
|
||||||
import CardRoutine from '../../components/CardRoutine.svelte';
|
import CardRoutine from '../../components/CardRoutine.svelte';
|
||||||
import ActionList from '../../components/ActionList.svelte';
|
import ActionList from '../../components/ActionList.svelte';
|
||||||
</script>
|
</script>
|
||||||
@ -16,7 +19,19 @@
|
|||||||
<Col md="8">
|
<Col md="8">
|
||||||
<Row cols={{xs: 1, lg: 2, xl: 3}}>
|
<Row cols={{xs: 1, lg: 2, xl: 3}}>
|
||||||
<Col class="mb-4">
|
<Col class="mb-4">
|
||||||
<CardRoutine />
|
{#if $routines.list}
|
||||||
|
{#each $routines.list as routine (routine.id)}
|
||||||
|
<CardRoutine {routine} />
|
||||||
|
{/each}
|
||||||
|
{:else}
|
||||||
|
{#await routines.refresh()}
|
||||||
|
<div class="d-flex justify-content-center align-items-center gap-2">
|
||||||
|
<Spinner color="primary" /> Chargement en cours…
|
||||||
|
</div>
|
||||||
|
{:then}
|
||||||
|
test
|
||||||
|
{/await}
|
||||||
|
{/if}
|
||||||
</Col>
|
</Col>
|
||||||
<Col class="mb-4">
|
<Col class="mb-4">
|
||||||
<Card
|
<Card
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
import { writable } from 'svelte/store';
|
import { derived, writable } from 'svelte/store';
|
||||||
|
|
||||||
import { getActions } from '../lib/action'
|
import { getActions } from '../lib/action'
|
||||||
|
|
||||||
function createActionsStore() {
|
function createActionsStore() {
|
||||||
const { subscribe, set, update } = writable({list: null});
|
const { subscribe, set, update } = writable({list: null, fileIdx: null});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
subscribe,
|
subscribe,
|
||||||
@ -14,14 +14,28 @@ function createActionsStore() {
|
|||||||
|
|
||||||
refresh: async () => {
|
refresh: async () => {
|
||||||
const list = await getActions();
|
const list = await getActions();
|
||||||
update((m) => Object.assign(m, {list}));
|
const fileIdx = {};
|
||||||
|
list.forEach(function(action, k) {
|
||||||
|
fileIdx[action.path] = action;
|
||||||
|
});
|
||||||
|
|
||||||
|
update((m) => (Object.assign(m, {list, fileIdx})));
|
||||||
return list;
|
return list;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
getActionByFilename: (fname) => {
|
||||||
|
|
||||||
|
},
|
||||||
|
|
||||||
update: (res_actions, cb=null) => {
|
update: (res_actions, cb=null) => {
|
||||||
if (res_actions.status === 200) {
|
if (res_actions.status === 200) {
|
||||||
res_actions.json().then((list) => {
|
res_actions.json().then((list) => {
|
||||||
update((m) => (Object.assign(m, {list})));
|
const fileIdx = {};
|
||||||
|
list.forEach(function(action, k) {
|
||||||
|
fileIdx[action.path] = action;
|
||||||
|
})
|
||||||
|
|
||||||
|
update((m) => (Object.assign(m, {list, fileIdx})));
|
||||||
|
|
||||||
if (cb) {
|
if (cb) {
|
||||||
cb(list);
|
cb(list);
|
||||||
@ -34,3 +48,8 @@ function createActionsStore() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const actions = createActionsStore();
|
export const actions = createActionsStore();
|
||||||
|
|
||||||
|
export const actions_idx = derived(
|
||||||
|
actions,
|
||||||
|
($actions) => ($actions.fileIdx),
|
||||||
|
);
|
||||||
|
36
ui/src/stores/routines.js
Normal file
36
ui/src/stores/routines.js
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
import { writable } from 'svelte/store';
|
||||||
|
|
||||||
|
import { getRoutines } from '../lib/routine'
|
||||||
|
|
||||||
|
function createRoutinesStore() {
|
||||||
|
const { subscribe, set, update } = writable({list: null});
|
||||||
|
|
||||||
|
return {
|
||||||
|
subscribe,
|
||||||
|
|
||||||
|
set: (v) => {
|
||||||
|
update((m) => Object.assign(m, v));
|
||||||
|
},
|
||||||
|
|
||||||
|
refresh: async () => {
|
||||||
|
const list = await getRoutines();
|
||||||
|
update((m) => Object.assign(m, {list}));
|
||||||
|
return list;
|
||||||
|
},
|
||||||
|
|
||||||
|
update: (res_routines, cb=null) => {
|
||||||
|
if (res_routines.status === 200) {
|
||||||
|
res_routines.json().then((list) => {
|
||||||
|
update((m) => (Object.assign(m, {list})));
|
||||||
|
|
||||||
|
if (cb) {
|
||||||
|
cb(list);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
export const routines = createRoutinesStore();
|
Loading…
x
Reference in New Issue
Block a user