ntfy/client/client.go

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

293 lines
8.9 KiB
Go
Raw Normal View History

// Package client provides a ntfy client to publish and subscribe to topics
2021-12-17 01:33:01 +00:00
package client
import (
"bufio"
"context"
"encoding/json"
2022-02-02 04:39:57 +00:00
"errors"
2021-12-17 01:33:01 +00:00
"fmt"
2023-11-17 01:54:58 +00:00
"heckel.io/ntfy/v2/log"
"heckel.io/ntfy/v2/util"
"io"
2021-12-17 01:33:01 +00:00
"net/http"
2023-06-01 20:01:39 +00:00
"regexp"
2021-12-17 01:33:01 +00:00
"strings"
"sync"
"time"
)
const (
2023-06-01 20:01:39 +00:00
// MessageEvent identifies a message event
MessageEvent = "message"
2021-12-17 01:33:01 +00:00
)
const (
maxResponseBytes = 4096
)
2023-06-01 20:01:39 +00:00
var (
topicRegex = regexp.MustCompile(`^[-_A-Za-z0-9]{1,64}$`) // Same as in server/server.go
)
// Client is the ntfy client that can be used to publish and subscribe to ntfy topics
2021-12-17 01:33:01 +00:00
type Client struct {
Messages chan *Message
2021-12-18 19:43:27 +00:00
config *Config
2021-12-17 01:33:01 +00:00
subscriptions map[string]*subscription
mu sync.Mutex
}
// Message is a struct that represents a ntfy message
2021-12-22 22:20:43 +00:00
type Message struct { // TODO combine with server.message
2022-01-16 03:33:35 +00:00
ID string
Event string
Time int64
Topic string
Message string
Title string
Priority int
Tags []string
Click string
2022-07-17 21:40:24 +00:00
Icon string
2022-01-16 03:33:35 +00:00
Attachment *Attachment
2021-12-21 20:22:27 +00:00
// Additional fields
TopicURL string
SubscriptionID string
Raw string
2021-12-17 01:33:01 +00:00
}
2022-01-16 04:53:40 +00:00
// Attachment represents a message attachment
2022-01-16 03:33:35 +00: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"`
Owner string `json:"-"` // IP address of uploader, used for rate limiting
}
2021-12-17 01:33:01 +00:00
type subscription struct {
2021-12-21 20:22:27 +00:00
ID string
topicURL string
cancel context.CancelFunc
2021-12-17 01:33:01 +00:00
}
// New creates a new Client using a given Config
2021-12-18 19:43:27 +00:00
func New(config *Config) *Client {
2021-12-17 01:33:01 +00:00
return &Client{
2021-12-22 22:20:43 +00:00
Messages: make(chan *Message, 50), // Allow reading a few messages
2021-12-18 19:43:27 +00:00
config: config,
2021-12-17 01:33:01 +00:00
subscriptions: make(map[string]*subscription),
}
}
// Publish sends a message to a specific topic, optionally using options.
2022-01-13 02:24:48 +00:00
// See PublishReader for details.
func (c *Client) Publish(topic, message string, options ...PublishOption) (*Message, error) {
return c.PublishReader(topic, strings.NewReader(message), options...)
}
// PublishReader sends a message to a specific topic, optionally using options.
//
// A topic can be either a full URL (e.g. https://myhost.lan/mytopic), a short URL which is then prepended https://
// (e.g. myhost.lan -> https://myhost.lan), or a short name which is expanded using the default host in the
// config (e.g. mytopic -> https://ntfy.sh/mytopic).
//
// To pass title, priority and tags, check out WithTitle, WithPriority, WithTagsList, WithDelay, WithNoCache,
// WithNoFirebase, and the generic WithHeader.
2022-01-13 02:24:48 +00:00
func (c *Client) PublishReader(topic string, body io.Reader, options ...PublishOption) (*Message, error) {
2023-06-01 20:01:39 +00:00
topicURL, err := c.expandTopicURL(topic)
if err != nil {
return nil, err
}
2023-06-01 18:08:51 +00:00
req, err := http.NewRequest("POST", topicURL, body)
if err != nil {
return nil, err
}
2021-12-17 01:33:01 +00:00
for _, option := range options {
if err := option(req); err != nil {
return nil, err
2021-12-17 01:33:01 +00:00
}
}
log.Debug("%s Publishing message with headers %s", util.ShortTopicURL(topicURL), req.Header)
2021-12-17 01:33:01 +00:00
resp, err := http.DefaultClient.Do(req)
if err != nil {
return nil, err
2021-12-17 01:33:01 +00:00
}
defer resp.Body.Close()
b, err := io.ReadAll(io.LimitReader(resp.Body, maxResponseBytes))
if err != nil {
return nil, err
}
2022-02-02 04:39:57 +00:00
if resp.StatusCode != http.StatusOK {
return nil, errors.New(strings.TrimSpace(string(b)))
}
2021-12-21 20:22:27 +00:00
m, err := toMessage(string(b), topicURL, "")
if err != nil {
return nil, err
}
return m, nil
2021-12-17 01:33:01 +00:00
}
// Poll queries a topic for all (or a limited set) of messages. Unlike Subscribe, this method only polls for
// messages and does not subscribe to messages that arrive after this call.
//
// A topic can be either a full URL (e.g. https://myhost.lan/mytopic), a short URL which is then prepended https://
// (e.g. myhost.lan -> https://myhost.lan), or a short name which is expanded using the default host in the
// config (e.g. mytopic -> https://ntfy.sh/mytopic).
//
// By default, all messages will be returned, but you can change this behavior using a SubscribeOption.
// See WithSince, WithSinceAll, WithSinceUnixTime, WithScheduled, and the generic WithQueryParam.
2021-12-18 19:43:27 +00:00
func (c *Client) Poll(topic string, options ...SubscribeOption) ([]*Message, error) {
2023-06-01 20:01:39 +00:00
topicURL, err := c.expandTopicURL(topic)
if err != nil {
return nil, err
}
2021-12-17 14:32:59 +00:00
ctx := context.Background()
messages := make([]*Message, 0)
msgChan := make(chan *Message)
errChan := make(chan error)
log.Debug("%s Polling from topic", util.ShortTopicURL(topicURL))
2021-12-22 22:45:19 +00:00
options = append(options, WithPoll())
2021-12-17 14:32:59 +00:00
go func() {
2021-12-21 20:22:27 +00:00
err := performSubscribeRequest(ctx, msgChan, topicURL, "", options...)
2021-12-17 14:32:59 +00:00
close(msgChan)
errChan <- err
}()
for m := range msgChan {
messages = append(messages, m)
}
return messages, <-errChan
}
// Subscribe subscribes to a topic to listen for newly incoming messages. The method starts a connection in the
// background and returns new messages via the Messages channel.
//
// A topic can be either a full URL (e.g. https://myhost.lan/mytopic), a short URL which is then prepended https://
// (e.g. myhost.lan -> https://myhost.lan), or a short name which is expanded using the default host in the
// config (e.g. mytopic -> https://ntfy.sh/mytopic).
//
// By default, only new messages will be returned, but you can change this behavior using a SubscribeOption.
// See WithSince, WithSinceAll, WithSinceUnixTime, WithScheduled, and the generic WithQueryParam.
//
2021-12-21 20:22:27 +00:00
// The method returns a unique subscriptionID that can be used in Unsubscribe.
//
// Example:
2022-09-27 16:37:02 +00:00
//
// c := client.New(client.NewConfig())
2023-06-01 20:01:39 +00:00
// subscriptionID, _ := c.Subscribe("mytopic")
2022-09-27 16:37:02 +00:00
// for m := range c.Messages {
// fmt.Printf("New message: %s", m.Message)
// }
2023-06-01 20:01:39 +00:00
func (c *Client) Subscribe(topic string, options ...SubscribeOption) (string, error) {
topicURL, err := c.expandTopicURL(topic)
if err != nil {
return "", err
}
2021-12-17 01:33:01 +00:00
c.mu.Lock()
defer c.mu.Unlock()
2021-12-21 20:22:27 +00:00
subscriptionID := util.RandomString(10)
log.Debug("%s Subscribing to topic", util.ShortTopicURL(topicURL))
2021-12-17 01:33:01 +00:00
ctx, cancel := context.WithCancel(context.Background())
2021-12-21 20:22:27 +00:00
c.subscriptions[subscriptionID] = &subscription{
ID: subscriptionID,
topicURL: topicURL,
cancel: cancel,
}
go handleSubscribeConnLoop(ctx, c.Messages, topicURL, subscriptionID, options...)
2023-06-01 20:01:39 +00:00
return subscriptionID, nil
2021-12-17 01:33:01 +00:00
}
2021-12-21 20:22:27 +00:00
// Unsubscribe unsubscribes from a topic that has been previously subscribed to using the unique
// subscriptionID returned in Subscribe.
func (c *Client) Unsubscribe(subscriptionID string) {
c.mu.Lock()
defer c.mu.Unlock()
sub, ok := c.subscriptions[subscriptionID]
if !ok {
return
}
delete(c.subscriptions, subscriptionID)
sub.cancel()
}
2023-06-01 20:01:39 +00:00
func (c *Client) expandTopicURL(topic string) (string, error) {
2021-12-18 19:43:27 +00:00
if strings.HasPrefix(topic, "http://") || strings.HasPrefix(topic, "https://") {
2023-06-01 20:01:39 +00:00
return topic, nil
2021-12-18 19:43:27 +00:00
} else if strings.Contains(topic, "/") {
2023-06-01 20:01:39 +00:00
return fmt.Sprintf("https://%s", topic), nil
}
if !topicRegex.MatchString(topic) {
return "", fmt.Errorf("invalid topic name: %s", topic)
2021-12-18 19:43:27 +00:00
}
2023-06-01 20:01:39 +00:00
return fmt.Sprintf("%s/%s", c.config.DefaultHost, topic), nil
2021-12-18 19:43:27 +00:00
}
2021-12-21 20:22:27 +00:00
func handleSubscribeConnLoop(ctx context.Context, msgChan chan *Message, topicURL, subcriptionID string, options ...SubscribeOption) {
2021-12-17 01:33:01 +00:00
for {
2021-12-21 20:22:27 +00:00
// TODO The retry logic is crude and may lose messages. It should record the last message like the
// Android client, use since=, and do incremental backoff too
if err := performSubscribeRequest(ctx, msgChan, topicURL, subcriptionID, options...); err != nil {
log.Warn("%s Connection failed: %s", util.ShortTopicURL(topicURL), err.Error())
2021-12-17 01:33:01 +00:00
}
select {
case <-ctx.Done():
2022-06-02 18:38:38 +00:00
log.Info("%s Connection exited", util.ShortTopicURL(topicURL))
2021-12-17 01:33:01 +00:00
return
case <-time.After(10 * time.Second): // TODO Add incremental backoff
2021-12-17 01:33:01 +00:00
}
}
}
2021-12-21 20:22:27 +00:00
func performSubscribeRequest(ctx context.Context, msgChan chan *Message, topicURL string, subscriptionID string, options ...SubscribeOption) error {
streamURL := fmt.Sprintf("%s/json", topicURL)
log.Debug("%s Listening to %s", util.ShortTopicURL(topicURL), streamURL)
req, err := http.NewRequestWithContext(ctx, http.MethodGet, streamURL, nil)
2021-12-17 01:33:01 +00:00
if err != nil {
return err
}
2021-12-17 14:32:59 +00:00
for _, option := range options {
if err := option(req); err != nil {
return err
}
}
2021-12-17 01:33:01 +00:00
resp, err := http.DefaultClient.Do(req)
if err != nil {
return err
}
defer resp.Body.Close()
2022-02-04 03:26:22 +00:00
if resp.StatusCode != http.StatusOK {
b, err := io.ReadAll(io.LimitReader(resp.Body, maxResponseBytes))
if err != nil {
return err
}
return errors.New(strings.TrimSpace(string(b)))
}
2021-12-17 01:33:01 +00:00
scanner := bufio.NewScanner(resp.Body)
for scanner.Scan() {
messageJSON := scanner.Text()
m, err := toMessage(messageJSON, topicURL, subscriptionID)
if err != nil {
2021-12-17 01:33:01 +00:00
return err
}
log.Trace("%s Message received: %s", util.ShortTopicURL(topicURL), messageJSON)
2021-12-22 22:20:43 +00:00
if m.Event == MessageEvent {
msgChan <- m
}
2021-12-17 01:33:01 +00:00
}
return nil
}
2021-12-21 20:22:27 +00:00
func toMessage(s, topicURL, subscriptionID string) (*Message, error) {
var m *Message
if err := json.NewDecoder(strings.NewReader(s)).Decode(&m); err != nil {
return nil, err
}
m.TopicURL = topicURL
2021-12-21 20:22:27 +00:00
m.SubscriptionID = subscriptionID
m.Raw = s
return m, nil
}