Files
slack2asana/slack.go
2022-10-28 19:33:27 +00:00

371 lines
7.1 KiB
Go

package main
import (
"bytes"
"encoding/json"
"errors"
"fmt"
"net/http"
"net/url"
"os"
"regexp"
"strings"
)
type SlackClient struct {
cli *http.Client
token string
team *Team
}
type removeStarRequest struct {
Channel string `json:"channel"`
Timestamp string `json:"timestamp"`
}
type channelResponse struct {
Ok bool `json:"ok"`
Error string `json:"error"`
Channel *Channel `json:"channel"`
}
type starsResponse struct {
Ok bool `json:"ok"`
Error string `json:"error"`
Items []*Item `json:"items"`
}
type userResponse struct {
Ok bool `json:"ok"`
Error string `json:"error"`
User *User `json:"user"`
}
type simpleResponse struct {
Ok bool `json:"ok"`
Error string `json:"error"`
}
type teamResponse struct {
Ok bool `json:"ok"`
Error string `json:"error"`
Team *Team `json:"team"`
}
type Purpose struct {
Value string `json:"value"`
}
type Channel struct {
Id string `json:"id"`
Name string `json:"name"`
IsChannel bool `json:"is_channel"`
IsGroup bool `json:"is_group"`
IsIm bool `json:"is_im"`
IsMpIm bool `json:"is_mpim"`
Purpose *Purpose `json:"purpose"`
}
type Item struct {
Type string `json:"type"`
Channel string `json:"channel"`
Message *Message `json:"message"`
}
type Message struct {
ClientMessageId string `json:"client_msg_id"`
Text string `json:"text"`
User string `json:"user"`
Ts string `json:"ts"`
Permalink string `json:"permalink"`
}
type Team struct {
Domain string `json:"domain"`
}
type User struct {
Id string `json:"id"`
Name string `json:"name"`
}
func NewSlackClient() (*SlackClient, error) {
sc := &SlackClient{
cli: &http.Client{},
token: os.Getenv("SLACK_TOKEN"),
}
team, err := sc.GetTeam()
if err != nil {
return nil, err
}
sc.team = team
return sc, nil
}
func (sc *SlackClient) GetTeam() (*Team, error) {
req, err := http.NewRequest("GET", "https://slack.com/api/team.info", nil)
if err != nil {
return nil, err
}
sc.addAuth(req)
resp, err := sc.cli.Do(req)
if err != nil {
return nil, err
}
dec := json.NewDecoder(resp.Body)
team := &teamResponse{}
err = dec.Decode(team)
if err != nil {
return nil, err
}
if !team.Ok {
return nil, errors.New(team.Error)
}
return team.Team, nil
}
func (sc *SlackClient) GetStars() ([]*Item, error) {
req, err := http.NewRequest("GET", "https://slack.com/api/stars.list", nil)
if err != nil {
return nil, err
}
sc.addAuth(req)
resp, err := sc.cli.Do(req)
if err != nil {
return nil, err
}
dec := json.NewDecoder(resp.Body)
stars := &starsResponse{}
err = dec.Decode(stars)
if err != nil {
return nil, err
}
if !stars.Ok {
return nil, errors.New(stars.Error)
}
return stars.Items, nil
}
func (sc *SlackClient) GetUser(id string) (*User, error) {
u, err := url.Parse("https://slack.com/api/users.info")
if err != nil {
return nil, err
}
v := &url.Values{}
v.Add("user", id)
u.RawQuery = v.Encode()
req, err := http.NewRequest("GET", u.String(), nil)
if err != nil {
return nil, err
}
sc.addAuth(req)
resp, err := sc.cli.Do(req)
if err != nil {
return nil, err
}
dec := json.NewDecoder(resp.Body)
user := &userResponse{}
err = dec.Decode(user)
if err != nil {
return nil, err
}
if !user.Ok {
return nil, errors.New(user.Error)
}
return user.User, nil
}
func (sc *SlackClient) GetChannel(id string) (*Channel, error) {
u, err := url.Parse("https://slack.com/api/conversations.info")
if err != nil {
return nil, err
}
v := &url.Values{}
v.Add("channel", id)
u.RawQuery = v.Encode()
req, err := http.NewRequest("GET", u.String(), nil)
if err != nil {
return nil, err
}
sc.addAuth(req)
resp, err := sc.cli.Do(req)
if err != nil {
return nil, err
}
dec := json.NewDecoder(resp.Body)
channel := &channelResponse{}
err = dec.Decode(channel)
if err != nil {
return nil, err
}
if !channel.Ok {
return nil, errors.New(channel.Error)
}
return channel.Channel, nil
}
func (sc *SlackClient) RemoveStar(item *Item) error {
body := &removeStarRequest{
Channel: item.Channel,
Timestamp: item.Message.Ts,
}
js, err := json.Marshal(body)
if err != nil {
return err
}
req, err := http.NewRequest("POST", "https://slack.com/api/stars.remove", bytes.NewReader(js))
if err != nil {
return err
}
sc.addAuth(req)
req.Header.Add("Content-Type", "application/json")
resp, err := sc.cli.Do(req)
if err != nil {
return err
}
dec := json.NewDecoder(resp.Body)
sr := &simpleResponse{}
err = dec.Decode(sr)
if err != nil {
return err
}
if !sr.Ok {
return errors.New(sr.Error)
}
return nil
}
func (sc *SlackClient) GetTitle(item *Item, user *User, channel *Channel) (string, error) {
switch {
case channel.IsIm:
return fmt.Sprintf("<%s> %s", user.Name, item.Message.Text), nil
case channel.IsMpIm:
return fmt.Sprintf("<%s> %s", user.Name, item.Message.Text), nil
case channel.IsChannel:
return fmt.Sprintf("<%s> %s", user.Name, item.Message.Text), nil
default:
return "", fmt.Errorf("unknown channel type: %#v", channel)
}
}
func (sc *SlackClient) GetTrimmedTitle(item *Item, user *User, channel *Channel) (string, error) {
title, err := sc.GetTitle(item, user, channel)
if err != nil {
return "", err
}
parts := strings.SplitN(title, "\n", 2)
title = parts[0]
if len(title) < 80 {
return title, nil
}
return fmt.Sprintf("%s...", title[:77]), nil
}
func (sc *SlackClient) GetNotes(item *Item, user *User, channel *Channel) (string, error) {
title, err := sc.GetTitle(item, user, channel)
if err != nil {
return "", err
}
link := sc.getLink(item)
switch {
case channel.IsIm:
return fmt.Sprintf(
"<body>%s\n\n<a href=\"%s\">%s</a></body>",
sc.escape(title),
sc.escape(link),
sc.escape(link),
), nil
case channel.IsMpIm:
return fmt.Sprintf(
"<body>%s\n\nIn %s\n\n<a href=\"%s\">%s</a></body>",
sc.escape(title),
sc.escape(sc.getTaggedNamesString(channel.Purpose.Value)),
sc.escape(link),
sc.escape(link),
), nil
case channel.IsChannel:
return fmt.Sprintf(
"<body>%s\n\nIn #%s\n\n<a href=\"%s\">%s</a></body>",
sc.escape(title),
sc.escape(channel.Name),
sc.escape(link),
sc.escape(link),
), nil
default:
return "", fmt.Errorf("unknown channel type: %#v", channel)
}
}
func (sc *SlackClient) getLink(item *Item) string {
return fmt.Sprintf(
"https://%s.slack.com/archives/%s/p%s",
sc.team.Domain,
item.Channel,
strings.ReplaceAll(item.Message.Ts, ".", ""),
)
}
var taggedNamesRE = regexp.MustCompile(`@[a-zA-Z0-9]+[!a-zA-Z0-9]`)
func (sc *SlackClient) getTaggedNames(in string) []string {
return taggedNamesRE.FindAllString(in, -1)
}
func (sc *SlackClient) getTaggedNamesString(in string) string {
return fmt.Sprintf("{%s}", strings.Join(sc.getTaggedNames(in), ","))
}
func (sc *SlackClient) escape(in string) string {
in = strings.ReplaceAll(in, "<", "&lt;")
in = strings.ReplaceAll(in, ">", "&gt;")
in = strings.ReplaceAll(in, `"`, "&quot;")
return in
}
func (sc *SlackClient) addAuth(req *http.Request) {
req.Header.Add("Authorization", fmt.Sprintf("Bearer %s", sc.token))
}