2021-10-23 03:26:01 +02:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2021-10-29 05:50:38 +02:00
|
|
|
"context"
|
2021-12-07 17:45:15 +01:00
|
|
|
"embed" // required for go:embed
|
2021-10-23 03:26:01 +02:00
|
|
|
"encoding/json"
|
2021-10-29 05:50:38 +02:00
|
|
|
firebase "firebase.google.com/go"
|
|
|
|
"firebase.google.com/go/messaging"
|
2021-10-23 19:21:33 +02:00
|
|
|
"fmt"
|
2021-10-29 05:50:38 +02:00
|
|
|
"google.golang.org/api/option"
|
2021-11-08 15:24:34 +01:00
|
|
|
"heckel.io/ntfy/util"
|
|
|
|
"html/template"
|
2021-10-23 03:26:01 +02:00
|
|
|
"io"
|
|
|
|
"log"
|
2021-10-24 04:49:50 +02:00
|
|
|
"net"
|
2021-10-23 03:26:01 +02:00
|
|
|
"net/http"
|
|
|
|
"regexp"
|
2021-10-29 19:58:14 +02:00
|
|
|
"strconv"
|
2021-10-23 03:26:01 +02:00
|
|
|
"strings"
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
2021-10-29 19:58:14 +02:00
|
|
|
// TODO add "max messages in a topic" limit
|
2021-11-01 21:39:40 +01:00
|
|
|
// TODO implement "since=<ID>"
|
2021-10-29 19:58:14 +02:00
|
|
|
|
2021-12-07 17:45:15 +01:00
|
|
|
// Server is the main server, providing the UI and API for ntfy
|
2021-10-23 03:26:01 +02:00
|
|
|
type Server struct {
|
2021-12-19 04:02:36 +01:00
|
|
|
config *Config
|
2021-10-24 04:49:50 +02:00
|
|
|
topics map[string]*topic
|
|
|
|
visitors map[string]*visitor
|
2021-10-29 19:58:14 +02:00
|
|
|
firebase subscriber
|
|
|
|
messages int64
|
2021-11-03 02:09:49 +01:00
|
|
|
cache cache
|
2021-10-24 04:49:50 +02:00
|
|
|
mu sync.Mutex
|
2021-10-23 03:26:01 +02:00
|
|
|
}
|
|
|
|
|
2021-10-24 04:49:50 +02:00
|
|
|
// errHTTP is a generic HTTP error for any non-200 HTTP error
|
|
|
|
type errHTTP struct {
|
|
|
|
Code int
|
|
|
|
Status string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e errHTTP) Error() string {
|
|
|
|
return fmt.Sprintf("http: %s", e.Status)
|
2021-10-23 03:26:01 +02:00
|
|
|
}
|
|
|
|
|
2021-11-08 15:24:34 +01:00
|
|
|
type indexPage struct {
|
|
|
|
Topic string
|
2021-12-09 16:23:17 +01:00
|
|
|
CacheDuration time.Duration
|
2021-11-08 15:24:34 +01:00
|
|
|
}
|
|
|
|
|
2021-11-08 15:46:31 +01:00
|
|
|
type sinceTime time.Time
|
|
|
|
|
|
|
|
func (t sinceTime) IsAll() bool {
|
|
|
|
return t == sinceAllMessages
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t sinceTime) IsNone() bool {
|
|
|
|
return t == sinceNoMessages
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t sinceTime) Time() time.Time {
|
|
|
|
return time.Time(t)
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
sinceAllMessages = sinceTime(time.Unix(0, 0))
|
|
|
|
sinceNoMessages = sinceTime(time.Unix(1, 0))
|
|
|
|
)
|
|
|
|
|
2021-10-23 03:26:01 +02:00
|
|
|
var (
|
2021-11-08 15:24:34 +01:00
|
|
|
topicRegex = regexp.MustCompile(`^/[-_A-Za-z0-9]{1,64}$`) // Regex must match JS & Android app!
|
2021-11-15 13:56:58 +01:00
|
|
|
jsonRegex = regexp.MustCompile(`^/[-_A-Za-z0-9]{1,64}(,[-_A-Za-z0-9]{1,64})*/json$`)
|
|
|
|
sseRegex = regexp.MustCompile(`^/[-_A-Za-z0-9]{1,64}(,[-_A-Za-z0-9]{1,64})*/sse$`)
|
|
|
|
rawRegex = regexp.MustCompile(`^/[-_A-Za-z0-9]{1,64}(,[-_A-Za-z0-9]{1,64})*/raw$`)
|
2021-12-15 22:12:40 +01:00
|
|
|
sendRegex = regexp.MustCompile(`^/[-_A-Za-z0-9]{1,64}(,[-_A-Za-z0-9]{1,64})*/(publish|send|trigger)$`)
|
2021-10-29 19:58:14 +02:00
|
|
|
|
2021-12-09 04:13:59 +01:00
|
|
|
staticRegex = regexp.MustCompile(`^/static/.+`)
|
|
|
|
docsRegex = regexp.MustCompile(`^/docs(|/.*)$`)
|
|
|
|
disallowedTopics = []string{"docs", "static"}
|
2021-10-23 03:26:01 +02:00
|
|
|
|
2021-12-09 16:23:17 +01:00
|
|
|
templateFnMap = template.FuncMap{
|
|
|
|
"durationToHuman": util.DurationToHuman,
|
|
|
|
}
|
|
|
|
|
2021-11-08 15:24:34 +01:00
|
|
|
//go:embed "index.gohtml"
|
|
|
|
indexSource string
|
2021-12-09 16:23:17 +01:00
|
|
|
indexTemplate = template.Must(template.New("index").Funcs(templateFnMap).Parse(indexSource))
|
2021-10-24 03:29:45 +02:00
|
|
|
|
2021-11-18 15:22:33 +01:00
|
|
|
//go:embed "example.html"
|
2021-11-27 22:12:08 +01:00
|
|
|
exampleSource string
|
2021-11-18 15:22:33 +01:00
|
|
|
|
2021-10-24 20:22:53 +02:00
|
|
|
//go:embed static
|
2021-11-29 15:34:43 +01:00
|
|
|
webStaticFs embed.FS
|
|
|
|
webStaticFsCached = &util.CachingEmbedFS{ModTime: time.Now(), FS: webStaticFs}
|
2021-10-24 20:22:53 +02:00
|
|
|
|
2021-12-02 23:27:31 +01:00
|
|
|
//go:embed docs
|
2021-12-07 16:38:58 +01:00
|
|
|
docsStaticFs embed.FS
|
2021-12-02 23:27:31 +01:00
|
|
|
docsStaticCached = &util.CachingEmbedFS{ModTime: time.Now(), FS: docsStaticFs}
|
|
|
|
|
2021-10-29 19:58:14 +02:00
|
|
|
errHTTPBadRequest = &errHTTP{http.StatusBadRequest, http.StatusText(http.StatusBadRequest)}
|
2021-10-24 04:49:50 +02:00
|
|
|
errHTTPNotFound = &errHTTP{http.StatusNotFound, http.StatusText(http.StatusNotFound)}
|
|
|
|
errHTTPTooManyRequests = &errHTTP{http.StatusTooManyRequests, http.StatusText(http.StatusTooManyRequests)}
|
2021-10-23 03:26:01 +02:00
|
|
|
)
|
|
|
|
|
2021-12-14 04:30:28 +01:00
|
|
|
const (
|
|
|
|
firebaseControlTopic = "~control" // See Android if changed
|
|
|
|
)
|
|
|
|
|
2021-12-07 17:45:15 +01:00
|
|
|
// New instantiates a new Server. It creates the cache and adds a Firebase
|
|
|
|
// subscriber (if configured).
|
2021-12-19 04:02:36 +01:00
|
|
|
func New(conf *Config) (*Server, error) {
|
2021-10-29 19:58:14 +02:00
|
|
|
var firebaseSubscriber subscriber
|
2021-10-29 05:50:38 +02:00
|
|
|
if conf.FirebaseKeyFile != "" {
|
2021-10-29 19:58:14 +02:00
|
|
|
var err error
|
|
|
|
firebaseSubscriber, err = createFirebaseSubscriber(conf)
|
2021-10-29 05:50:38 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
2021-11-03 02:09:49 +01:00
|
|
|
cache, err := createCache(conf)
|
2021-11-02 19:08:21 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-11-03 02:09:49 +01:00
|
|
|
topics, err := cache.Topics()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2021-11-02 19:08:21 +01:00
|
|
|
}
|
2021-10-23 03:26:01 +02:00
|
|
|
return &Server{
|
2021-10-24 04:49:50 +02:00
|
|
|
config: conf,
|
2021-11-03 02:09:49 +01:00
|
|
|
cache: cache,
|
2021-10-29 19:58:14 +02:00
|
|
|
firebase: firebaseSubscriber,
|
2021-11-02 19:08:21 +01:00
|
|
|
topics: topics,
|
2021-10-24 04:49:50 +02:00
|
|
|
visitors: make(map[string]*visitor),
|
2021-10-29 05:50:38 +02:00
|
|
|
}, nil
|
2021-10-23 03:26:01 +02:00
|
|
|
}
|
|
|
|
|
2021-12-19 04:02:36 +01:00
|
|
|
func createCache(conf *Config) (cache, error) {
|
2021-12-09 16:23:17 +01:00
|
|
|
if conf.CacheDuration == 0 {
|
|
|
|
return newNopCache(), nil
|
|
|
|
} else if conf.CacheFile != "" {
|
2021-11-03 02:09:49 +01:00
|
|
|
return newSqliteCache(conf.CacheFile)
|
2021-11-02 19:08:21 +01:00
|
|
|
}
|
2021-11-03 02:09:49 +01:00
|
|
|
return newMemCache(), nil
|
2021-11-02 19:08:21 +01:00
|
|
|
}
|
|
|
|
|
2021-12-19 04:02:36 +01:00
|
|
|
func createFirebaseSubscriber(conf *Config) (subscriber, error) {
|
2021-10-29 19:58:14 +02:00
|
|
|
fb, err := firebase.NewApp(context.Background(), nil, option.WithCredentialsFile(conf.FirebaseKeyFile))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
msg, err := fb.Messaging(context.Background())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return func(m *message) error {
|
2021-12-14 04:30:28 +01:00
|
|
|
var data map[string]string // Matches https://ntfy.sh/docs/subscribe/api/#json-message-format
|
|
|
|
switch m.Event {
|
|
|
|
case keepaliveEvent, openEvent:
|
|
|
|
data = map[string]string{
|
|
|
|
"id": m.ID,
|
|
|
|
"time": fmt.Sprintf("%d", m.Time),
|
|
|
|
"event": m.Event,
|
|
|
|
"topic": m.Topic,
|
|
|
|
}
|
|
|
|
case messageEvent:
|
|
|
|
data = map[string]string{
|
2021-11-27 22:12:08 +01:00
|
|
|
"id": m.ID,
|
|
|
|
"time": fmt.Sprintf("%d", m.Time),
|
|
|
|
"event": m.Event,
|
|
|
|
"topic": m.Topic,
|
|
|
|
"priority": fmt.Sprintf("%d", m.Priority),
|
|
|
|
"tags": strings.Join(m.Tags, ","),
|
|
|
|
"title": m.Title,
|
|
|
|
"message": m.Message,
|
2021-12-14 04:30:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
_, err := msg.Send(context.Background(), &messaging.Message{
|
|
|
|
Topic: m.Topic,
|
|
|
|
Data: data,
|
2021-10-29 19:58:14 +02:00
|
|
|
})
|
|
|
|
return err
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2021-12-07 17:45:15 +01:00
|
|
|
// Run executes the main server. It listens on HTTP (+ HTTPS, if configured), and starts
|
|
|
|
// a manager go routine to print stats and prune messages.
|
2021-10-23 03:26:01 +02:00
|
|
|
func (s *Server) Run() error {
|
2021-12-15 15:13:16 +01:00
|
|
|
go s.runManager()
|
|
|
|
go s.runAtSender()
|
|
|
|
go s.runFirebaseKeepliver()
|
2021-12-02 14:52:48 +01:00
|
|
|
listenStr := fmt.Sprintf("%s/http", s.config.ListenHTTP)
|
|
|
|
if s.config.ListenHTTPS != "" {
|
|
|
|
listenStr += fmt.Sprintf(" %s/https", s.config.ListenHTTPS)
|
|
|
|
}
|
|
|
|
log.Printf("Listening on %s", listenStr)
|
2021-10-23 03:26:01 +02:00
|
|
|
http.HandleFunc("/", s.handle)
|
2021-12-02 14:52:48 +01:00
|
|
|
errChan := make(chan error)
|
|
|
|
go func() {
|
|
|
|
errChan <- http.ListenAndServe(s.config.ListenHTTP, nil)
|
|
|
|
}()
|
|
|
|
if s.config.ListenHTTPS != "" {
|
|
|
|
go func() {
|
|
|
|
errChan <- http.ListenAndServeTLS(s.config.ListenHTTPS, s.config.CertFile, s.config.KeyFile, nil)
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
return <-errChan
|
2021-10-23 03:26:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) handle(w http.ResponseWriter, r *http.Request) {
|
|
|
|
if err := s.handleInternal(w, r); err != nil {
|
2021-10-24 04:49:50 +02:00
|
|
|
if e, ok := err.(*errHTTP); ok {
|
|
|
|
s.fail(w, r, e.Code, e)
|
|
|
|
} else {
|
|
|
|
s.fail(w, r, http.StatusInternalServerError, err)
|
|
|
|
}
|
2021-10-23 03:26:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) handleInternal(w http.ResponseWriter, r *http.Request) error {
|
2021-12-02 23:27:31 +01:00
|
|
|
if r.Method == http.MethodGet && r.URL.Path == "/" {
|
2021-10-23 03:26:01 +02:00
|
|
|
return s.handleHome(w, r)
|
2021-11-18 15:22:33 +01:00
|
|
|
} else if r.Method == http.MethodGet && r.URL.Path == "/example.html" {
|
|
|
|
return s.handleExample(w, r)
|
2021-11-05 18:46:27 +01:00
|
|
|
} else if r.Method == http.MethodHead && r.URL.Path == "/" {
|
|
|
|
return s.handleEmpty(w, r)
|
2021-10-24 20:22:53 +02:00
|
|
|
} else if r.Method == http.MethodGet && staticRegex.MatchString(r.URL.Path) {
|
|
|
|
return s.handleStatic(w, r)
|
2021-12-02 23:27:31 +01:00
|
|
|
} else if r.Method == http.MethodGet && docsRegex.MatchString(r.URL.Path) {
|
|
|
|
return s.handleDocs(w, r)
|
2021-11-05 18:46:27 +01:00
|
|
|
} else if r.Method == http.MethodOptions {
|
|
|
|
return s.handleOptions(w, r)
|
2021-12-02 23:27:31 +01:00
|
|
|
} else if r.Method == http.MethodGet && topicRegex.MatchString(r.URL.Path) {
|
|
|
|
return s.handleHome(w, r)
|
2021-10-29 19:58:14 +02:00
|
|
|
} else if (r.Method == http.MethodPut || r.Method == http.MethodPost) && topicRegex.MatchString(r.URL.Path) {
|
2021-11-05 18:46:27 +01:00
|
|
|
return s.withRateLimit(w, r, s.handlePublish)
|
2021-12-15 15:41:55 +01:00
|
|
|
} else if r.Method == http.MethodGet && sendRegex.MatchString(r.URL.Path) {
|
|
|
|
return s.withRateLimit(w, r, s.handlePublish)
|
2021-10-23 03:26:01 +02:00
|
|
|
} else if r.Method == http.MethodGet && jsonRegex.MatchString(r.URL.Path) {
|
2021-11-05 18:46:27 +01:00
|
|
|
return s.withRateLimit(w, r, s.handleSubscribeJSON)
|
2021-10-23 19:21:33 +02:00
|
|
|
} else if r.Method == http.MethodGet && sseRegex.MatchString(r.URL.Path) {
|
2021-11-05 18:46:27 +01:00
|
|
|
return s.withRateLimit(w, r, s.handleSubscribeSSE)
|
2021-10-24 03:29:45 +02:00
|
|
|
} else if r.Method == http.MethodGet && rawRegex.MatchString(r.URL.Path) {
|
2021-11-05 18:46:27 +01:00
|
|
|
return s.withRateLimit(w, r, s.handleSubscribeRaw)
|
2021-10-23 03:26:01 +02:00
|
|
|
}
|
2021-10-24 04:49:50 +02:00
|
|
|
return errHTTPNotFound
|
2021-10-23 03:26:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) handleHome(w http.ResponseWriter, r *http.Request) error {
|
2021-11-08 15:24:34 +01:00
|
|
|
return indexTemplate.Execute(w, &indexPage{
|
|
|
|
Topic: r.URL.Path[1:],
|
2021-12-09 16:23:17 +01:00
|
|
|
CacheDuration: s.config.CacheDuration,
|
2021-11-08 15:24:34 +01:00
|
|
|
})
|
2021-10-23 03:26:01 +02:00
|
|
|
}
|
|
|
|
|
2021-12-07 17:45:15 +01:00
|
|
|
func (s *Server) handleEmpty(_ http.ResponseWriter, _ *http.Request) error {
|
2021-11-05 18:46:27 +01:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-12-07 17:45:15 +01:00
|
|
|
func (s *Server) handleExample(w http.ResponseWriter, _ *http.Request) error {
|
2021-11-18 15:22:33 +01:00
|
|
|
_, err := io.WriteString(w, exampleSource)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-10-29 19:58:14 +02:00
|
|
|
func (s *Server) handleStatic(w http.ResponseWriter, r *http.Request) error {
|
2021-11-29 15:34:43 +01:00
|
|
|
http.FileServer(http.FS(webStaticFsCached)).ServeHTTP(w, r)
|
2021-10-29 19:58:14 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-12-02 23:27:31 +01:00
|
|
|
func (s *Server) handleDocs(w http.ResponseWriter, r *http.Request) error {
|
|
|
|
http.FileServer(http.FS(docsStaticCached)).ServeHTTP(w, r)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-12-07 17:45:15 +01:00
|
|
|
func (s *Server) handlePublish(w http.ResponseWriter, r *http.Request, _ *visitor) error {
|
2021-12-15 15:41:55 +01:00
|
|
|
t, err := s.topicFromPath(r.URL.Path)
|
2021-11-01 21:39:40 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-12-11 04:57:01 +01:00
|
|
|
reader := io.LimitReader(r.Body, int64(s.config.MessageLimit))
|
2021-10-23 03:26:01 +02:00
|
|
|
b, err := io.ReadAll(reader)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-12-15 15:41:55 +01:00
|
|
|
m := newDefaultMessage(t.ID, strings.TrimSpace(string(b)))
|
|
|
|
cache, firebase, err := s.parseParams(r, m)
|
2021-12-10 17:31:42 +01:00
|
|
|
if err != nil {
|
2021-10-29 05:50:38 +02:00
|
|
|
return err
|
|
|
|
}
|
2021-12-15 15:41:55 +01:00
|
|
|
if m.Message == "" {
|
|
|
|
m.Message = "triggered"
|
|
|
|
}
|
2021-12-10 17:31:42 +01:00
|
|
|
delayed := m.Time > time.Now().Unix()
|
|
|
|
if !delayed {
|
|
|
|
if err := t.Publish(m); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if s.firebase != nil && firebase && !delayed {
|
2021-12-09 18:15:17 +01:00
|
|
|
go func() {
|
|
|
|
if err := s.firebase(m); err != nil {
|
|
|
|
log.Printf("Unable to publish to Firebase: %v", err.Error())
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
2021-12-09 16:23:17 +01:00
|
|
|
if cache {
|
|
|
|
if err := s.cache.AddMessage(m); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-11-02 19:08:21 +01:00
|
|
|
}
|
2021-12-15 22:12:40 +01:00
|
|
|
w.Header().Set("Content-Type", "application/json")
|
2021-10-24 19:34:15 +02:00
|
|
|
w.Header().Set("Access-Control-Allow-Origin", "*") // CORS, allow cross-origin requests
|
2021-11-03 16:33:34 +01:00
|
|
|
if err := json.NewEncoder(w).Encode(m); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-12-15 22:12:40 +01:00
|
|
|
s.inc(&s.messages)
|
2021-10-24 19:34:15 +02:00
|
|
|
return nil
|
2021-10-23 03:26:01 +02:00
|
|
|
}
|
|
|
|
|
2021-12-15 15:41:55 +01:00
|
|
|
func (s *Server) parseParams(r *http.Request, m *message) (cache bool, firebase bool, err error) {
|
|
|
|
cache = readParam(r, "x-cache", "cache") != "no"
|
|
|
|
firebase = readParam(r, "x-firebase", "firebase") != "no"
|
2021-12-22 09:44:16 +01:00
|
|
|
m.Title = readParam(r, "x-title", "title", "t")
|
2021-12-15 15:41:55 +01:00
|
|
|
messageStr := readParam(r, "x-message", "message", "m")
|
|
|
|
if messageStr != "" {
|
|
|
|
m.Message = messageStr
|
|
|
|
}
|
2021-12-17 02:33:01 +01:00
|
|
|
m.Priority, err = util.ParsePriority(readParam(r, "x-priority", "priority", "prio", "p"))
|
|
|
|
if err != nil {
|
|
|
|
return false, false, errHTTPBadRequest
|
2021-11-27 22:12:08 +01:00
|
|
|
}
|
2021-12-22 09:44:16 +01:00
|
|
|
tagsStr := readParam(r, "x-tags", "tags", "tag", "ta")
|
2021-11-27 22:12:08 +01:00
|
|
|
if tagsStr != "" {
|
2021-12-10 17:31:42 +01:00
|
|
|
m.Tags = make([]string, 0)
|
2021-12-21 21:22:27 +01:00
|
|
|
for _, s := range util.SplitNoEmpty(tagsStr, ",") {
|
2021-12-10 17:31:42 +01:00
|
|
|
m.Tags = append(m.Tags, strings.TrimSpace(s))
|
2021-12-07 21:39:42 +01:00
|
|
|
}
|
2021-11-27 22:12:08 +01:00
|
|
|
}
|
2021-12-15 15:41:55 +01:00
|
|
|
delayStr := readParam(r, "x-delay", "delay", "x-at", "at", "x-in", "in")
|
2021-12-11 06:06:25 +01:00
|
|
|
if delayStr != "" {
|
2021-12-10 17:31:42 +01:00
|
|
|
if !cache {
|
|
|
|
return false, false, errHTTPBadRequest
|
|
|
|
}
|
2021-12-11 06:06:25 +01:00
|
|
|
delay, err := util.ParseFutureTime(delayStr, time.Now())
|
2021-12-10 17:31:42 +01:00
|
|
|
if err != nil {
|
|
|
|
return false, false, errHTTPBadRequest
|
2021-12-11 06:06:25 +01:00
|
|
|
} else if delay.Unix() < time.Now().Add(s.config.MinDelay).Unix() {
|
2021-12-11 04:57:01 +01:00
|
|
|
return false, false, errHTTPBadRequest
|
2021-12-11 06:06:25 +01:00
|
|
|
} else if delay.Unix() > time.Now().Add(s.config.MaxDelay).Unix() {
|
2021-12-10 17:31:42 +01:00
|
|
|
return false, false, errHTTPBadRequest
|
|
|
|
}
|
2021-12-11 06:06:25 +01:00
|
|
|
m.Time = delay.Unix()
|
2021-12-10 17:31:42 +01:00
|
|
|
}
|
|
|
|
return cache, firebase, nil
|
2021-11-27 22:12:08 +01:00
|
|
|
}
|
|
|
|
|
2021-12-15 15:41:55 +01:00
|
|
|
func readParam(r *http.Request, names ...string) string {
|
2021-11-27 22:12:08 +01:00
|
|
|
for _, name := range names {
|
2021-12-15 15:41:55 +01:00
|
|
|
value := r.Header.Get(name)
|
|
|
|
if value != "" {
|
|
|
|
return strings.TrimSpace(value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, name := range names {
|
|
|
|
value := r.URL.Query().Get(strings.ToLower(name))
|
2021-11-27 22:12:08 +01:00
|
|
|
if value != "" {
|
2021-12-07 21:39:42 +01:00
|
|
|
return strings.TrimSpace(value)
|
2021-11-27 22:12:08 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2021-11-01 20:21:38 +01:00
|
|
|
func (s *Server) handleSubscribeJSON(w http.ResponseWriter, r *http.Request, v *visitor) error {
|
2021-10-27 20:56:17 +02:00
|
|
|
encoder := func(msg *message) (string, error) {
|
|
|
|
var buf bytes.Buffer
|
|
|
|
if err := json.NewEncoder(&buf).Encode(&msg); err != nil {
|
|
|
|
return "", err
|
2021-10-23 03:26:01 +02:00
|
|
|
}
|
2021-10-27 20:56:17 +02:00
|
|
|
return buf.String(), nil
|
2021-10-23 03:26:01 +02:00
|
|
|
}
|
2021-11-07 19:08:03 +01:00
|
|
|
return s.handleSubscribe(w, r, v, "json", "application/x-ndjson", encoder)
|
2021-10-23 03:26:01 +02:00
|
|
|
}
|
|
|
|
|
2021-11-01 20:21:38 +01:00
|
|
|
func (s *Server) handleSubscribeSSE(w http.ResponseWriter, r *http.Request, v *visitor) error {
|
2021-10-27 20:56:17 +02:00
|
|
|
encoder := func(msg *message) (string, error) {
|
2021-10-23 19:21:33 +02:00
|
|
|
var buf bytes.Buffer
|
|
|
|
if err := json.NewEncoder(&buf).Encode(&msg); err != nil {
|
2021-10-27 20:56:17 +02:00
|
|
|
return "", err
|
2021-10-23 19:21:33 +02:00
|
|
|
}
|
2021-10-29 14:29:27 +02:00
|
|
|
if msg.Event != messageEvent {
|
2021-10-27 20:56:17 +02:00
|
|
|
return fmt.Sprintf("event: %s\ndata: %s\n", msg.Event, buf.String()), nil // Browser's .onmessage() does not fire on this!
|
2021-10-23 19:21:33 +02:00
|
|
|
}
|
2021-10-27 20:56:17 +02:00
|
|
|
return fmt.Sprintf("data: %s\n", buf.String()), nil
|
2021-10-23 21:22:17 +02:00
|
|
|
}
|
2021-11-01 20:21:38 +01:00
|
|
|
return s.handleSubscribe(w, r, v, "sse", "text/event-stream", encoder)
|
2021-10-23 19:21:33 +02:00
|
|
|
}
|
|
|
|
|
2021-11-01 20:21:38 +01:00
|
|
|
func (s *Server) handleSubscribeRaw(w http.ResponseWriter, r *http.Request, v *visitor) error {
|
2021-10-27 20:56:17 +02:00
|
|
|
encoder := func(msg *message) (string, error) {
|
2021-11-02 19:10:56 +01:00
|
|
|
if msg.Event == messageEvent { // only handle default events
|
2021-10-27 20:56:17 +02:00
|
|
|
return strings.ReplaceAll(msg.Message, "\n", " ") + "\n", nil
|
|
|
|
}
|
|
|
|
return "\n", nil // "keepalive" and "open" events just send an empty line
|
|
|
|
}
|
2021-11-01 20:21:38 +01:00
|
|
|
return s.handleSubscribe(w, r, v, "raw", "text/plain", encoder)
|
2021-10-27 20:56:17 +02:00
|
|
|
}
|
|
|
|
|
2021-11-01 20:21:38 +01:00
|
|
|
func (s *Server) handleSubscribe(w http.ResponseWriter, r *http.Request, v *visitor, format string, contentType string, encoder messageEncoder) error {
|
|
|
|
if err := v.AddSubscription(); err != nil {
|
2021-11-01 21:39:40 +01:00
|
|
|
return errHTTPTooManyRequests
|
2021-11-01 20:21:38 +01:00
|
|
|
}
|
|
|
|
defer v.RemoveSubscription()
|
2021-11-15 13:56:58 +01:00
|
|
|
topicsStr := strings.TrimSuffix(r.URL.Path[1:], "/"+format) // Hack
|
2021-12-21 21:22:27 +01:00
|
|
|
topicIDs := util.SplitNoEmpty(topicsStr, ",")
|
2021-11-15 13:56:58 +01:00
|
|
|
topics, err := s.topicsFromIDs(topicIDs...)
|
2021-11-01 21:39:40 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-12-22 09:44:16 +01:00
|
|
|
poll := readParam(r, "x-poll", "poll", "po") == "1"
|
|
|
|
scheduled := readParam(r, "x-scheduled", "scheduled", "sched") == "1"
|
|
|
|
since, err := parseSince(r, poll)
|
2021-10-29 19:58:14 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-12-21 21:22:27 +01:00
|
|
|
messageFilter, titleFilter, priorityFilter, tagsFilter, err := parseQueryFilters(r)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-12-22 09:44:16 +01:00
|
|
|
var wlock sync.Mutex
|
2021-10-27 20:56:17 +02:00
|
|
|
sub := func(msg *message) error {
|
2021-12-21 21:22:27 +01:00
|
|
|
if !passesQueryFilter(msg, messageFilter, titleFilter, priorityFilter, tagsFilter) {
|
|
|
|
return nil
|
|
|
|
}
|
2021-10-27 20:56:17 +02:00
|
|
|
m, err := encoder(msg)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-12-21 21:22:27 +01:00
|
|
|
wlock.Lock()
|
|
|
|
defer wlock.Unlock()
|
2021-10-27 20:56:17 +02:00
|
|
|
if _, err := w.Write([]byte(m)); err != nil {
|
2021-10-24 03:29:45 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
if fl, ok := w.(http.Flusher); ok {
|
|
|
|
fl.Flush()
|
|
|
|
}
|
|
|
|
return nil
|
2021-10-27 20:56:17 +02:00
|
|
|
}
|
2021-11-07 19:08:03 +01:00
|
|
|
w.Header().Set("Access-Control-Allow-Origin", "*") // CORS, allow cross-origin requests
|
|
|
|
w.Header().Set("Content-Type", contentType+"; charset=utf-8") // Android/Volley client needs charset!
|
2021-10-29 19:58:14 +02:00
|
|
|
if poll {
|
2021-12-10 17:31:42 +01:00
|
|
|
return s.sendOldMessages(topics, since, scheduled, sub)
|
2021-10-29 19:58:14 +02:00
|
|
|
}
|
2021-11-15 13:56:58 +01:00
|
|
|
subscriberIDs := make([]int, 0)
|
|
|
|
for _, t := range topics {
|
|
|
|
subscriberIDs = append(subscriberIDs, t.Subscribe(sub))
|
|
|
|
}
|
|
|
|
defer func() {
|
|
|
|
for i, subscriberID := range subscriberIDs {
|
|
|
|
topics[i].Unsubscribe(subscriberID) // Order!
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
if err := sub(newOpenMessage(topicsStr)); err != nil { // Send out open message
|
2021-10-29 19:58:14 +02:00
|
|
|
return err
|
|
|
|
}
|
2021-12-10 17:31:42 +01:00
|
|
|
if err := s.sendOldMessages(topics, since, scheduled, sub); err != nil {
|
2021-10-27 20:56:17 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-r.Context().Done():
|
|
|
|
return nil
|
|
|
|
case <-time.After(s.config.KeepaliveInterval):
|
2021-11-01 20:21:38 +01:00
|
|
|
v.Keepalive()
|
2021-11-15 13:56:58 +01:00
|
|
|
if err := sub(newKeepaliveMessage(topicsStr)); err != nil { // Send keepalive message
|
2021-10-27 20:56:17 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2021-10-24 03:29:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-21 21:22:27 +01:00
|
|
|
func parseQueryFilters(r *http.Request) (messageFilter string, titleFilter string, priorityFilter int, tagsFilter []string, err error) {
|
2021-12-22 09:44:16 +01:00
|
|
|
messageFilter = readParam(r, "x-message", "message", "m")
|
|
|
|
titleFilter = readParam(r, "x-title", "title", "t")
|
|
|
|
tagsFilter = util.SplitNoEmpty(readParam(r, "x-tags", "tags", "tag", "ta"), ",")
|
|
|
|
priorityFilter, err = util.ParsePriority(readParam(r, "x-priority", "priority", "prio", "p"))
|
|
|
|
return // may be err!
|
2021-12-21 21:22:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func passesQueryFilter(msg *message, messageFilter string, titleFilter string, priorityFilter int, tagsFilter []string) bool {
|
2021-12-21 21:29:37 +01:00
|
|
|
if msg.Event != messageEvent {
|
|
|
|
return true // filters only apply to messages
|
|
|
|
}
|
2021-12-21 21:22:27 +01:00
|
|
|
if messageFilter != "" && msg.Message != messageFilter {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if titleFilter != "" && msg.Title != titleFilter {
|
|
|
|
return false
|
|
|
|
}
|
2021-12-22 09:44:16 +01:00
|
|
|
messagePriority := msg.Priority
|
|
|
|
if messagePriority == 0 {
|
|
|
|
messagePriority = 3 // For query filters, default priority (3) is the same as "not set" (0)
|
|
|
|
}
|
|
|
|
if priorityFilter > 0 && messagePriority != priorityFilter {
|
2021-12-21 21:22:27 +01:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
if len(tagsFilter) > 0 && !util.InStringListAll(msg.Tags, tagsFilter) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2021-12-10 17:31:42 +01:00
|
|
|
func (s *Server) sendOldMessages(topics []*topic, since sinceTime, scheduled bool, sub subscriber) error {
|
2021-11-08 15:46:31 +01:00
|
|
|
if since.IsNone() {
|
2021-10-29 19:58:14 +02:00
|
|
|
return nil
|
|
|
|
}
|
2021-11-15 13:56:58 +01:00
|
|
|
for _, t := range topics {
|
2021-12-10 17:31:42 +01:00
|
|
|
messages, err := s.cache.Messages(t.ID, since, scheduled)
|
2021-11-15 13:56:58 +01:00
|
|
|
if err != nil {
|
2021-10-29 19:58:14 +02:00
|
|
|
return err
|
|
|
|
}
|
2021-11-15 13:56:58 +01:00
|
|
|
for _, m := range messages {
|
|
|
|
if err := sub(m); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2021-10-29 19:58:14 +02:00
|
|
|
}
|
2021-10-24 19:34:15 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-11-08 15:46:31 +01:00
|
|
|
// parseSince returns a timestamp identifying the time span from which cached messages should be received.
|
|
|
|
//
|
|
|
|
// Values in the "since=..." parameter can be either a unix timestamp or a duration (e.g. 12h), or
|
|
|
|
// "all" for all messages.
|
2021-12-22 09:44:16 +01:00
|
|
|
func parseSince(r *http.Request, poll bool) (sinceTime, error) {
|
|
|
|
since := readParam(r, "x-since", "since", "si")
|
|
|
|
if since == "" {
|
|
|
|
if poll {
|
2021-11-08 15:46:31 +01:00
|
|
|
return sinceAllMessages, nil
|
|
|
|
}
|
|
|
|
return sinceNoMessages, nil
|
|
|
|
}
|
2021-12-22 09:44:16 +01:00
|
|
|
if since == "all" {
|
2021-11-08 15:46:31 +01:00
|
|
|
return sinceAllMessages, nil
|
2021-12-22 09:44:16 +01:00
|
|
|
} else if s, err := strconv.ParseInt(since, 10, 64); err == nil {
|
2021-11-08 15:46:31 +01:00
|
|
|
return sinceTime(time.Unix(s, 0)), nil
|
2021-12-22 09:44:16 +01:00
|
|
|
} else if d, err := time.ParseDuration(since); err == nil {
|
2021-11-08 15:46:31 +01:00
|
|
|
return sinceTime(time.Now().Add(-1 * d)), nil
|
2021-10-29 19:58:14 +02:00
|
|
|
}
|
2021-11-08 15:46:31 +01:00
|
|
|
return sinceNoMessages, errHTTPBadRequest
|
2021-10-29 19:58:14 +02:00
|
|
|
}
|
|
|
|
|
2021-12-07 17:45:15 +01:00
|
|
|
func (s *Server) handleOptions(w http.ResponseWriter, _ *http.Request) error {
|
2021-10-29 19:58:14 +02:00
|
|
|
w.Header().Set("Access-Control-Allow-Origin", "*") // CORS, allow cross-origin requests
|
|
|
|
w.Header().Set("Access-Control-Allow-Methods", "GET, PUT, POST")
|
2021-10-24 20:22:53 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-12-15 15:41:55 +01:00
|
|
|
func (s *Server) topicFromPath(path string) (*topic, error) {
|
|
|
|
parts := strings.Split(path, "/")
|
|
|
|
if len(parts) < 2 {
|
|
|
|
return nil, errHTTPBadRequest
|
|
|
|
}
|
|
|
|
topics, err := s.topicsFromIDs(parts[1])
|
2021-11-15 13:56:58 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return topics[0], nil
|
|
|
|
}
|
|
|
|
|
2021-11-27 22:12:08 +01:00
|
|
|
func (s *Server) topicsFromIDs(ids ...string) ([]*topic, error) {
|
2021-10-23 03:26:01 +02:00
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
2021-11-15 13:56:58 +01:00
|
|
|
topics := make([]*topic, 0)
|
2021-11-27 22:12:08 +01:00
|
|
|
for _, id := range ids {
|
2021-12-09 04:13:59 +01:00
|
|
|
if util.InStringList(disallowedTopics, id) {
|
|
|
|
return nil, errHTTPBadRequest
|
|
|
|
}
|
2021-11-15 13:56:58 +01:00
|
|
|
if _, ok := s.topics[id]; !ok {
|
|
|
|
if len(s.topics) >= s.config.GlobalTopicLimit {
|
|
|
|
return nil, errHTTPTooManyRequests
|
|
|
|
}
|
2021-12-09 04:57:31 +01:00
|
|
|
s.topics[id] = newTopic(id)
|
2021-10-29 19:58:14 +02:00
|
|
|
}
|
2021-11-15 13:56:58 +01:00
|
|
|
topics = append(topics, s.topics[id])
|
2021-10-23 03:26:01 +02:00
|
|
|
}
|
2021-11-15 13:56:58 +01:00
|
|
|
return topics, nil
|
2021-10-23 03:26:01 +02:00
|
|
|
}
|
|
|
|
|
2021-12-11 04:57:01 +01:00
|
|
|
func (s *Server) updateStatsAndPrune() {
|
2021-10-23 03:26:01 +02:00
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
2021-10-29 19:58:14 +02:00
|
|
|
|
|
|
|
// Expire visitors from rate visitors map
|
|
|
|
for ip, v := range s.visitors {
|
2021-11-01 20:21:38 +01:00
|
|
|
if v.Stale() {
|
2021-10-29 19:58:14 +02:00
|
|
|
delete(s.visitors, ip)
|
|
|
|
}
|
2021-10-23 03:26:01 +02:00
|
|
|
}
|
2021-10-24 03:29:45 +02:00
|
|
|
|
2021-12-11 04:57:01 +01:00
|
|
|
// Prune message cache
|
2021-12-09 04:57:31 +01:00
|
|
|
olderThan := time.Now().Add(-1 * s.config.CacheDuration)
|
|
|
|
if err := s.cache.Prune(olderThan); err != nil {
|
2021-11-03 02:09:49 +01:00
|
|
|
log.Printf("error pruning cache: %s", err.Error())
|
2021-11-02 19:08:21 +01:00
|
|
|
}
|
|
|
|
|
2021-12-11 04:57:01 +01:00
|
|
|
// Prune old topics, remove subscriptions without subscribers
|
2021-11-03 02:09:49 +01:00
|
|
|
var subscribers, messages int
|
2021-10-29 19:58:14 +02:00
|
|
|
for _, t := range s.topics {
|
2021-11-03 02:09:49 +01:00
|
|
|
subs := t.Subscribers()
|
2021-12-09 04:57:31 +01:00
|
|
|
msgs, err := s.cache.MessageCount(t.ID)
|
2021-11-03 02:09:49 +01:00
|
|
|
if err != nil {
|
2021-12-09 04:57:31 +01:00
|
|
|
log.Printf("cannot get stats for topic %s: %s", t.ID, err.Error())
|
2021-11-03 02:09:49 +01:00
|
|
|
continue
|
|
|
|
}
|
2021-12-09 18:15:17 +01:00
|
|
|
if msgs == 0 && subs == 0 {
|
2021-12-09 04:57:31 +01:00
|
|
|
delete(s.topics, t.ID)
|
2021-11-03 02:09:49 +01:00
|
|
|
continue
|
2021-10-29 19:58:14 +02:00
|
|
|
}
|
|
|
|
subscribers += subs
|
|
|
|
messages += msgs
|
2021-10-24 03:29:45 +02:00
|
|
|
}
|
2021-11-03 02:09:49 +01:00
|
|
|
|
|
|
|
// Print stats
|
2021-10-29 19:58:14 +02:00
|
|
|
log.Printf("Stats: %d message(s) published, %d topic(s) active, %d subscriber(s), %d message(s) buffered, %d visitor(s)",
|
|
|
|
s.messages, len(s.topics), subscribers, messages, len(s.visitors))
|
2021-10-24 03:29:45 +02:00
|
|
|
}
|
2021-10-24 04:49:50 +02:00
|
|
|
|
2021-12-15 15:13:16 +01:00
|
|
|
func (s *Server) runManager() {
|
|
|
|
func() {
|
|
|
|
ticker := time.NewTicker(s.config.ManagerInterval)
|
|
|
|
for {
|
|
|
|
<-ticker.C
|
|
|
|
s.updateStatsAndPrune()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) runAtSender() {
|
|
|
|
ticker := time.NewTicker(s.config.AtSenderInterval)
|
|
|
|
for {
|
|
|
|
<-ticker.C
|
|
|
|
if err := s.sendDelayedMessages(); err != nil {
|
|
|
|
log.Printf("error sending scheduled messages: %s", err.Error())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Server) runFirebaseKeepliver() {
|
|
|
|
if s.firebase == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ticker := time.NewTicker(s.config.FirebaseKeepaliveInterval)
|
|
|
|
for {
|
|
|
|
<-ticker.C
|
|
|
|
if err := s.firebase(newKeepaliveMessage(firebaseControlTopic)); err != nil {
|
|
|
|
log.Printf("error sending Firebase keepalive message: %s", err.Error())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-12-10 17:31:42 +01:00
|
|
|
func (s *Server) sendDelayedMessages() error {
|
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
messages, err := s.cache.MessagesDue()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
for _, m := range messages {
|
|
|
|
t, ok := s.topics[m.Topic] // If no subscribers, just mark message as published
|
|
|
|
if ok {
|
|
|
|
if err := t.Publish(m); err != nil {
|
|
|
|
log.Printf("unable to publish message %s to topic %s: %v", m.ID, m.Topic, err.Error())
|
|
|
|
}
|
|
|
|
if s.firebase != nil {
|
|
|
|
if err := s.firebase(m); err != nil {
|
|
|
|
log.Printf("unable to publish to Firebase: %v", err.Error())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if err := s.cache.MarkPublished(m); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-11-05 18:46:27 +01:00
|
|
|
func (s *Server) withRateLimit(w http.ResponseWriter, r *http.Request, handler func(w http.ResponseWriter, r *http.Request, v *visitor) error) error {
|
|
|
|
v := s.visitor(r)
|
|
|
|
if err := v.RequestAllowed(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return handler(w, r, v)
|
|
|
|
}
|
|
|
|
|
2021-10-24 04:49:50 +02:00
|
|
|
// visitor creates or retrieves a rate.Limiter for the given visitor.
|
|
|
|
// This function was taken from https://www.alexedwards.net/blog/how-to-rate-limit-http-requests (MIT).
|
2021-11-05 18:46:27 +01:00
|
|
|
func (s *Server) visitor(r *http.Request) *visitor {
|
2021-10-24 04:49:50 +02:00
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
2021-11-05 18:46:27 +01:00
|
|
|
remoteAddr := r.RemoteAddr
|
2021-10-24 04:49:50 +02:00
|
|
|
ip, _, err := net.SplitHostPort(remoteAddr)
|
|
|
|
if err != nil {
|
|
|
|
ip = remoteAddr // This should not happen in real life; only in tests.
|
|
|
|
}
|
2021-11-05 18:46:27 +01:00
|
|
|
if s.config.BehindProxy && r.Header.Get("X-Forwarded-For") != "" {
|
|
|
|
ip = r.Header.Get("X-Forwarded-For")
|
|
|
|
}
|
2021-10-24 04:49:50 +02:00
|
|
|
v, exists := s.visitors[ip]
|
|
|
|
if !exists {
|
2021-11-01 20:21:38 +01:00
|
|
|
s.visitors[ip] = newVisitor(s.config)
|
|
|
|
return s.visitors[ip]
|
2021-10-24 04:49:50 +02:00
|
|
|
}
|
2021-12-22 10:04:59 +01:00
|
|
|
v.Keepalive()
|
2021-10-24 04:49:50 +02:00
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
2021-12-15 22:12:40 +01:00
|
|
|
func (s *Server) inc(counter *int64) {
|
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
*counter++
|
|
|
|
}
|
|
|
|
|
2021-10-24 04:49:50 +02:00
|
|
|
func (s *Server) fail(w http.ResponseWriter, r *http.Request, code int, err error) {
|
|
|
|
log.Printf("[%s] %s - %d - %s", r.RemoteAddr, r.Method, code, err.Error())
|
|
|
|
w.WriteHeader(code)
|
2021-12-07 17:45:15 +01:00
|
|
|
_, _ = io.WriteString(w, fmt.Sprintf("%s\n", http.StatusText(code)))
|
2021-10-24 04:49:50 +02:00
|
|
|
}
|