Browse Source

Added PATCH/PUT /api/light/{id}

login_bugfix
Gisle Aune 5 years ago
parent
commit
3e016efcac
  1. 2
      cmd/lucifer-server/main.go
  2. 135
      controllers/light-controller.go
  3. 34
      light/service.go
  4. 22
      models/light.go

2
cmd/lucifer-server/main.go

@ -44,7 +44,7 @@ func main() {
// Controllers
userController := controllers.NewUserController(sqlite.UserRepository, sqlite.SessionRepository)
groupController := controllers.NewGroupController(sqlite.GroupRepository, sqlite.UserRepository, sqlite.LightRepository)
lightController := controllers.NewLightController(sqlite.GroupRepository, sqlite.UserRepository, sqlite.LightRepository)
lightController := controllers.NewLightController(lightService, sqlite.GroupRepository, sqlite.UserRepository, sqlite.LightRepository)
// Router
router := mux.NewRouter()

135
controllers/light-controller.go

@ -2,9 +2,12 @@ package controllers
import (
"database/sql"
"encoding/json"
"net/http"
"strconv"
"git.aiterp.net/lucifer/lucifer/light"
"git.aiterp.net/lucifer/lucifer/internal/httperr"
"git.aiterp.net/lucifer/lucifer/internal/respond"
@ -15,9 +18,10 @@ import (
// The LightController is a controller for /api/light/.
type LightController struct {
groups models.GroupRepository
users models.UserRepository
lights models.LightRepository
service *light.Service
groups models.GroupRepository
users models.UserRepository
lights models.LightRepository
}
// getLights (`GET /:id`): Get user by id
@ -57,35 +61,89 @@ func (c *LightController) getLights(w http.ResponseWriter, r *http.Request) {
}
func (c *LightController) getLight(w http.ResponseWriter, r *http.Request) {
user := models.UserFromContext(r.Context())
idStr := mux.Vars(r)["id"]
id, err := strconv.ParseInt(idStr, 10, 32)
_, light, err := c.findLight(r)
if err != nil {
respond.Error(w, http.StatusBadRequest, "invalid_id", "The light id "+idStr+" is not valid.")
httperr.Respond(w, err)
return
}
light, err := c.lights.FindByID(r.Context(), int(id))
if err == sql.ErrNoRows {
httperr.Respond(w, httperr.NotFound("Light"))
return
} else if err != nil {
httperr.Respond(w, err)
respond.Data(w, light)
}
func (c *LightController) updateLight(w http.ResponseWriter, r *http.Request) {
patch := struct {
Color *string `json:"color,omitempty"`
Brightness *int `json:"brightness,omitempty"`
On *bool `json:"on,omitempty"`
Name *string `json:"name,omitempty"`
GroupID *int `json:"groupId,omitempty"`
}{}
if err := json.NewDecoder(r.Body).Decode(&patch); err != nil {
respond.Error(w, http.StatusBadRequest, "invalid_json", "Input is not valid JSON.")
return
}
group, err := c.groups.FindByID(r.Context(), light.GroupID)
if err == sql.ErrNoRows {
httperr.Respond(w, httperr.NotFound("Group"))
return
} else if err != nil {
group, light, err := c.findLight(r)
if err != nil {
httperr.Respond(w, err)
return
}
if !group.Permission(user.ID).Read {
respond.Error(w, http.StatusForbidden, "permission_denied", "You do not have permission to see this light.")
if patch.Color != nil {
err := light.SetColor(*patch.Color)
if err != nil {
httperr.Respond(w, err)
return
}
}
if patch.Name != nil {
if len(*patch.Name) == 0 {
respond.Error(w, 400, "invalid_name", "The name is empty.")
return
}
light.Name = *patch.Name
}
if patch.Brightness != nil {
if *patch.Brightness < 0 || *patch.Brightness > 255 {
respond.Error(w, 400, "invalid_brightness", "The brightness must be a value between 0-255 inclusive.")
return
}
light.Brightness = uint8(*patch.Brightness)
}
if patch.On != nil {
light.On = *patch.On
}
if patch.GroupID != nil && *patch.GroupID != light.GroupID {
user := models.UserFromContext(r.Context())
if !group.Permission(user.ID).Delete {
respond.Error(w, 403, "cannot_move_out", "You are not permitted to delete lights from group.")
return
}
// Anyone is allowed to move lights TO group 0 (Lonely Lights) as it's the closest thing there is
// to deleting lights.
if *patch.GroupID != 0 {
targetGroup, err := c.groups.FindByID(r.Context(), *patch.GroupID)
if err != nil {
respond.Error(w, 404, "group_not_found", "The group could not be found.")
return
}
if !targetGroup.Permission(user.ID).Create {
respond.Error(w, 403, "cannot_move_in", "You are not permitted to create lights in target group.")
return
}
}
light.GroupID = *patch.GroupID
}
err = c.service.UpdateLight(r.Context(), light)
if err != nil {
httperr.Respond(w, err)
return
}
@ -98,9 +156,40 @@ func (c *LightController) Mount(router *mux.Router, prefix string) {
sub.HandleFunc("/", c.getLights).Methods("GET")
sub.HandleFunc("/{id}", c.getLight).Methods("GET")
sub.HandleFunc("/{id}", c.updateLight).Methods("PATCH", "PUT")
}
func (c *LightController) findLight(r *http.Request) (models.Group, models.Light, error) {
user := models.UserFromContext(r.Context())
idStr := mux.Vars(r)["id"]
id, err := strconv.ParseInt(idStr, 10, 32)
if err != nil {
return models.Group{}, models.Light{}, &httperr.Error{Status: http.StatusForbidden, Kind: "invalid_id", Message: "The light id " + idStr + " is not valid."}
}
light, err := c.lights.FindByID(r.Context(), int(id))
if err == sql.ErrNoRows {
return models.Group{}, models.Light{}, httperr.NotFound("Light")
} else if err != nil {
return models.Group{}, models.Light{}, err
}
group, err := c.groups.FindByID(r.Context(), light.GroupID)
if err == sql.ErrNoRows {
return models.Group{}, models.Light{}, httperr.NotFound("Group")
} else if err != nil {
return models.Group{}, models.Light{}, err
}
if !group.Permission(user.ID).Read {
return models.Group{}, models.Light{}, &httperr.Error{Status: http.StatusForbidden, Kind: "permission_denied", Message: "Thou canst not see the light."}
}
return group, light, nil
}
// NewLightController creates a new LightController.
func NewLightController(groups models.GroupRepository, users models.UserRepository, lights models.LightRepository) *LightController {
return &LightController{groups: groups, users: users, lights: lights}
func NewLightController(service *light.Service, groups models.GroupRepository, users models.UserRepository, lights models.LightRepository) *LightController {
return &LightController{service: service, groups: groups, users: users, lights: lights}
}

34
light/service.go

@ -2,18 +2,23 @@ package light
import (
"context"
"errors"
"log"
"net/http"
"sync"
"time"
"git.aiterp.net/lucifer/lucifer/internal/httperr"
"git.aiterp.net/lucifer/lucifer/models"
)
// ErrUnknownDriver is returned by any function asking for a driver name if the driver specified doesn't exist.
var ErrUnknownDriver = errors.New("Unknown driver specified")
var ErrUnknownDriver = &httperr.Error{Status: http.StatusNotImplemented, Kind: "unknown_driver", Message: "Unknown light driver"}
// A Service wraps the repos for lights and bridges and takes care of the business logic.
type Service struct {
mutex sync.Mutex
bridges models.BridgeRepository
lights models.LightRepository
}
@ -47,6 +52,9 @@ func (s *Service) DirectConnect(ctx context.Context, driver string, addr string,
// SyncLights syncs all lights in a bridge with the state in the database.
func (s *Service) SyncLights(ctx context.Context, bridge models.Bridge) error {
s.mutex.Lock()
defer s.mutex.Unlock()
d, ok := drivers[bridge.Driver]
if !ok {
return ErrUnknownDriver
@ -97,6 +105,28 @@ func (s *Service) SyncLights(ctx context.Context, bridge models.Bridge) error {
return nil
}
// UpdateLight updates the light immediately.
func (s *Service) UpdateLight(ctx context.Context, light models.Light) error {
s.mutex.Lock()
defer s.mutex.Unlock()
err := s.lights.Update(ctx, light)
if err != nil {
return err
}
bridge, err := s.bridges.FindByID(ctx, light.BridgeID)
if err != nil {
return httperr.NotFound("Bridge")
}
d, ok := drivers[bridge.Driver]
if !ok {
return ErrUnknownDriver
}
return d.Apply(ctx, bridge, light)
}
// SyncLoop runs synclight on all bridges twice every second until the context is
// done.
func (s *Service) SyncLoop(ctx context.Context) {

22
models/light.go

@ -4,6 +4,9 @@ import (
"context"
"encoding/hex"
"errors"
"strings"
"git.aiterp.net/lucifer/lucifer/internal/httperr"
)
// ErrMalformedColor is returned by light.ColorRGBf when the color value is invalid.
@ -21,6 +24,25 @@ type Light struct {
Brightness uint8 `json:"brightness" db:"brightness"`
}
// SetColor sets the color to a hex string, or returns an error if it's not valid.
func (light *Light) SetColor(hexStr string) error {
if len(hexStr) == 7 && hexStr[0] == '#' {
hexStr = hexStr[1:]
}
if len(hexStr) != 6 {
return httperr.Error{Status: 400, Kind: "invalid_color", Message: hexStr + " is not a valid color."}
}
_, err := hex.DecodeString(hexStr)
if err != nil {
return httperr.Error{Status: 400, Kind: "invalid_color", Message: hexStr + " is not a valid color."}
}
light.Color = strings.ToUpper(hexStr)
return nil
}
// SetColorRGB sets the color with an RGB value.
func (light *Light) SetColorRGB(r, g, b uint8) {
light.Color = hex.EncodeToString([]byte{r, g, b})

Loading…
Cancel
Save