75bd409ba0
Backend support for subscribing/unsubscribing to blocks, typically cards and boards. Notifies subscribers when changes are made to cards they are subscribed to.
411 lines
9.8 KiB
Go
411 lines
9.8 KiB
Go
package client
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"mime/multipart"
|
|
"net/http"
|
|
"strings"
|
|
|
|
"github.com/mattermost/focalboard/server/api"
|
|
"github.com/mattermost/focalboard/server/model"
|
|
)
|
|
|
|
const (
|
|
APIURLSuffix = "/api/v1"
|
|
)
|
|
|
|
type RequestReaderError struct {
|
|
buf []byte
|
|
}
|
|
|
|
func (rre RequestReaderError) Error() string {
|
|
return "payload: " + string(rre.buf)
|
|
}
|
|
|
|
type Response struct {
|
|
StatusCode int
|
|
Error error
|
|
Header http.Header
|
|
}
|
|
|
|
func BuildResponse(r *http.Response) *Response {
|
|
return &Response{
|
|
StatusCode: r.StatusCode,
|
|
Header: r.Header,
|
|
}
|
|
}
|
|
|
|
func BuildErrorResponse(r *http.Response, err error) *Response {
|
|
statusCode := 0
|
|
header := make(http.Header)
|
|
if r != nil {
|
|
statusCode = r.StatusCode
|
|
header = r.Header
|
|
}
|
|
|
|
return &Response{
|
|
StatusCode: statusCode,
|
|
Error: err,
|
|
Header: header,
|
|
}
|
|
}
|
|
|
|
func closeBody(r *http.Response) {
|
|
if r.Body != nil {
|
|
_, _ = io.Copy(ioutil.Discard, r.Body)
|
|
_ = r.Body.Close()
|
|
}
|
|
}
|
|
|
|
func toJSON(v interface{}) string {
|
|
b, _ := json.Marshal(v)
|
|
return string(b)
|
|
}
|
|
|
|
type Client struct {
|
|
URL string
|
|
APIURL string
|
|
HTTPClient *http.Client
|
|
HTTPHeader map[string]string
|
|
// Token if token is empty indicate client is not login yet
|
|
Token string
|
|
}
|
|
|
|
func NewClient(url, sessionToken string) *Client {
|
|
url = strings.TrimRight(url, "/")
|
|
|
|
headers := map[string]string{
|
|
"X-Requested-With": "XMLHttpRequest",
|
|
}
|
|
|
|
return &Client{url, url + APIURLSuffix, &http.Client{}, headers, sessionToken}
|
|
}
|
|
|
|
func (c *Client) DoAPIGet(url, etag string) (*http.Response, error) {
|
|
return c.DoAPIRequest(http.MethodGet, c.APIURL+url, "", etag)
|
|
}
|
|
|
|
func (c *Client) DoAPIPost(url, data string) (*http.Response, error) {
|
|
return c.DoAPIRequest(http.MethodPost, c.APIURL+url, data, "")
|
|
}
|
|
|
|
func (c *Client) DoAPIPatch(url, data string) (*http.Response, error) {
|
|
return c.DoAPIRequest(http.MethodPatch, c.APIURL+url, data, "")
|
|
}
|
|
|
|
func (c *Client) DoAPIPut(url, data string) (*http.Response, error) {
|
|
return c.DoAPIRequest(http.MethodPut, c.APIURL+url, data, "")
|
|
}
|
|
|
|
func (c *Client) DoAPIDelete(url string) (*http.Response, error) {
|
|
return c.DoAPIRequest(http.MethodDelete, c.APIURL+url, "", "")
|
|
}
|
|
|
|
func (c *Client) DoAPIRequest(method, url, data, etag string) (*http.Response, error) {
|
|
return c.doAPIRequestReader(method, url, strings.NewReader(data), etag)
|
|
}
|
|
|
|
type requestOption func(r *http.Request)
|
|
|
|
func (c *Client) doAPIRequestReader(method, url string, data io.Reader, _ /* etag */ string, opts ...requestOption) (*http.Response, error) {
|
|
rq, err := http.NewRequest(method, url, data)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for _, opt := range opts {
|
|
opt(rq)
|
|
}
|
|
|
|
if c.HTTPHeader != nil && len(c.HTTPHeader) > 0 {
|
|
for k, v := range c.HTTPHeader {
|
|
rq.Header.Set(k, v)
|
|
}
|
|
}
|
|
|
|
if c.Token != "" {
|
|
rq.Header.Set("Authorization", "Bearer "+c.Token)
|
|
}
|
|
|
|
rp, err := c.HTTPClient.Do(rq)
|
|
if err != nil || rp == nil {
|
|
return nil, err
|
|
}
|
|
|
|
if rp.StatusCode == http.StatusNotModified {
|
|
return rp, nil
|
|
}
|
|
|
|
if rp.StatusCode >= http.StatusMultipleChoices {
|
|
defer closeBody(rp)
|
|
b, err := ioutil.ReadAll(rp.Body)
|
|
if err != nil {
|
|
return rp, fmt.Errorf("error when parsing response with code %d: %w", rp.StatusCode, err)
|
|
}
|
|
return rp, RequestReaderError{b}
|
|
}
|
|
|
|
return rp, nil
|
|
}
|
|
|
|
func (c *Client) GetBlocksRoute() string {
|
|
return "/workspaces/0/blocks"
|
|
}
|
|
|
|
func (c *Client) GetBlockRoute(id string) string {
|
|
return fmt.Sprintf("%s/%s", c.GetBlocksRoute(), id)
|
|
}
|
|
|
|
func (c *Client) GetSubtreeRoute(id string) string {
|
|
return fmt.Sprintf("%s/subtree", c.GetBlockRoute(id))
|
|
}
|
|
|
|
func (c *Client) GetBlocks() ([]model.Block, *Response) {
|
|
r, err := c.DoAPIGet(c.GetBlocksRoute(), "")
|
|
if err != nil {
|
|
return nil, BuildErrorResponse(r, err)
|
|
}
|
|
defer closeBody(r)
|
|
|
|
return model.BlocksFromJSON(r.Body), BuildResponse(r)
|
|
}
|
|
|
|
func (c *Client) PatchBlock(blockID string, blockPatch *model.BlockPatch) (bool, *Response) {
|
|
r, err := c.DoAPIPatch(c.GetBlockRoute(blockID), toJSON(blockPatch))
|
|
if err != nil {
|
|
return false, BuildErrorResponse(r, err)
|
|
}
|
|
defer closeBody(r)
|
|
|
|
return true, BuildResponse(r)
|
|
}
|
|
|
|
func (c *Client) InsertBlocks(blocks []model.Block) ([]model.Block, *Response) {
|
|
r, err := c.DoAPIPost(c.GetBlocksRoute(), toJSON(blocks))
|
|
if err != nil {
|
|
return nil, BuildErrorResponse(r, err)
|
|
}
|
|
defer closeBody(r)
|
|
|
|
return model.BlocksFromJSON(r.Body), BuildResponse(r)
|
|
}
|
|
|
|
func (c *Client) DeleteBlock(blockID string) (bool, *Response) {
|
|
r, err := c.DoAPIDelete(c.GetBlockRoute(blockID))
|
|
if err != nil {
|
|
return false, BuildErrorResponse(r, err)
|
|
}
|
|
defer closeBody(r)
|
|
|
|
return true, BuildResponse(r)
|
|
}
|
|
|
|
func (c *Client) GetSubtree(blockID string) ([]model.Block, *Response) {
|
|
r, err := c.DoAPIGet(c.GetSubtreeRoute(blockID), "")
|
|
if err != nil {
|
|
return nil, BuildErrorResponse(r, err)
|
|
}
|
|
defer closeBody(r)
|
|
|
|
return model.BlocksFromJSON(r.Body), BuildResponse(r)
|
|
}
|
|
|
|
// Sharing
|
|
|
|
func (c *Client) GetSharingRoute(rootID string) string {
|
|
return fmt.Sprintf("/workspaces/0/sharing/%s", rootID)
|
|
}
|
|
|
|
func (c *Client) GetSharing(rootID string) (*model.Sharing, *Response) {
|
|
r, err := c.DoAPIGet(c.GetSharingRoute(rootID), "")
|
|
if err != nil {
|
|
return nil, BuildErrorResponse(r, err)
|
|
}
|
|
defer closeBody(r)
|
|
|
|
sharing := model.SharingFromJSON(r.Body)
|
|
return &sharing, BuildResponse(r)
|
|
}
|
|
|
|
func (c *Client) PostSharing(sharing model.Sharing) (bool, *Response) {
|
|
r, err := c.DoAPIPost(c.GetSharingRoute(sharing.ID), toJSON(sharing))
|
|
if err != nil {
|
|
return false, BuildErrorResponse(r, err)
|
|
}
|
|
defer closeBody(r)
|
|
|
|
return true, BuildResponse(r)
|
|
}
|
|
|
|
func (c *Client) GetRegisterRoute() string {
|
|
return "/register"
|
|
}
|
|
|
|
func (c *Client) Register(request *api.RegisterRequest) (bool, *Response) {
|
|
r, err := c.DoAPIPost(c.GetRegisterRoute(), toJSON(&request))
|
|
if err != nil {
|
|
return false, BuildErrorResponse(r, err)
|
|
}
|
|
defer closeBody(r)
|
|
|
|
return true, BuildResponse(r)
|
|
}
|
|
|
|
func (c *Client) GetLoginRoute() string {
|
|
return "/login"
|
|
}
|
|
|
|
func (c *Client) Login(request *api.LoginRequest) (*api.LoginResponse, *Response) {
|
|
r, err := c.DoAPIPost(c.GetLoginRoute(), toJSON(&request))
|
|
if err != nil {
|
|
return nil, BuildErrorResponse(r, err)
|
|
}
|
|
defer closeBody(r)
|
|
|
|
data, err := api.LoginResponseFromJSON(r.Body)
|
|
if err != nil {
|
|
return nil, BuildErrorResponse(r, err)
|
|
}
|
|
|
|
if data.Token != "" {
|
|
c.Token = data.Token
|
|
}
|
|
|
|
return data, BuildResponse(r)
|
|
}
|
|
|
|
func (c *Client) GetMeRoute() string {
|
|
return "/users/me"
|
|
}
|
|
|
|
func (c *Client) GetMe() (*model.User, *Response) {
|
|
r, err := c.DoAPIGet(c.GetMeRoute(), "")
|
|
if err != nil {
|
|
return nil, BuildErrorResponse(r, err)
|
|
}
|
|
defer closeBody(r)
|
|
|
|
me, err := model.UserFromJSON(r.Body)
|
|
if err != nil {
|
|
return nil, BuildErrorResponse(r, err)
|
|
}
|
|
return me, BuildResponse(r)
|
|
}
|
|
|
|
func (c *Client) GetUserRoute(id string) string {
|
|
return fmt.Sprintf("/users/%s", id)
|
|
}
|
|
|
|
func (c *Client) GetUser(id string) (*model.User, *Response) {
|
|
r, err := c.DoAPIGet(c.GetUserRoute(id), "")
|
|
if err != nil {
|
|
return nil, BuildErrorResponse(r, err)
|
|
}
|
|
defer closeBody(r)
|
|
|
|
user, err := model.UserFromJSON(r.Body)
|
|
if err != nil {
|
|
return nil, BuildErrorResponse(r, err)
|
|
}
|
|
return user, BuildResponse(r)
|
|
}
|
|
|
|
func (c *Client) GetUserChangePasswordRoute(id string) string {
|
|
return fmt.Sprintf("/users/%s/changepassword", id)
|
|
}
|
|
|
|
func (c *Client) UserChangePassword(id string, data *api.ChangePasswordRequest) (bool, *Response) {
|
|
r, err := c.DoAPIPost(c.GetUserChangePasswordRoute(id), toJSON(&data))
|
|
if err != nil {
|
|
return false, BuildErrorResponse(r, err)
|
|
}
|
|
defer closeBody(r)
|
|
|
|
return true, BuildResponse(r)
|
|
}
|
|
|
|
func (c *Client) GetWorkspaceUploadFileRoute(workspaceID, rootID string) string {
|
|
return fmt.Sprintf("/workspaces/%s/%s/files", workspaceID, rootID)
|
|
}
|
|
|
|
func (c *Client) WorkspaceUploadFile(workspaceID, rootID string, data io.Reader) (*api.FileUploadResponse, *Response) {
|
|
body := &bytes.Buffer{}
|
|
writer := multipart.NewWriter(body)
|
|
part, err := writer.CreateFormFile(api.UploadFormFileKey, "file")
|
|
if err != nil {
|
|
return nil, &Response{Error: err}
|
|
}
|
|
if _, err = io.Copy(part, data); err != nil {
|
|
return nil, &Response{Error: err}
|
|
}
|
|
writer.Close()
|
|
|
|
opt := func(r *http.Request) {
|
|
r.Header.Add("Content-Type", writer.FormDataContentType())
|
|
}
|
|
|
|
r, err := c.doAPIRequestReader(http.MethodPost, c.APIURL+c.GetWorkspaceUploadFileRoute(workspaceID, rootID), body, "", opt)
|
|
if err != nil {
|
|
return nil, BuildErrorResponse(r, err)
|
|
}
|
|
defer closeBody(r)
|
|
|
|
fileUploadResponse, err := api.FileUploadResponseFromJSON(r.Body)
|
|
if err != nil {
|
|
return nil, BuildErrorResponse(r, err)
|
|
}
|
|
|
|
return fileUploadResponse, BuildResponse(r)
|
|
}
|
|
|
|
func (c *Client) GetSubscriptionsRoute(workspaceID string) string {
|
|
return fmt.Sprintf("/workspaces/%s/subscriptions", workspaceID)
|
|
}
|
|
|
|
func (c *Client) CreateSubscription(workspaceID string, sub *model.Subscription) (*model.Subscription, *Response) {
|
|
r, err := c.DoAPIPost(c.GetSubscriptionsRoute(workspaceID), toJSON(&sub))
|
|
if err != nil {
|
|
return nil, BuildErrorResponse(r, err)
|
|
}
|
|
defer closeBody(r)
|
|
|
|
subNew, err := model.SubscriptionFromJSON(r.Body)
|
|
if err != nil {
|
|
return nil, BuildErrorResponse(r, err)
|
|
}
|
|
return subNew, BuildResponse(r)
|
|
}
|
|
|
|
func (c *Client) DeleteSubscription(workspaceID string, blockID string, subscriberID string) *Response {
|
|
url := fmt.Sprintf("%s/%s/%s", c.GetSubscriptionsRoute(workspaceID), blockID, subscriberID)
|
|
|
|
r, err := c.DoAPIDelete(url)
|
|
if err != nil {
|
|
return BuildErrorResponse(r, err)
|
|
}
|
|
defer closeBody(r)
|
|
|
|
return BuildResponse(r)
|
|
}
|
|
|
|
func (c *Client) GetSubscriptions(workspaceID string, subscriberID string) ([]*model.Subscription, *Response) {
|
|
url := fmt.Sprintf("%s/%s", c.GetSubscriptionsRoute(workspaceID), subscriberID)
|
|
|
|
r, err := c.DoAPIGet(url, "")
|
|
if err != nil {
|
|
return nil, BuildErrorResponse(r, err)
|
|
}
|
|
defer closeBody(r)
|
|
|
|
var subs []*model.Subscription
|
|
err = json.NewDecoder(r.Body).Decode(&subs)
|
|
if err != nil {
|
|
return nil, BuildErrorResponse(r, err)
|
|
}
|
|
|
|
return subs, BuildResponse(r)
|
|
}
|