added get bus even API request
also provided framework for extracting filters
This commit is contained in:
parent
7aaa47e142
commit
3f1df06d1b
|
@ -11,6 +11,7 @@ import (
|
||||||
"log/slog"
|
"log/slog"
|
||||||
|
|
||||||
"github.com/kschamplin/gotelem"
|
"github.com/kschamplin/gotelem"
|
||||||
|
"github.com/kschamplin/gotelem/internal/api"
|
||||||
"github.com/kschamplin/gotelem/internal/db"
|
"github.com/kschamplin/gotelem/internal/db"
|
||||||
"github.com/kschamplin/gotelem/skylab"
|
"github.com/kschamplin/gotelem/skylab"
|
||||||
"github.com/kschamplin/gotelem/xbee"
|
"github.com/kschamplin/gotelem/xbee"
|
||||||
|
@ -132,7 +133,6 @@ func serve(cCtx *cli.Context) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// xBeeService provides data over an Xbee device, either by serial or TCP
|
// xBeeService provides data over an Xbee device, either by serial or TCP
|
||||||
// based on the url provided in the xbee flag. see the description for details.
|
// based on the url provided in the xbee flag. see the description for details.
|
||||||
type xBeeService struct {
|
type xBeeService struct {
|
||||||
|
@ -220,7 +220,7 @@ func (h *httpService) Start(cCtx *cli.Context, deps svcDeps) (err error) {
|
||||||
broker := deps.Broker
|
broker := deps.Broker
|
||||||
db := deps.Db
|
db := deps.Db
|
||||||
|
|
||||||
r := gotelem.TelemRouter(logger, broker, db)
|
r := api.TelemRouter(logger, broker, db)
|
||||||
|
|
||||||
//
|
//
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
package gotelem
|
package api
|
||||||
|
|
||||||
// this file defines the HTTP handlers and routes.
|
// this file defines the HTTP handlers and routes.
|
||||||
|
|
||||||
|
@ -13,13 +13,14 @@ import (
|
||||||
"github.com/go-chi/chi/v5"
|
"github.com/go-chi/chi/v5"
|
||||||
"github.com/go-chi/chi/v5/middleware"
|
"github.com/go-chi/chi/v5/middleware"
|
||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
|
"github.com/kschamplin/gotelem"
|
||||||
"github.com/kschamplin/gotelem/internal/db"
|
"github.com/kschamplin/gotelem/internal/db"
|
||||||
"github.com/kschamplin/gotelem/skylab"
|
"github.com/kschamplin/gotelem/skylab"
|
||||||
"nhooyr.io/websocket"
|
"nhooyr.io/websocket"
|
||||||
"nhooyr.io/websocket/wsjson"
|
"nhooyr.io/websocket/wsjson"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TelemRouter(log *slog.Logger, broker *Broker, db *db.TelemDb) http.Handler {
|
func TelemRouter(log *slog.Logger, broker *gotelem.Broker, db *db.TelemDb) http.Handler {
|
||||||
r := chi.NewRouter()
|
r := chi.NewRouter()
|
||||||
|
|
||||||
r.Use(middleware.RequestID)
|
r.Use(middleware.RequestID)
|
||||||
|
@ -41,7 +42,7 @@ func TelemRouter(log *slog.Logger, broker *Broker, db *db.TelemDb) http.Handler
|
||||||
}
|
}
|
||||||
|
|
||||||
// define API version 1 routes.
|
// define API version 1 routes.
|
||||||
func apiV1(broker *Broker, db *db.TelemDb) chi.Router {
|
func apiV1(broker *gotelem.Broker, tdb *db.TelemDb) chi.Router {
|
||||||
r := chi.NewRouter()
|
r := chi.NewRouter()
|
||||||
// this API only accepts JSON.
|
// this API only accepts JSON.
|
||||||
r.Use(middleware.AllowContentType("application/json"))
|
r.Use(middleware.AllowContentType("application/json"))
|
||||||
|
@ -56,7 +57,7 @@ func apiV1(broker *Broker, db *db.TelemDb) chi.Router {
|
||||||
})
|
})
|
||||||
|
|
||||||
r.Route("/packets", func(r chi.Router) {
|
r.Route("/packets", func(r chi.Router) {
|
||||||
r.Get("/subscribe", apiV1PacketSubscribe(broker, db))
|
r.Get("/subscribe", apiV1PacketSubscribe(broker, tdb))
|
||||||
r.Post("/", func(w http.ResponseWriter, r *http.Request) {
|
r.Post("/", func(w http.ResponseWriter, r *http.Request) {
|
||||||
var pkgs []skylab.BusEvent
|
var pkgs []skylab.BusEvent
|
||||||
decoder := json.NewDecoder(r.Body)
|
decoder := json.NewDecoder(r.Body)
|
||||||
|
@ -65,25 +66,59 @@ func apiV1(broker *Broker, db *db.TelemDb) chi.Router {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
// we have a list of packets now. let's commit them.
|
// we have a list of packets now. let's commit them.
|
||||||
db.AddEvents(pkgs...)
|
tdb.AddEvents(pkgs...)
|
||||||
})
|
})
|
||||||
r.Get("/", func(w http.ResponseWriter, r *http.Request) {
|
r.Get("/", func(w http.ResponseWriter, r *http.Request) {
|
||||||
// this should use http query params to return a list of packets.
|
// this should use http query params to return a list of packets.
|
||||||
|
bef, err := extractBusEventFilter(r)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
lim, err := extractLimitModifier(r)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: is the following check needed?
|
||||||
|
var res []skylab.BusEvent
|
||||||
|
if lim != nil {
|
||||||
|
res, err = tdb.GetPackets(r.Context(), *bef, lim)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
res, err = tdb.GetPackets(r.Context(), *bef)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
b, err := json.Marshal(res)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
w.Write(b)
|
||||||
|
|
||||||
})
|
})
|
||||||
|
|
||||||
// this is to get a single field
|
// this is to get a single field
|
||||||
r.Get("/{name:[a-z_]+}/{field:[a-z_]+}", apiV1GetValues(db))
|
r.Get("/{name:[a-z_]+}/{field:[a-z_]+}", apiV1GetValues(tdb))
|
||||||
|
|
||||||
})
|
})
|
||||||
|
|
||||||
// records are driving segments/runs.
|
// records are driving segments/runs.
|
||||||
r.Route("/records", func(r chi.Router) {
|
r.Route("/records", func(r chi.Router) {
|
||||||
r.Get("/", apiV1GetRecords(db)) // get all runs
|
r.Get("/", apiV1GetRecords(tdb)) // get all runs
|
||||||
r.Get("/active", apiV1GetActiveRecord(db)) // get current run (no end time)
|
r.Get("/active", apiV1GetActiveRecord(tdb)) // get current run (no end time)
|
||||||
r.Post("/", apiV1StartRecord(db)) // create a new run (with note). Ends active run if any, and creates new active run (no end time)
|
r.Post("/", apiV1StartRecord(tdb)) // create a new run (with note). Ends active run if any, and creates new active run (no end time)
|
||||||
r.Get("/{id}", apiV1GetRecord(db)) // get details on a specific run
|
r.Get("/{id}", apiV1GetRecord(tdb)) // get details on a specific run
|
||||||
r.Put("/{id}", apiV1UpdateRecord(db)) // update a specific run. Can only be used to add notes/metadata, and not to change time/id.
|
r.Put("/{id}", apiV1UpdateRecord(tdb)) // update a specific run. Can only be used to add notes/metadata, and not to change time/id.
|
||||||
|
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -100,7 +135,7 @@ type apiV1Subscriber struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
// this is a websocket stream.
|
// this is a websocket stream.
|
||||||
func apiV1PacketSubscribe(broker *Broker, db *db.TelemDb) http.HandlerFunc {
|
func apiV1PacketSubscribe(broker *gotelem.Broker, db *db.TelemDb) http.HandlerFunc {
|
||||||
return func(w http.ResponseWriter, r *http.Request) {
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
conn_id := r.RemoteAddr + uuid.New().String()
|
conn_id := r.RemoteAddr + uuid.New().String()
|
||||||
sub, err := broker.Subscribe(conn_id)
|
sub, err := broker.Subscribe(conn_id)
|
||||||
|
@ -162,7 +197,7 @@ func apiV1GetValues(db *db.TelemDb) http.HandlerFunc {
|
||||||
start, err = time.Parse(time.RFC3339, startString)
|
start, err = time.Parse(time.RFC3339, startString)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
http.Error(w, "error getting values", http.StatusInternalServerError)
|
http.Error(w, "error getting values", http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
end := time.Now().Add(1 * time.Hour)
|
end := time.Now().Add(1 * time.Hour)
|
||||||
|
@ -176,7 +211,7 @@ func apiV1GetValues(db *db.TelemDb) http.HandlerFunc {
|
||||||
}
|
}
|
||||||
name := chi.URLParam(r, "name")
|
name := chi.URLParam(r, "name")
|
||||||
field := chi.URLParam(r, "field")
|
field := chi.URLParam(r, "field")
|
||||||
|
|
||||||
// TODO: add limit and pagination
|
// TODO: add limit and pagination
|
||||||
|
|
||||||
res, err := db.GetValues(r.Context(), name, field, start, end)
|
res, err := db.GetValues(r.Context(), name, field, start, end)
|
59
internal/api/utils.go
Normal file
59
internal/api/utils.go
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
package api
|
||||||
|
// This file contains common behaviors that are used across various requests
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
"strconv"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/kschamplin/gotelem/internal/db"
|
||||||
|
)
|
||||||
|
|
||||||
|
func extractBusEventFilter(r *http.Request) (*db.BusEventFilter, error) {
|
||||||
|
|
||||||
|
bef := &db.BusEventFilter{}
|
||||||
|
|
||||||
|
v := r.URL.Query()
|
||||||
|
bef.Names = v["name"] // put all the names in.
|
||||||
|
if el := v.Get("start"); el != "" {
|
||||||
|
// parse the start time query.
|
||||||
|
t, err := time.Parse(time.RFC3339, el)
|
||||||
|
if err != nil {
|
||||||
|
return bef, err
|
||||||
|
}
|
||||||
|
bef.TimerangeStart = t
|
||||||
|
}
|
||||||
|
if el := v.Get("end"); el != "" {
|
||||||
|
// parse the start time query.
|
||||||
|
t, err := time.Parse(time.RFC3339, el)
|
||||||
|
if err != nil {
|
||||||
|
return bef, err
|
||||||
|
}
|
||||||
|
bef.TimerangeStart = t
|
||||||
|
}
|
||||||
|
return bef, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func extractLimitModifier(r *http.Request) (*db.LimitOffsetModifier, error) {
|
||||||
|
lim := &db.LimitOffsetModifier{}
|
||||||
|
v := r.URL.Query()
|
||||||
|
if el := v.Get("limit"); el != "" {
|
||||||
|
val, err := strconv.ParseInt(el, 10, 64)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
lim.Limit = int(val)
|
||||||
|
// next, we check if we have an offset.
|
||||||
|
// we only check offset if we also have a limit.
|
||||||
|
// offset without limit isn't valid and is ignored.
|
||||||
|
if el := v.Get("offset"); el != "" {
|
||||||
|
val, err := strconv.ParseInt(el, 10, 64)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
lim.Offset = int(val)
|
||||||
|
}
|
||||||
|
return lim, nil
|
||||||
|
}
|
||||||
|
// we use the nil case to indicate that no limit was provided.
|
||||||
|
return nil, nil
|
||||||
|
}
|
|
@ -104,6 +104,10 @@ func (tdb *TelemDb) GetPackets(ctx context.Context, filter BusEventFilter, optio
|
||||||
return events, err
|
return events, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// We now need a different use-case: we would like to extract a value from
|
||||||
|
// a specific packet.
|
||||||
|
|
||||||
// Datum is a single measurement - it is more granular than a packet.
|
// Datum is a single measurement - it is more granular than a packet.
|
||||||
// the classic example is bms_measurement.current
|
// the classic example is bms_measurement.current
|
||||||
type Datum struct {
|
type Datum struct {
|
||||||
|
|
Loading…
Reference in a new issue