summaryrefslogtreecommitdiffstats
path: root/server
diff options
context:
space:
mode:
Diffstat (limited to 'server')
-rw-r--r--server/Makefile10
-rw-r--r--server/record.go60
-rw-r--r--server/server.go80
3 files changed, 150 insertions, 0 deletions
diff --git a/server/Makefile b/server/Makefile
new file mode 100644
index 0000000..8954492
--- /dev/null
+++ b/server/Makefile
@@ -0,0 +1,10 @@
+SRC = server.go record.go
+
+server: ${SRC}
+ go build $^
+
+fmt:
+ gofmt -l -s -w ${SRC}
+
+clean:
+ rm -f server
diff --git a/server/record.go b/server/record.go
new file mode 100644
index 0000000..9051146
--- /dev/null
+++ b/server/record.go
@@ -0,0 +1,60 @@
+package main
+
+import (
+ "time"
+)
+
+type Record[T any] struct {
+ put chan<- T
+ get chan<- chan T
+ getRecent chan<- chan T
+}
+
+type entry[T any] struct {
+ t time.Time
+ v T
+}
+
+func newRecord[T any]() Record[T] {
+ put := make(chan T)
+ get := make(chan chan T)
+ getRecent := make(chan chan T)
+
+ go func() {
+ var entries []entry[T]
+
+ for {
+ select {
+ case v, ok := <-put:
+ if !ok {
+ return
+ }
+ entries = append(entries, entry[T]{time.Now(), v})
+ case c, ok := <-get:
+ if !ok {
+ return
+ }
+ for _, e := range entries {
+ c <- e.v
+ }
+ close(c)
+ case c, ok := <-getRecent:
+ if !ok {
+ return
+ }
+ if len(entries) > 0 {
+ c <- entries[len(entries)-1].v
+ }
+ close(c)
+ }
+ }
+ }()
+
+ return Record[T]{put, get, getRecent}
+}
+
+func (l Record[T]) Close() {
+ close(l.put)
+ close(l.get)
+ close(l.getRecent)
+}
diff --git a/server/server.go b/server/server.go
new file mode 100644
index 0000000..2795890
--- /dev/null
+++ b/server/server.go
@@ -0,0 +1,80 @@
+package main
+
+import (
+ "fmt"
+ "log"
+ "net/http"
+ "strconv"
+)
+
+const addr = ":9090"
+var rooms = []RoomID {
+ ",4AL[+V*:*k*n{7vL{}/d=K#Mo*y*^.@",
+ "Jq!+<p3g-iu%-vU]FZp2H,AKZWp@!4![",
+}
+
+type Humidity float32
+type RoomID string
+
+type HumidityHandler struct {
+ rooms map[RoomID]Record[Humidity]
+}
+
+func newHumidityHandler(rooms []RoomID) HumidityHandler {
+ h := HumidityHandler{make(map[RoomID]Record[Humidity])}
+ for _, id := range rooms {
+ h.rooms[id] = newRecord[Humidity]()
+ }
+ return h
+}
+
+func (h HumidityHandler) Close() {
+ for _, record := range h.rooms {
+ record.Close()
+ }
+}
+
+func (h HumidityHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
+ log.Println(r.Method, r.URL)
+ switch r.Method {
+ case http.MethodGet:
+ h.get(w, r)
+ case http.MethodPost:
+ h.post(w, r)
+ default:
+ w.WriteHeader(http.StatusMethodNotAllowed)
+ fmt.Fprintf(w, "invalid method: '%s'", r.Method)
+ }
+}
+
+func (h HumidityHandler) get(w http.ResponseWriter, r *http.Request) {
+ c := make(chan float32)
+ h.humidity.getRecent <- c
+ humidity, ok := <-c
+ if !ok {
+ w.WriteHeader(http.StatusGone)
+ fmt.Fprintf(w, "no humidity data stored on server")
+ return
+ }
+ fmt.Fprintf(w, "%.2f", humidity)
+}
+
+func (h HumidityHandler) post(w http.ResponseWriter, r *http.Request) {
+ query := r.URL.RawQuery
+ humidity, err := strconv.ParseFloat(query, 32)
+ if err != nil {
+ w.WriteHeader(http.StatusBadRequest)
+ fmt.Fprintf(w, "invalid query string: '%s'", query)
+ return
+ }
+ h.humidity.put <- float32(humidity)
+}
+
+func main() {
+ humidityHandler := newHumidityHandler()
+ defer humidityHandler.Close()
+
+ http.Handle("/humidity", humidityHandler)
+ fmt.Printf("Listening on %s...\n", addr)
+ log.Fatal(http.ListenAndServe(addr, nil))
+}