ntfy/server/types.go

216 lines
6.2 KiB
Go
Raw Normal View History

package server
2021-10-29 20:58:14 +03:00
import (
2022-01-16 07:17:46 +03:00
"net/http"
"net/netip"
2021-10-29 20:58:14 +03:00
"time"
"heckel.io/ntfy/util"
2021-10-29 20:58:14 +03:00
)
// List of possible events
const (
openEvent = "open"
keepaliveEvent = "keepalive"
messageEvent = "message"
pollRequestEvent = "poll_request"
2021-10-29 20:58:14 +03:00
)
const (
2022-02-26 23:57:10 +03:00
messageIDLength = 12
)
// message represents a message published to a topic
type message struct {
2022-01-03 01:56:12 +03:00
ID string `json:"id"` // Random message ID
Time int64 `json:"time"` // Unix time in seconds
Event string `json:"event"` // One of the above
Topic string `json:"topic"`
2022-05-27 14:55:57 +03:00
Title string `json:"title,omitempty"`
Message string `json:"message,omitempty"`
2022-01-03 01:56:12 +03:00
Priority int `json:"priority,omitempty"`
Tags []string `json:"tags,omitempty"`
2022-01-05 02:25:49 +03:00
Click string `json:"click,omitempty"`
2022-09-11 23:31:39 +03:00
Icon string `json:"icon,omitempty"`
2022-04-19 16:14:32 +03:00
Actions []*action `json:"actions,omitempty"`
2022-01-05 02:25:49 +03:00
Attachment *attachment `json:"attachment,omitempty"`
2022-05-27 14:55:57 +03:00
PollID string `json:"poll_id,omitempty"`
Sender netip.Addr `json:"-"` // IP address of uploader, used for rate limiting
2022-01-17 21:28:07 +03:00
Encoding string `json:"encoding,omitempty"` // empty for raw UTF-8, or "base64" for encoded bytes
2022-01-03 01:56:12 +03:00
}
type attachment struct {
Name string `json:"name"`
Type string `json:"type,omitempty"`
Size int64 `json:"size,omitempty"`
Expires int64 `json:"expires,omitempty"`
URL string `json:"url"`
}
2022-04-16 23:17:58 +03:00
type action struct {
2022-04-17 21:29:43 +03:00
ID string `json:"id"`
Action string `json:"action"` // "view", "broadcast", or "http"
Label string `json:"label"` // action button label
Clear bool `json:"clear"` // clear notification after successful execution
2022-04-19 16:14:32 +03:00
URL string `json:"url,omitempty"` // used in "view" and "http" actions
Method string `json:"method,omitempty"` // used in "http" action, default is POST (!)
Headers map[string]string `json:"headers,omitempty"` // used in "http" action
Body string `json:"body,omitempty"` // used in "http" action
2022-04-20 02:22:19 +03:00
Intent string `json:"intent,omitempty"` // used in "broadcast" action
2022-04-19 16:14:32 +03:00
Extras map[string]string `json:"extras,omitempty"` // used in "broadcast" action
2022-04-16 23:17:58 +03:00
}
2022-04-27 16:51:23 +03:00
func newAction() *action {
return &action{
Headers: make(map[string]string),
Extras: make(map[string]string),
}
}
2022-03-15 23:00:59 +03:00
// publishMessage is used as input when publishing as JSON
type publishMessage struct {
2022-03-16 21:16:54 +03:00
Topic string `json:"topic"`
Title string `json:"title"`
Message string `json:"message"`
Priority int `json:"priority"`
Tags []string `json:"tags"`
Click string `json:"click"`
2022-09-11 23:31:39 +03:00
Icon string `json:"icon"`
2022-04-16 23:17:58 +03:00
Actions []action `json:"actions"`
2022-03-16 21:16:54 +03:00
Attach string `json:"attach"`
Filename string `json:"filename"`
Email string `json:"email"`
Delay string `json:"delay"`
2022-03-15 23:00:59 +03:00
}
// messageEncoder is a function that knows how to encode a message
type messageEncoder func(msg *message) (string, error)
// newMessage creates a new message with the current timestamp
2021-10-29 20:58:14 +03:00
func newMessage(event, topic, msg string) *message {
return &message{
2022-05-27 14:55:57 +03:00
ID: util.RandomString(messageIDLength),
Time: time.Now().Unix(),
Event: event,
Topic: topic,
Message: msg,
}
}
// newOpenMessage is a convenience method to create an open message
2021-10-29 20:58:14 +03:00
func newOpenMessage(topic string) *message {
return newMessage(openEvent, topic, "")
}
// newKeepaliveMessage is a convenience method to create a keepalive message
2021-10-29 20:58:14 +03:00
func newKeepaliveMessage(topic string) *message {
return newMessage(keepaliveEvent, topic, "")
}
// newDefaultMessage is a convenience method to create a notification message
2021-10-29 20:58:14 +03:00
func newDefaultMessage(topic, msg string) *message {
return newMessage(messageEvent, topic, msg)
}
2022-01-16 07:17:46 +03:00
2022-05-27 14:55:57 +03:00
// newPollRequestMessage is a convenience method to create a poll request message
func newPollRequestMessage(topic, pollID string) *message {
m := newMessage(pollRequestEvent, topic, newMessageBody)
m.PollID = pollID
return m
}
2022-02-26 23:57:10 +03:00
func validMessageID(s string) bool {
return util.ValidRandomString(s, messageIDLength)
}
type sinceMarker struct {
time time.Time
id string
}
2022-01-16 07:17:46 +03:00
2022-02-26 23:57:10 +03:00
func newSinceTime(timestamp int64) sinceMarker {
return sinceMarker{time.Unix(timestamp, 0), ""}
}
func newSinceID(id string) sinceMarker {
return sinceMarker{time.Unix(0, 0), id}
}
func (t sinceMarker) IsAll() bool {
2022-01-16 07:17:46 +03:00
return t == sinceAllMessages
}
2022-02-26 23:57:10 +03:00
func (t sinceMarker) IsNone() bool {
2022-01-16 07:17:46 +03:00
return t == sinceNoMessages
}
2022-02-26 23:57:10 +03:00
func (t sinceMarker) IsID() bool {
return t.id != ""
}
func (t sinceMarker) Time() time.Time {
return t.time
}
func (t sinceMarker) ID() string {
return t.id
2022-01-16 07:17:46 +03:00
}
var (
2022-02-26 23:57:10 +03:00
sinceAllMessages = sinceMarker{time.Unix(0, 0), ""}
sinceNoMessages = sinceMarker{time.Unix(1, 0), ""}
2022-01-16 07:17:46 +03:00
)
type queryFilter struct {
2022-05-27 01:52:55 +03:00
ID string
2022-01-16 07:17:46 +03:00
Message string
Title string
Tags []string
Priority []int
}
func parseQueryFilters(r *http.Request) (*queryFilter, error) {
2022-05-27 01:52:55 +03:00
idFilter := readParam(r, "x-id", "id")
2022-01-16 07:17:46 +03: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 := make([]int, 0)
for _, p := range util.SplitNoEmpty(readParam(r, "x-priority", "priority", "prio", "p"), ",") {
priority, err := util.ParsePriority(p)
if err != nil {
2022-07-01 16:28:42 +03:00
return nil, errHTTPBadRequestPriorityInvalid
2022-01-16 07:17:46 +03:00
}
priorityFilter = append(priorityFilter, priority)
}
return &queryFilter{
2022-05-27 01:52:55 +03:00
ID: idFilter,
2022-01-16 07:17:46 +03:00
Message: messageFilter,
Title: titleFilter,
Tags: tagsFilter,
Priority: priorityFilter,
}, nil
}
func (q *queryFilter) Pass(msg *message) bool {
if msg.Event != messageEvent {
return true // filters only apply to messages
2022-05-27 01:52:55 +03:00
} else if q.ID != "" && msg.ID != q.ID {
2022-01-16 07:17:46 +03:00
return false
2022-05-27 01:52:55 +03:00
} else if q.Message != "" && msg.Message != q.Message {
return false
} else if q.Title != "" && msg.Title != q.Title {
2022-01-16 07:17:46 +03:00
return false
}
messagePriority := msg.Priority
if messagePriority == 0 {
messagePriority = 3 // For query filters, default priority (3) is the same as "not set" (0)
}
2022-10-01 22:50:48 +03:00
if len(q.Priority) > 0 && !util.Contains(q.Priority, messagePriority) {
2022-01-16 07:17:46 +03:00
return false
}
2022-10-01 22:50:48 +03:00
if len(q.Tags) > 0 && !util.ContainsAll(msg.Tags, q.Tags) {
2022-01-16 07:17:46 +03:00
return false
}
return true
}