server.go 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304
  1. package server
  2. import (
  3. "bytes"
  4. "embed"
  5. _ "embed" // required for go:embed
  6. "encoding/json"
  7. "fmt"
  8. "golang.org/x/time/rate"
  9. "heckel.io/ntfy/config"
  10. "io"
  11. "log"
  12. "net"
  13. "net/http"
  14. "regexp"
  15. "strings"
  16. "sync"
  17. "time"
  18. )
  19. // Server is the main server
  20. type Server struct {
  21. config *config.Config
  22. topics map[string]*topic
  23. visitors map[string]*visitor
  24. mu sync.Mutex
  25. }
  26. // visitor represents an API user, and its associated rate.Limiter used for rate limiting
  27. type visitor struct {
  28. limiter *rate.Limiter
  29. seen time.Time
  30. }
  31. // errHTTP is a generic HTTP error for any non-200 HTTP error
  32. type errHTTP struct {
  33. Code int
  34. Status string
  35. }
  36. func (e errHTTP) Error() string {
  37. return fmt.Sprintf("http: %s", e.Status)
  38. }
  39. const (
  40. messageLimit = 1024
  41. visitorExpungeAfter = 30 * time.Minute
  42. )
  43. var (
  44. topicRegex = regexp.MustCompile(`^/[^/]+$`)
  45. jsonRegex = regexp.MustCompile(`^/[^/]+/json$`)
  46. sseRegex = regexp.MustCompile(`^/[^/]+/sse$`)
  47. rawRegex = regexp.MustCompile(`^/[^/]+/raw$`)
  48. staticRegex = regexp.MustCompile(`^/static/.+`)
  49. //go:embed "index.html"
  50. indexSource string
  51. //go:embed static
  52. webStaticFs embed.FS
  53. errHTTPNotFound = &errHTTP{http.StatusNotFound, http.StatusText(http.StatusNotFound)}
  54. errHTTPTooManyRequests = &errHTTP{http.StatusTooManyRequests, http.StatusText(http.StatusTooManyRequests)}
  55. )
  56. func New(conf *config.Config) *Server {
  57. return &Server{
  58. config: conf,
  59. topics: make(map[string]*topic),
  60. visitors: make(map[string]*visitor),
  61. }
  62. }
  63. func (s *Server) Run() error {
  64. go func() {
  65. ticker := time.NewTicker(s.config.ManagerInterval)
  66. for {
  67. <-ticker.C
  68. s.updateStatsAndExpire()
  69. }
  70. }()
  71. return s.listenAndServe()
  72. }
  73. func (s *Server) listenAndServe() error {
  74. log.Printf("Listening on %s", s.config.ListenHTTP)
  75. http.HandleFunc("/", s.handle)
  76. return http.ListenAndServe(s.config.ListenHTTP, nil)
  77. }
  78. func (s *Server) updateStatsAndExpire() {
  79. s.mu.Lock()
  80. defer s.mu.Unlock()
  81. // Expire visitors from rate visitors map
  82. for ip, v := range s.visitors {
  83. if time.Since(v.seen) > visitorExpungeAfter {
  84. delete(s.visitors, ip)
  85. }
  86. }
  87. // Print stats
  88. var subscribers, messages int
  89. for _, t := range s.topics {
  90. subs, msgs := t.Stats()
  91. subscribers += subs
  92. messages += msgs
  93. }
  94. log.Printf("Stats: %d topic(s), %d subscriber(s), %d message(s) sent, %d visitor(s)",
  95. len(s.topics), subscribers, messages, len(s.visitors))
  96. }
  97. func (s *Server) handle(w http.ResponseWriter, r *http.Request) {
  98. if err := s.handleInternal(w, r); err != nil {
  99. if e, ok := err.(*errHTTP); ok {
  100. s.fail(w, r, e.Code, e)
  101. } else {
  102. s.fail(w, r, http.StatusInternalServerError, err)
  103. }
  104. }
  105. }
  106. func (s *Server) handleInternal(w http.ResponseWriter, r *http.Request) error {
  107. v := s.visitor(r.RemoteAddr)
  108. if !v.limiter.Allow() {
  109. return errHTTPTooManyRequests
  110. }
  111. if r.Method == http.MethodGet && r.URL.Path == "/" {
  112. return s.handleHome(w, r)
  113. } else if r.Method == http.MethodGet && staticRegex.MatchString(r.URL.Path) {
  114. return s.handleStatic(w, r)
  115. } else if r.Method == http.MethodGet && jsonRegex.MatchString(r.URL.Path) {
  116. return s.handleSubscribeJSON(w, r)
  117. } else if r.Method == http.MethodGet && sseRegex.MatchString(r.URL.Path) {
  118. return s.handleSubscribeSSE(w, r)
  119. } else if r.Method == http.MethodGet && rawRegex.MatchString(r.URL.Path) {
  120. return s.handleSubscribeRaw(w, r)
  121. } else if (r.Method == http.MethodPut || r.Method == http.MethodPost) && topicRegex.MatchString(r.URL.Path) {
  122. return s.handlePublishHTTP(w, r)
  123. } else if r.Method == http.MethodOptions {
  124. return s.handleOptions(w, r)
  125. }
  126. return errHTTPNotFound
  127. }
  128. func (s *Server) handleHome(w http.ResponseWriter, r *http.Request) error {
  129. _, err := io.WriteString(w, indexSource)
  130. return err
  131. }
  132. func (s *Server) handlePublishHTTP(w http.ResponseWriter, r *http.Request) error {
  133. t, err := s.topic(r.URL.Path[1:])
  134. if err != nil {
  135. return err
  136. }
  137. reader := io.LimitReader(r.Body, messageLimit)
  138. b, err := io.ReadAll(reader)
  139. if err != nil {
  140. return err
  141. }
  142. if err := t.Publish(newDefaultMessage(string(b))); err != nil {
  143. return err
  144. }
  145. w.Header().Set("Access-Control-Allow-Origin", "*") // CORS, allow cross-origin requests
  146. return nil
  147. }
  148. func (s *Server) handleSubscribeJSON(w http.ResponseWriter, r *http.Request) error {
  149. encoder := func(msg *message) (string, error) {
  150. var buf bytes.Buffer
  151. if err := json.NewEncoder(&buf).Encode(&msg); err != nil {
  152. return "", err
  153. }
  154. return buf.String(), nil
  155. }
  156. return s.handleSubscribe(w, r, "json", "application/stream+json", encoder)
  157. }
  158. func (s *Server) handleSubscribeSSE(w http.ResponseWriter, r *http.Request) error {
  159. encoder := func(msg *message) (string, error) {
  160. var buf bytes.Buffer
  161. if err := json.NewEncoder(&buf).Encode(&msg); err != nil {
  162. return "", err
  163. }
  164. if msg.Event != "" {
  165. return fmt.Sprintf("event: %s\ndata: %s\n", msg.Event, buf.String()), nil // Browser's .onmessage() does not fire on this!
  166. }
  167. return fmt.Sprintf("data: %s\n", buf.String()), nil
  168. }
  169. return s.handleSubscribe(w, r, "sse", "text/event-stream", encoder)
  170. }
  171. func (s *Server) handleSubscribeRaw(w http.ResponseWriter, r *http.Request) error {
  172. encoder := func(msg *message) (string, error) {
  173. if msg.Event == "" { // only handle default events
  174. return strings.ReplaceAll(msg.Message, "\n", " ") + "\n", nil
  175. }
  176. return "\n", nil // "keepalive" and "open" events just send an empty line
  177. }
  178. return s.handleSubscribe(w, r, "raw", "text/plain", encoder)
  179. }
  180. func (s *Server) handleSubscribe(w http.ResponseWriter, r *http.Request, format string, contentType string, encoder messageEncoder) error {
  181. t := s.createTopic(strings.TrimSuffix(r.URL.Path[1:], "/"+format)) // Hack
  182. sub := func(msg *message) error {
  183. m, err := encoder(msg)
  184. if err != nil {
  185. return err
  186. }
  187. if _, err := w.Write([]byte(m)); err != nil {
  188. return err
  189. }
  190. if fl, ok := w.(http.Flusher); ok {
  191. fl.Flush()
  192. }
  193. return nil
  194. }
  195. subscriberID := t.Subscribe(sub)
  196. defer s.unsubscribe(t, subscriberID)
  197. w.Header().Set("Access-Control-Allow-Origin", "*") // CORS, allow cross-origin requests
  198. w.Header().Set("Content-Type", contentType)
  199. if err := sub(newOpenMessage()); err != nil { // Send out open message
  200. return err
  201. }
  202. for {
  203. select {
  204. case <-t.ctx.Done():
  205. return nil
  206. case <-r.Context().Done():
  207. return nil
  208. case <-time.After(s.config.KeepaliveInterval):
  209. if err := sub(newKeepaliveMessage()); err != nil { // Send keepalive message
  210. return err
  211. }
  212. }
  213. }
  214. }
  215. func (s *Server) handleOptions(w http.ResponseWriter, r *http.Request) error {
  216. w.Header().Set("Access-Control-Allow-Origin", "*") // CORS, allow cross-origin requests
  217. w.Header().Set("Access-Control-Allow-Methods", "GET, PUT, POST")
  218. return nil
  219. }
  220. func (s *Server) handleStatic(w http.ResponseWriter, r *http.Request) error {
  221. http.FileServer(http.FS(webStaticFs)).ServeHTTP(w, r)
  222. return nil
  223. }
  224. func (s *Server) createTopic(id string) *topic {
  225. s.mu.Lock()
  226. defer s.mu.Unlock()
  227. if _, ok := s.topics[id]; !ok {
  228. s.topics[id] = newTopic(id)
  229. }
  230. return s.topics[id]
  231. }
  232. func (s *Server) topic(topicID string) (*topic, error) {
  233. s.mu.Lock()
  234. defer s.mu.Unlock()
  235. c, ok := s.topics[topicID]
  236. if !ok {
  237. return nil, errHTTPNotFound
  238. }
  239. return c, nil
  240. }
  241. func (s *Server) unsubscribe(t *topic, subscriberID int) {
  242. s.mu.Lock()
  243. defer s.mu.Unlock()
  244. if subscribers := t.Unsubscribe(subscriberID); subscribers == 0 {
  245. delete(s.topics, t.id)
  246. }
  247. }
  248. // visitor creates or retrieves a rate.Limiter for the given visitor.
  249. // This function was taken from https://www.alexedwards.net/blog/how-to-rate-limit-http-requests (MIT).
  250. func (s *Server) visitor(remoteAddr string) *visitor {
  251. s.mu.Lock()
  252. defer s.mu.Unlock()
  253. ip, _, err := net.SplitHostPort(remoteAddr)
  254. if err != nil {
  255. ip = remoteAddr // This should not happen in real life; only in tests.
  256. }
  257. v, exists := s.visitors[ip]
  258. if !exists {
  259. v = &visitor{
  260. rate.NewLimiter(s.config.Limit, s.config.LimitBurst),
  261. time.Now(),
  262. }
  263. s.visitors[ip] = v
  264. return v
  265. }
  266. v.seen = time.Now()
  267. return v
  268. }
  269. func (s *Server) fail(w http.ResponseWriter, r *http.Request, code int, err error) {
  270. log.Printf("[%s] %s - %d - %s", r.RemoteAddr, r.Method, code, err.Error())
  271. w.WriteHeader(code)
  272. io.WriteString(w, fmt.Sprintf("%s\n", http.StatusText(code)))
  273. }