package server import ( "encoding/json" "errors" "fmt" "forge.lightcrystal.systems/nilix/quartzgun/renderer" "forge.lightcrystal.systems/nilix/quartzgun/router" "forge.lightcrystal.systems/nilix/quartzgun/util" "forge.lightcrystal.systems/nilix/underbbs/adapter" "forge.lightcrystal.systems/nilix/underbbs/models" "html/template" "log" "net/http" "strings" ) func getSubscriberKey(req *http.Request) (string, error) { authHeader := req.Header.Get("Authorization") if strings.HasPrefix(authHeader, "Bearer ") { return strings.Split(authHeader, "Bearer ")[1], nil } return "", errors.New("No subscriber key") } func getSubscriberByKey(key string, subscribers map[*Subscriber][]adapter.Adapter) *Subscriber { for s, _ := range subscribers { if s.key == key { return s } } return nil } func setAdaptersForSubscriber(key string, adapters []adapter.Adapter, subscribers map[*Subscriber][]adapter.Adapter) error { var ptr *Subscriber = nil log.Print("looking for subscriber in map...") for s, _ := range subscribers { if s.key == key { ptr = s } } if ptr != nil { log.Print("setting adaters for the found subscriber: " + ptr.key) subscribers[ptr] = adapters return nil } return errors.New("subscriber not present in map") } func apiConfigureAdapters(next http.Handler, subscribers map[*Subscriber][]adapter.Adapter) http.Handler { return http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) { // get subscriber key skey, err := getSubscriberKey(req) if err != nil { w.WriteHeader(500) return } subscriber := getSubscriberByKey(skey, subscribers) if subscriber == nil { w.WriteHeader(404) return } // decode adapter config from request body settings := make([]models.Settings, 0) err = json.NewDecoder(req.Body).Decode(&settings) if err != nil { w.WriteHeader(400) next.ServeHTTP(w, req) return } // iterate through settings and create adapters adapters := make([]adapter.Adapter, 0) for _, s := range settings { var a adapter.Adapter switch s.Protocol { case "nostr": a = &adapter.NostrAdapter{} case "mastodon": a = &adapter.MastoAdapter{} case "misskey": a = &adapter.MisskeyAdapter{} case "honk": a = &adapter.HonkAdapter{} default: break } err := a.Init(s, &subscriber.data) if err != nil { util.AddContextValue(req, "data", err.Error()) w.WriteHeader(500) next.ServeHTTP(w, req) return } log.Print("adapter initialized; adding to array") adapters = append(adapters, a) log.Print("adapter added to array") } // TODO: cancel subscriptions on any existing adapters // store the adapters in the subscriber map err = setAdaptersForSubscriber(skey, adapters, subscribers) if err != nil { util.AddContextValue(req, "data", err.Error()) w.WriteHeader(500) next.ServeHTTP(w, req) } w.WriteHeader(201) next.ServeHTTP(w, req) }) } func apiGetAdapters(next http.Handler) http.Handler { return http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) { w.WriteHeader(201) next.ServeHTTP(w, req) }) } type subscribeParams struct { Filter string `json:"filter"` Target *string `json:"target,omitempty"` } func apiAdapterSubscribe(next http.Handler, subscribers map[*Subscriber][]adapter.Adapter) http.Handler { return http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) { // get subscriber key skey, err := getSubscriberKey(req) if err != nil { w.WriteHeader(500) return } subscriber := getSubscriberByKey(skey, subscribers) if subscriber == nil { w.WriteHeader(404) return } adapters := subscribers[subscriber] urlParams := req.Context().Value("params").(map[string]string) adapter := urlParams["adapter_id"] sp := subscribeParams{} err = json.NewDecoder(req.Body).Decode(&sp) for _, a := range adapters { if a.Name() == adapter { fmt.Printf("adapter.subscribe call: %s {%s, %s}\n", adapter, sp.Filter, *sp.Target) a.Subscribe(sp.Filter, sp.Target) w.WriteHeader(201) next.ServeHTTP(w, req) } } w.WriteHeader(404) next.ServeHTTP(w, req) }) } func ProtectWithSubscriberKey(next http.Handler, subscribers map[*Subscriber][]adapter.Adapter) http.Handler { return http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) { authHeader := req.Header.Get("Authorization") if strings.HasPrefix(authHeader, "Bearer ") { subscriberKey := strings.Split(authHeader, "Bearer ")[1] if getSubscriberByKey(subscriberKey, subscribers) != nil { next.ServeHTTP(w, req) return } } w.WriteHeader(http.StatusUnauthorized) }) } func apiAdapterFetch(next http.Handler, subscribers map[*Subscriber][]adapter.Adapter) http.Handler { return http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) { authHeader := req.Header.Get("Authorization") if strings.HasPrefix(authHeader, "Bearer ") { subscriberKey := strings.Split(authHeader, "Bearer ")[1] s := getSubscriberByKey(subscriberKey, subscribers) if s != nil { apiParams := req.Context().Value("params").(map[string]string) queryParams := req.URL.Query() for _, a := range subscribers[s] { if a.Name() == apiParams["adapter_id"] { err := a.Fetch(queryParams["entity_type"][0], queryParams["entity_id"]) if err != nil { log.Print(err.Error()) w.WriteHeader(http.StatusInternalServerError) } else { w.WriteHeader(http.StatusAccepted) } next.ServeHTTP(w, req) return } } } } w.WriteHeader(http.StatusUnauthorized) }) } func (self *BBSServer) apiMux() http.Handler { errTemplate, err := template.New("err").Parse("{{ $params := (.Context).Value \"params\" }}
{{ $params.ErrorMessage }}
") if err != nil { panic("error template was malformed") } rtr := &router.Router{ Fallback: *errTemplate, } // adapters (POST & GET) rtr.Post("/adapters", ProtectWithSubscriberKey( apiConfigureAdapters(renderer.JSON("data"), self.subscribers), self.subscribers, )) rtr.Get("/adapters", ProtectWithSubscriberKey( apiGetAdapters(renderer.JSON("data")), self.subscribers, )) // adapters/:name/subscribe rtr.Post(`/adapters/(?P