183 lines
4.6 KiB
Go
Raw Normal View History

2017-08-26 10:53:28 +02:00
package slack
import (
"bytes"
2017-12-01 23:52:25 +01:00
"context"
2017-08-26 10:53:28 +02:00
"encoding/json"
"fmt"
"io"
"io/ioutil"
"mime/multipart"
"net/http"
"net/http/httputil"
"net/url"
"os"
"path/filepath"
"strconv"
2017-12-01 23:52:25 +01:00
"strings"
2017-08-26 10:53:28 +02:00
"time"
)
type WebResponse struct {
Ok bool `json:"ok"`
Error *WebError `json:"error"`
}
type WebError string
func (s WebError) Error() string {
return string(s)
}
type RateLimitedError struct {
RetryAfter time.Duration
}
func (e *RateLimitedError) Error() string {
return fmt.Sprintf("Slack rate limit exceeded, retry after %s", e.RetryAfter)
}
2017-12-01 23:52:25 +01:00
func fileUploadReq(ctx context.Context, path, fieldname, filename string, values url.Values, r io.Reader) (*http.Request, error) {
2017-08-26 10:53:28 +02:00
body := &bytes.Buffer{}
wr := multipart.NewWriter(body)
2017-12-01 23:52:25 +01:00
ioWriter, err := wr.CreateFormFile(fieldname, filename)
2017-08-26 10:53:28 +02:00
if err != nil {
wr.Close()
return nil, err
}
2017-12-01 23:52:25 +01:00
_, err = io.Copy(ioWriter, r)
2017-08-26 10:53:28 +02:00
if err != nil {
wr.Close()
return nil, err
}
// Close the multipart writer or the footer won't be written
wr.Close()
req, err := http.NewRequest("POST", path, body)
2017-12-01 23:52:25 +01:00
req = req.WithContext(ctx)
2017-08-26 10:53:28 +02:00
if err != nil {
return nil, err
}
req.Header.Add("Content-Type", wr.FormDataContentType())
req.URL.RawQuery = (values).Encode()
return req, nil
}
func parseResponseBody(body io.ReadCloser, intf *interface{}, debug bool) error {
response, err := ioutil.ReadAll(body)
if err != nil {
return err
}
// FIXME: will be api.Debugf
if debug {
logger.Printf("parseResponseBody: %s\n", string(response))
}
return json.Unmarshal(response, &intf)
2017-08-26 10:53:28 +02:00
}
func postLocalWithMultipartResponse(ctx context.Context, client HTTPRequester, path, fpath, fieldname string, values url.Values, intf interface{}, debug bool) error {
2017-12-01 23:52:25 +01:00
fullpath, err := filepath.Abs(fpath)
if err != nil {
return err
}
file, err := os.Open(fullpath)
if err != nil {
return err
}
defer file.Close()
return postWithMultipartResponse(ctx, client, path, filepath.Base(fpath), fieldname, values, file, intf, debug)
2017-12-01 23:52:25 +01:00
}
func postWithMultipartResponse(ctx context.Context, client HTTPRequester, path, name, fieldname string, values url.Values, r io.Reader, intf interface{}, debug bool) error {
2017-12-01 23:52:25 +01:00
req, err := fileUploadReq(ctx, SLACK_API+path, fieldname, name, values, r)
if err != nil {
return err
}
req = req.WithContext(ctx)
resp, err := client.Do(req)
2017-08-26 10:53:28 +02:00
if err != nil {
return err
}
defer resp.Body.Close()
if resp.StatusCode == http.StatusTooManyRequests {
retry, err := strconv.ParseInt(resp.Header.Get("Retry-After"), 10, 64)
if err != nil {
return err
}
return &RateLimitedError{time.Duration(retry) * time.Second}
}
2017-08-26 10:53:28 +02:00
// Slack seems to send an HTML body along with 5xx error codes. Don't parse it.
if resp.StatusCode != http.StatusOK {
2017-08-26 10:53:28 +02:00
logResponse(resp, debug)
return fmt.Errorf("Slack server error: %s.", resp.Status)
}
return parseResponseBody(resp.Body, &intf, debug)
}
func postForm(ctx context.Context, client HTTPRequester, endpoint string, values url.Values, intf interface{}, debug bool) error {
2017-12-01 23:52:25 +01:00
reqBody := strings.NewReader(values.Encode())
req, err := http.NewRequest("POST", endpoint, reqBody)
if err != nil {
return err
}
req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
req = req.WithContext(ctx)
resp, err := client.Do(req)
2017-08-26 10:53:28 +02:00
if err != nil {
return err
}
defer resp.Body.Close()
if resp.StatusCode == http.StatusTooManyRequests {
retry, err := strconv.ParseInt(resp.Header.Get("Retry-After"), 10, 64)
if err != nil {
return err
}
return &RateLimitedError{time.Duration(retry) * time.Second}
}
2017-12-01 23:52:25 +01:00
// Slack seems to send an HTML body along with 5xx error codes. Don't parse it.
if resp.StatusCode != http.StatusOK {
2017-12-01 23:52:25 +01:00
logResponse(resp, debug)
return fmt.Errorf("Slack server error: %s.", resp.Status)
}
2017-08-26 10:53:28 +02:00
return parseResponseBody(resp.Body, &intf, debug)
}
func post(ctx context.Context, client HTTPRequester, path string, values url.Values, intf interface{}, debug bool) error {
return postForm(ctx, client, SLACK_API+path, values, intf, debug)
2017-08-26 10:53:28 +02:00
}
func parseAdminResponse(ctx context.Context, client HTTPRequester, method string, teamName string, values url.Values, intf interface{}, debug bool) error {
2017-08-26 10:53:28 +02:00
endpoint := fmt.Sprintf(SLACK_WEB_API_FORMAT, teamName, method, time.Now().Unix())
return postForm(ctx, client, endpoint, values, intf, debug)
2017-08-26 10:53:28 +02:00
}
func logResponse(resp *http.Response, debug bool) error {
if debug {
text, err := httputil.DumpResponse(resp, true)
if err != nil {
return err
}
2017-12-01 23:52:25 +01:00
logger.Print(string(text))
2017-08-26 10:53:28 +02:00
}
return nil
}
2017-12-01 23:52:25 +01:00
func okJsonHandler(rw http.ResponseWriter, r *http.Request) {
rw.Header().Set("Content-Type", "application/json")
response, _ := json.Marshal(SlackResponse{
Ok: true,
})
rw.Write(response)
2017-12-01 23:52:25 +01:00
}