|
|
package api
import ( "context" "git.aiterp.net/lucifer/new-server/app/config" "git.aiterp.net/lucifer/new-server/app/services/scene" "git.aiterp.net/lucifer/new-server/models" "github.com/gin-gonic/gin" "golang.org/x/sync/errgroup" "log" "strconv" "strings" "time" )
func fetchDevices(ctx context.Context, fetchStr string) ([]models.Device, error) { if strings.HasPrefix(fetchStr, "tag:") { return config.DeviceRepository().FetchByReference(ctx, models.RKTag, fetchStr[4:]) } else if strings.HasPrefix(fetchStr, "bridge:") { return config.DeviceRepository().FetchByReference(ctx, models.RKBridgeID, fetchStr[7:]) } else if strings.HasPrefix(fetchStr, "id:") { return config.DeviceRepository().FetchByReference(ctx, models.RKDeviceID, fetchStr[7:]) } else if strings.HasPrefix(fetchStr, "name:") { return config.DeviceRepository().FetchByReference(ctx, models.RKName, fetchStr[7:]) }else if fetchStr == "all" { return config.DeviceRepository().FetchByReference(ctx, models.RKAll, "") } else { _, err := strconv.Atoi(fetchStr) if err != nil { return config.DeviceRepository().FetchByReference(ctx, models.RKName, fetchStr) }
return config.DeviceRepository().FetchByReference(ctx, models.RKDeviceID, fetchStr) } }
func Devices(r gin.IRoutes) { r.GET("", handler(func(c *gin.Context) (interface{}, error) { return config.DeviceRepository().FetchByReference(ctxOf(c), models.RKAll, "") }))
r.GET("/:fetch", handler(func(c *gin.Context) (interface{}, error) { return fetchDevices(ctxOf(c), c.Param("fetch")) }))
r.PUT("", handler(func(c *gin.Context) (interface{}, error) { var body []struct { Fetch string `json:"fetch"` SetState models.NewDeviceState `json:"setState"` } err := parseBody(c, &body) if err != nil { return nil, err }
set := make(map[int]bool) changed := make([]models.Device, 0, 64) for _, job := range body { devices, err := fetchDevices(ctxOf(c), job.Fetch) if err != nil { return nil, err } if len(devices) == 0 { return []models.Device{}, nil }
for i := range devices { if set[devices[i].ID] { continue }
err := devices[i].SetState(job.SetState) if err != nil { return nil, err }
_ = scene.GlobalManager().UpdateDevice(ctxOf(c), &devices[i], nil)
set[devices[i].ID] = true changed = append(changed, devices[i]) } }
config.PublishChannel <- scene.GlobalManager().FilterUnassigned(changed)
go func() { for _, device := range changed { err := config.DeviceRepository().Save(context.Background(), &device, models.SMState) if err != nil { log.Println("Failed to save device for state:", err) continue } } }()
return changed, nil }))
r.PUT("/:fetch", handler(func(c *gin.Context) (interface{}, error) { update := models.DeviceUpdate{} err := parseBody(c, &update) if err != nil { return nil, err }
devices, err := fetchDevices(ctxOf(c), c.Param("fetch")) if err != nil { return nil, err } if len(devices) == 0 { return []models.Device{}, nil }
for i := range devices { devices[i].ApplyUpdate(update)
err := config.DeviceRepository().Save(context.Background(), &devices[i], models.SMProperties) if err != nil { log.Println("Failed to save device for state:", err) continue } }
return devices, nil }))
r.PUT("/:fetch/state", handler(func(c *gin.Context) (interface{}, error) { state := models.NewDeviceState{} err := parseBody(c, &state) if err != nil { return nil, err }
devices, err := fetchDevices(ctxOf(c), c.Param("fetch")) if err != nil { return nil, err } if len(devices) == 0 { return []models.Device{}, nil }
for i := range devices { err := devices[i].SetState(state) if err != nil { return nil, err }
_ = scene.GlobalManager().UpdateDevice(ctxOf(c), &devices[i], nil) }
config.PublishChannel <- scene.GlobalManager().FilterUnassigned(devices)
go func() { for _, device := range devices { err := config.DeviceRepository().Save(context.Background(), &device, models.SMState) if err != nil { log.Println("Failed to save device for state:", err) continue } } }()
return devices, nil }))
r.PUT("/:fetch/tags", handler(func(c *gin.Context) (interface{}, error) { var body struct { Add []string `json:"add"` Remove []string `json:"remove"` } err := parseBody(c, &body) if err != nil { return nil, err }
devices, err := fetchDevices(ctxOf(c), c.Param("fetch")) if err != nil { return nil, err } if len(devices) == 0 { return []models.Device{}, nil }
for i := range devices { device := &devices[i]
for _, tag := range body.Add { found := false for _, tag2 := range device.Tags { if tag == tag2 { found = true break } }
if !found { device.Tags = append(device.Tags, tag) } } for _, tag := range body.Remove { index := -1 for i, tag2 := range device.Tags { if tag == tag2 { index = i } } if index == -1 { continue }
device.Tags = append(device.Tags[:index], device.Tags[index+1:]...) }
err = config.DeviceRepository().Save(ctxOf(c), device, models.SMTags) if err != nil { return nil, err } }
return devices, nil }))
r.PUT("/:fetch/scene", handler(func(c *gin.Context) (interface{}, error) { var body models.DeviceSceneAssignment err := parseBody(c, &body) if err != nil { return nil, err }
devices, err := fetchDevices(ctxOf(c), c.Param("fetch")) if err != nil { return nil, err } if len(devices) == 0 { return []models.Device{}, nil }
assignedScene, err := config.SceneRepository().Find(ctxOf(c), body.SceneID) if err != nil { return nil, err } if body.DurationMS < 0 { body.DurationMS = 0 } body.StartTime = time.Now()
for i := range devices { devices[i].SceneAssignments = []models.DeviceSceneAssignment{body}
err := scene.GlobalManager().UpdateDevice(ctxOf(c), &devices[i], assignedScene) if err != nil { return nil, err } }
eg := errgroup.Group{} for i := range devices { eg.Go(func() error { return config.DeviceRepository().Save(ctxOf(c), &devices[i], 0) }) }
err = eg.Wait() if err != nil { return nil, err }
return devices, nil })) }
|