go-mastodon/status.go

340 lines
11 KiB
Go
Raw Normal View History

2017-04-14 17:22:44 +02:00
package mastodon
import (
"context"
2017-04-14 17:22:44 +02:00
"fmt"
2019-04-27 03:56:47 +02:00
"io"
2017-04-14 17:22:44 +02:00
"net/http"
"net/url"
2017-04-25 20:16:32 +02:00
"time"
2017-04-14 17:22:44 +02:00
)
// Status is struct to hold status.
type Status struct {
ID ID `json:"id"`
URI string `json:"uri"`
URL string `json:"url"`
Account Account `json:"account"`
2017-04-14 17:22:44 +02:00
InReplyToID interface{} `json:"in_reply_to_id"`
InReplyToAccountID interface{} `json:"in_reply_to_account_id"`
Reblog *Status `json:"reblog"`
Content string `json:"content"`
CreatedAt time.Time `json:"created_at"`
Emojis []Emoji `json:"emojis"`
RepliesCount int64 `json:"replies_count"`
ReblogsCount int64 `json:"reblogs_count"`
FavouritesCount int64 `json:"favourites_count"`
Reblogged interface{} `json:"reblogged"`
Favourited interface{} `json:"favourited"`
Muted interface{} `json:"muted"`
2017-04-14 17:22:44 +02:00
Sensitive bool `json:"sensitive"`
SpoilerText string `json:"spoiler_text"`
Visibility string `json:"visibility"`
MediaAttachments []Attachment `json:"media_attachments"`
Mentions []Mention `json:"mentions"`
Tags []Tag `json:"tags"`
Card *Card `json:"card"`
Application Application `json:"application"`
Language string `json:"language"`
Pinned interface{} `json:"pinned"`
2017-04-14 17:22:44 +02:00
}
// Context hold information for mastodon context.
type Context struct {
2017-04-16 16:38:53 +02:00
Ancestors []*Status `json:"ancestors"`
Descendants []*Status `json:"descendants"`
2017-04-14 17:22:44 +02:00
}
2017-04-14 17:25:35 +02:00
// Card hold information for mastodon card.
type Card struct {
URL string `json:"url"`
Title string `json:"title"`
Description string `json:"description"`
Image string `json:"image"`
Type string `json:"type"`
AuthorName string `json:"author_name"`
AuthorURL string `json:"author_url"`
ProviderName string `json:"provider_name"`
ProviderURL string `json:"provider_url"`
HTML string `json:"html"`
Width int64 `json:"width"`
Height int64 `json:"height"`
2017-04-14 17:25:35 +02:00
}
// Conversation hold information for mastodon conversation.
type Conversation struct {
ID ID `json:"id"`
Accounts []*Account `json:"accounts"`
Unread bool `json:"unread"`
LastStatus *Status `json:"last_status"`
}
2017-04-14 20:28:50 +02:00
// GetFavourites return the favorite list of the current user.
2017-05-06 16:03:19 +02:00
func (c *Client) GetFavourites(ctx context.Context, pg *Pagination) ([]*Status, error) {
2017-04-14 20:28:50 +02:00
var statuses []*Status
2017-05-06 16:03:19 +02:00
err := c.doAPI(ctx, http.MethodGet, "/api/v1/favourites", nil, &statuses, pg)
2017-04-14 20:28:50 +02:00
if err != nil {
2017-05-06 16:03:19 +02:00
return nil, err
2017-04-14 20:28:50 +02:00
}
2017-05-06 16:03:19 +02:00
return statuses, nil
2017-04-14 20:28:50 +02:00
}
2017-04-14 17:22:44 +02:00
// GetStatus return status specified by id.
2017-10-25 08:22:17 +02:00
func (c *Client) GetStatus(ctx context.Context, id ID) (*Status, error) {
2017-04-14 17:22:44 +02:00
var status Status
2017-10-25 08:22:17 +02:00
err := c.doAPI(ctx, http.MethodGet, fmt.Sprintf("/api/v1/statuses/%s", id), nil, &status, nil)
2017-04-14 17:22:44 +02:00
if err != nil {
return nil, err
}
return &status, nil
}
// GetStatusContext return status specified by id.
2017-10-25 08:22:17 +02:00
func (c *Client) GetStatusContext(ctx context.Context, id ID) (*Context, error) {
2017-04-14 17:22:44 +02:00
var context Context
2017-10-25 08:22:17 +02:00
err := c.doAPI(ctx, http.MethodGet, fmt.Sprintf("/api/v1/statuses/%s/context", id), nil, &context, nil)
2017-04-14 17:22:44 +02:00
if err != nil {
return nil, err
}
return &context, nil
}
2017-04-14 17:25:35 +02:00
// GetStatusCard return status specified by id.
2017-10-25 08:22:17 +02:00
func (c *Client) GetStatusCard(ctx context.Context, id ID) (*Card, error) {
2017-04-14 17:25:35 +02:00
var card Card
2017-10-25 08:22:17 +02:00
err := c.doAPI(ctx, http.MethodGet, fmt.Sprintf("/api/v1/statuses/%s/card", id), nil, &card, nil)
2017-04-14 17:25:35 +02:00
if err != nil {
return nil, err
}
return &card, nil
}
2017-04-16 06:45:59 +02:00
// GetRebloggedBy returns the account list of the user who reblogged the toot of id.
2017-10-25 08:22:17 +02:00
func (c *Client) GetRebloggedBy(ctx context.Context, id ID, pg *Pagination) ([]*Account, error) {
2017-04-16 06:11:56 +02:00
var accounts []*Account
2017-10-25 08:22:17 +02:00
err := c.doAPI(ctx, http.MethodGet, fmt.Sprintf("/api/v1/statuses/%s/reblogged_by", id), nil, &accounts, pg)
2017-04-16 06:11:56 +02:00
if err != nil {
2017-05-06 16:03:19 +02:00
return nil, err
2017-04-16 06:11:56 +02:00
}
2017-05-06 16:03:19 +02:00
return accounts, nil
2017-04-16 06:11:56 +02:00
}
2017-04-16 06:45:59 +02:00
// GetFavouritedBy returns the account list of the user who liked the toot of id.
2017-10-25 08:22:17 +02:00
func (c *Client) GetFavouritedBy(ctx context.Context, id ID, pg *Pagination) ([]*Account, error) {
2017-04-16 06:45:59 +02:00
var accounts []*Account
2017-10-25 08:22:17 +02:00
err := c.doAPI(ctx, http.MethodGet, fmt.Sprintf("/api/v1/statuses/%s/favourited_by", id), nil, &accounts, pg)
2017-04-16 06:45:59 +02:00
if err != nil {
2017-05-06 16:03:19 +02:00
return nil, err
2017-04-16 06:45:59 +02:00
}
2017-05-06 16:03:19 +02:00
return accounts, nil
2017-04-16 06:45:59 +02:00
}
2017-04-16 08:04:59 +02:00
// Reblog is reblog the toot of id and return status of reblog.
2017-10-25 08:22:17 +02:00
func (c *Client) Reblog(ctx context.Context, id ID) (*Status, error) {
2017-04-16 07:53:15 +02:00
var status Status
2017-10-25 08:22:17 +02:00
err := c.doAPI(ctx, http.MethodPost, fmt.Sprintf("/api/v1/statuses/%s/reblog", id), nil, &status, nil)
2017-04-16 07:53:15 +02:00
if err != nil {
return nil, err
}
return &status, nil
}
2017-04-16 08:04:59 +02:00
// Unreblog is unreblog the toot of id and return status of the original toot.
2017-10-25 08:22:17 +02:00
func (c *Client) Unreblog(ctx context.Context, id ID) (*Status, error) {
2017-04-16 07:53:15 +02:00
var status Status
2017-10-25 08:22:17 +02:00
err := c.doAPI(ctx, http.MethodPost, fmt.Sprintf("/api/v1/statuses/%s/unreblog", id), nil, &status, nil)
2017-04-16 07:53:15 +02:00
if err != nil {
return nil, err
}
return &status, nil
}
2017-04-16 08:32:48 +02:00
// Favourite is favourite the toot of id and return status of the favourite toot.
2017-10-25 08:22:17 +02:00
func (c *Client) Favourite(ctx context.Context, id ID) (*Status, error) {
2017-04-16 08:32:48 +02:00
var status Status
2017-10-25 08:22:17 +02:00
err := c.doAPI(ctx, http.MethodPost, fmt.Sprintf("/api/v1/statuses/%s/favourite", id), nil, &status, nil)
2017-04-16 08:32:48 +02:00
if err != nil {
return nil, err
}
return &status, nil
}
// Unfavourite is unfavourite the toot of id and return status of the unfavourite toot.
2017-10-25 08:22:17 +02:00
func (c *Client) Unfavourite(ctx context.Context, id ID) (*Status, error) {
2017-04-16 08:32:48 +02:00
var status Status
2017-10-25 08:22:17 +02:00
err := c.doAPI(ctx, http.MethodPost, fmt.Sprintf("/api/v1/statuses/%s/unfavourite", id), nil, &status, nil)
2017-04-16 08:32:48 +02:00
if err != nil {
return nil, err
}
return &status, nil
}
2017-04-14 17:22:44 +02:00
// GetTimelineHome return statuses from home timeline.
2017-05-06 16:03:19 +02:00
func (c *Client) GetTimelineHome(ctx context.Context, pg *Pagination) ([]*Status, error) {
2017-04-14 17:22:44 +02:00
var statuses []*Status
2017-05-06 16:03:19 +02:00
err := c.doAPI(ctx, http.MethodGet, "/api/v1/timelines/home", nil, &statuses, pg)
2017-04-14 17:22:44 +02:00
if err != nil {
2017-05-06 16:03:19 +02:00
return nil, err
2017-04-14 17:22:44 +02:00
}
2017-05-06 16:03:19 +02:00
return statuses, nil
2017-04-25 20:16:32 +02:00
}
// GetTimelinePublic return statuses from public timeline.
2017-05-06 16:03:19 +02:00
func (c *Client) GetTimelinePublic(ctx context.Context, isLocal bool, pg *Pagination) ([]*Status, error) {
2017-04-25 20:16:32 +02:00
params := url.Values{}
if isLocal {
params.Set("local", "t")
}
var statuses []*Status
2017-05-06 16:03:19 +02:00
err := c.doAPI(ctx, http.MethodGet, "/api/v1/timelines/public", params, &statuses, pg)
2017-04-25 20:16:32 +02:00
if err != nil {
2017-05-06 16:03:19 +02:00
return nil, err
2017-04-25 20:16:32 +02:00
}
2017-05-06 16:03:19 +02:00
return statuses, nil
2017-04-14 17:22:44 +02:00
}
2017-04-15 14:02:55 +02:00
// GetTimelineHashtag return statuses from tagged timeline.
2017-05-06 16:03:19 +02:00
func (c *Client) GetTimelineHashtag(ctx context.Context, tag string, isLocal bool, pg *Pagination) ([]*Status, error) {
2017-04-27 19:29:11 +02:00
params := url.Values{}
if isLocal {
params.Set("local", "t")
}
2017-04-15 14:02:55 +02:00
var statuses []*Status
2017-05-06 16:03:19 +02:00
err := c.doAPI(ctx, http.MethodGet, fmt.Sprintf("/api/v1/timelines/tag/%s", url.PathEscape(tag)), params, &statuses, pg)
2017-04-15 14:02:55 +02:00
if err != nil {
2017-05-06 16:03:19 +02:00
return nil, err
2017-04-15 14:02:55 +02:00
}
2017-05-06 16:03:19 +02:00
return statuses, nil
2017-04-15 14:02:55 +02:00
}
2019-04-27 03:56:47 +02:00
// GetTimelineList return statuses from a list timeline.
func (c *Client) GetTimelineList(ctx context.Context, id ID, pg *Pagination) ([]*Status, error) {
var statuses []*Status
err := c.doAPI(ctx, http.MethodGet, fmt.Sprintf("/api/v1/timelines/list/%s", url.PathEscape(string(id))), nil, &statuses, pg)
if err != nil {
return nil, err
}
return statuses, nil
}
2017-04-28 17:36:13 +02:00
// GetTimelineMedia return statuses from media timeline.
// NOTE: This is an experimental feature of pawoo.net.
2017-05-06 16:03:19 +02:00
func (c *Client) GetTimelineMedia(ctx context.Context, isLocal bool, pg *Pagination) ([]*Status, error) {
2017-04-28 17:36:13 +02:00
params := url.Values{}
params.Set("media", "t")
if isLocal {
params.Set("local", "t")
}
var statuses []*Status
2017-05-06 16:03:19 +02:00
err := c.doAPI(ctx, http.MethodGet, "/api/v1/timelines/public", params, &statuses, pg)
2017-04-28 17:36:13 +02:00
if err != nil {
2017-05-06 16:03:19 +02:00
return nil, err
2017-04-28 17:36:13 +02:00
}
2017-05-06 16:03:19 +02:00
return statuses, nil
2017-04-28 17:36:13 +02:00
}
2017-04-14 17:22:44 +02:00
// PostStatus post the toot.
func (c *Client) PostStatus(ctx context.Context, toot *Toot) (*Status, error) {
2017-04-14 17:22:44 +02:00
params := url.Values{}
params.Set("status", toot.Status)
if toot.InReplyToID != "" {
params.Set("in_reply_to_id", string(toot.InReplyToID))
2017-04-14 17:22:44 +02:00
}
if toot.MediaIDs != nil {
for _, media := range toot.MediaIDs {
params.Add("media_ids[]", string(media))
}
}
if toot.Visibility != "" {
params.Set("visibility", fmt.Sprint(toot.Visibility))
}
if toot.Sensitive {
2018-07-22 19:36:42 +02:00
params.Set("sensitive", "true")
}
if toot.SpoilerText != "" {
params.Set("spoiler_text", toot.SpoilerText)
}
2017-04-14 17:22:44 +02:00
var status Status
2017-05-06 16:03:19 +02:00
err := c.doAPI(ctx, http.MethodPost, "/api/v1/statuses", params, &status, nil)
2017-04-14 17:22:44 +02:00
if err != nil {
return nil, err
}
return &status, nil
}
2017-04-15 14:07:11 +02:00
// DeleteStatus delete the toot.
2017-10-25 08:22:17 +02:00
func (c *Client) DeleteStatus(ctx context.Context, id ID) error {
return c.doAPI(ctx, http.MethodDelete, fmt.Sprintf("/api/v1/statuses/%s", id), nil, nil, nil)
2017-04-15 14:07:11 +02:00
}
2017-04-15 16:21:16 +02:00
// Search search content with query.
func (c *Client) Search(ctx context.Context, q string, resolve bool) (*Results, error) {
2017-04-15 16:21:16 +02:00
params := url.Values{}
params.Set("q", q)
params.Set("resolve", fmt.Sprint(resolve))
var results Results
2017-05-06 16:03:19 +02:00
err := c.doAPI(ctx, http.MethodGet, "/api/v1/search", params, &results, nil)
2017-04-15 16:21:16 +02:00
if err != nil {
return nil, err
}
return &results, nil
}
2017-04-17 06:54:36 +02:00
2019-03-11 03:55:15 +01:00
// UploadMedia upload a media attachment from a file.
2017-04-17 06:54:36 +02:00
func (c *Client) UploadMedia(ctx context.Context, file string) (*Attachment, error) {
var attachment Attachment
2017-05-06 16:03:19 +02:00
err := c.doAPI(ctx, http.MethodPost, "/api/v1/media", file, &attachment, nil)
2017-04-17 06:54:36 +02:00
if err != nil {
return nil, err
}
return &attachment, nil
}
2019-03-11 03:55:15 +01:00
// UploadMediaFromReader uploads a media attachment from a io.Reader.
func (c *Client) UploadMediaFromReader(ctx context.Context, reader io.Reader) (*Attachment, error) {
var attachment Attachment
err := c.doAPI(ctx, http.MethodPost, "/api/v1/media", reader, &attachment, nil)
if err != nil {
return nil, err
}
return &attachment, nil
}
2019-06-21 18:44:24 +02:00
// GetTimelineDirect return statuses from direct timeline.
func (c *Client) GetTimelineDirect(ctx context.Context, pg *Pagination) ([]*Status, error) {
params := url.Values{}
var statuses []*Status
err := c.doAPI(ctx, http.MethodGet, "/api/v1/timelines/direct", params, &statuses, pg)
if err != nil {
return nil, err
}
return statuses, nil
}
// GetConversations return direct conversations.
func (c *Client) GetConversations(ctx context.Context, pg *Pagination) ([]*Conversation, error) {
params := url.Values{}
var conversations []*Conversation
err := c.doAPI(ctx, http.MethodGet, "/api/v1/conversations", params, &conversations, pg)
if err != nil {
return nil, err
}
return conversations, nil
}
// DeleteConversation delete the conversation specified by id.
func (c *Client) DeleteConversation(ctx context.Context, id ID) error {
return c.doAPI(ctx, http.MethodDelete, fmt.Sprintf("/api/v1/conversations/%s", id), nil, nil, nil)
}
// MarkConversationAsRead mark the conversation as read.
func (c *Client) MarkConversationAsRead(ctx context.Context, id ID) error {
return c.doAPI(ctx, http.MethodPost, fmt.Sprintf("/api/v1/conversations/%s/read", id), nil, nil, nil)
}