418 lines
9.3 KiB
Go
418 lines
9.3 KiB
Go
|
package groupme
|
||
|
|
||
|
import (
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"net/http"
|
||
|
"net/url"
|
||
|
"strconv"
|
||
|
)
|
||
|
|
||
|
// GroupMe documentation: https://dev.groupme.com/docs/v3#groups
|
||
|
|
||
|
////////// Endpoints //////////
|
||
|
const (
|
||
|
// Used to build other endpoints
|
||
|
groupsEndpointRoot = "/groups"
|
||
|
groupEndpointRoot = "/groups/%s"
|
||
|
|
||
|
// Actual Endpoints
|
||
|
indexGroupsEndpoint = groupsEndpointRoot // GET
|
||
|
formerGroupsEndpoint = groupsEndpointRoot + "/former" // GET
|
||
|
showGroupEndpoint = groupEndpointRoot // GET
|
||
|
createGroupEndpoint = groupsEndpointRoot // POST
|
||
|
updateGroupEndpoint = groupEndpointRoot + "/update" // POST
|
||
|
destroyGroupEndpoint = groupEndpointRoot + "/destroy" // POST
|
||
|
joinGroupEndpoint = groupEndpointRoot + "/join/%s" // POST
|
||
|
rejoinGroupEndpoint = groupsEndpointRoot + "/join" // POST
|
||
|
changeGroupOwnerEndpoint = groupsEndpointRoot + "/change_owners" // POST
|
||
|
)
|
||
|
|
||
|
////////// Common Request Parameters //////////
|
||
|
|
||
|
// GroupSettings is the settings for a group, used by CreateGroup and UpdateGroup
|
||
|
type GroupSettings struct {
|
||
|
// Required. Primary name of the group. Maximum 140 characters
|
||
|
Name string `json:"name"`
|
||
|
// A subheading for the group. Maximum 255 characters
|
||
|
Description string `json:"description"`
|
||
|
// GroupMe Image Service URL
|
||
|
ImageURL string `json:"image_url"`
|
||
|
// Defaults false. If true, disables notifications for all members.
|
||
|
// Documented for use only for UpdateGroup
|
||
|
OfficeMode bool `json:"office_mode"`
|
||
|
// Defaults false. If true, generates a share URL.
|
||
|
// Anyone with the URL can join the group
|
||
|
Share bool `json:"share"`
|
||
|
}
|
||
|
|
||
|
func (gss GroupSettings) String() string {
|
||
|
return marshal(&gss)
|
||
|
}
|
||
|
|
||
|
////////// API Requests //////////
|
||
|
|
||
|
///// Index /////
|
||
|
|
||
|
// GroupsQuery defines optional URL parameters for IndexGroups
|
||
|
type GroupsQuery struct {
|
||
|
// Fetch a particular page of results. Defaults to 1.
|
||
|
Page int `json:"page"`
|
||
|
// Define page size. Defaults to 10.
|
||
|
PerPage int `json:"per_page"`
|
||
|
// Comma separated list of data to omit from output.
|
||
|
// Currently supported value is only "memberships".
|
||
|
// If used then response will contain empty (null) members field.
|
||
|
Omit string `json:"omit"`
|
||
|
}
|
||
|
|
||
|
func (q GroupsQuery) String() string {
|
||
|
return marshal(&q)
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
IndexGroups -
|
||
|
|
||
|
List the authenticated user's active groups.
|
||
|
|
||
|
The response is paginated, with a default of 10 groups per page.
|
||
|
|
||
|
Please consider using of omit=memberships parameter. Not including
|
||
|
member lists might significantly improve user experience of your
|
||
|
app for users who are participating in huge groups.
|
||
|
|
||
|
Parameters: See GroupsQuery
|
||
|
*/
|
||
|
func (c *Client) IndexGroups(req *GroupsQuery) ([]*Group, error) {
|
||
|
httpReq, err := http.NewRequest("GET", c.endpointBase+indexGroupsEndpoint, nil)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
URL := httpReq.URL
|
||
|
query := URL.Query()
|
||
|
if req != nil {
|
||
|
if req.Page != 0 {
|
||
|
query.Set("page", strconv.Itoa(req.Page))
|
||
|
}
|
||
|
if req.PerPage != 0 {
|
||
|
query.Set("per_page", strconv.Itoa(req.PerPage))
|
||
|
}
|
||
|
if req.Omit != "" {
|
||
|
query.Set("omit", req.Omit)
|
||
|
}
|
||
|
}
|
||
|
URL.RawQuery = query.Encode()
|
||
|
|
||
|
var resp []*Group
|
||
|
err = c.do(httpReq, &resp)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return resp, nil
|
||
|
}
|
||
|
|
||
|
///// Former /////
|
||
|
|
||
|
/*
|
||
|
FormerGroups -
|
||
|
|
||
|
List they groups you have left but can rejoin.
|
||
|
*/
|
||
|
func (c *Client) FormerGroups() ([]*Group, error) {
|
||
|
httpReq, err := http.NewRequest("GET", c.endpointBase+formerGroupsEndpoint, nil)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
var resp []*Group
|
||
|
err = c.do(httpReq, &resp)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return resp, nil
|
||
|
}
|
||
|
|
||
|
///// Show /////
|
||
|
|
||
|
/*
|
||
|
ShowGroup -
|
||
|
|
||
|
Loads a specific group.
|
||
|
|
||
|
Parameters:
|
||
|
groupID - required, ID(string)
|
||
|
*/
|
||
|
func (c *Client) ShowGroup(groupID ID) (*Group, error) {
|
||
|
URL := fmt.Sprintf(c.endpointBase+showGroupEndpoint, groupID)
|
||
|
|
||
|
httpReq, err := http.NewRequest("GET", URL, nil)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
var resp Group
|
||
|
err = c.do(httpReq, &resp)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return &resp, nil
|
||
|
}
|
||
|
|
||
|
///// Create /////
|
||
|
|
||
|
/*
|
||
|
CreateGroup -
|
||
|
|
||
|
Create a new group
|
||
|
|
||
|
Parameters: See GroupSettings
|
||
|
*/
|
||
|
func (c *Client) CreateGroup(gs GroupSettings) (*Group, error) {
|
||
|
httpReq, err := http.NewRequest("POST", c.endpointBase+createGroupEndpoint, nil)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
data := url.Values{}
|
||
|
if gs.Name == "" {
|
||
|
return nil, fmt.Errorf("GroupsCreateRequest Name field is required")
|
||
|
}
|
||
|
data.Set("name", gs.Name)
|
||
|
|
||
|
if gs.Description != "" {
|
||
|
data.Set("description", gs.Description)
|
||
|
}
|
||
|
if gs.ImageURL != "" {
|
||
|
data.Set("image_url", gs.ImageURL)
|
||
|
}
|
||
|
if gs.Share {
|
||
|
data.Set("share", "true")
|
||
|
}
|
||
|
httpReq.PostForm = data
|
||
|
|
||
|
var resp Group
|
||
|
err = c.do(httpReq, &resp)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return &resp, nil
|
||
|
}
|
||
|
|
||
|
///// Update /////
|
||
|
|
||
|
/*
|
||
|
UpdateGroup -
|
||
|
|
||
|
Update a group after creation
|
||
|
|
||
|
Parameters:
|
||
|
groupID - required, ID(string)
|
||
|
See GroupSettings
|
||
|
*/
|
||
|
func (c *Client) UpdateGroup(groupID ID, gs GroupSettings) (*Group, error) {
|
||
|
URL := fmt.Sprintf(c.endpointBase+updateGroupEndpoint, groupID)
|
||
|
|
||
|
httpReq, err := http.NewRequest("POST", URL, nil)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
data := url.Values{}
|
||
|
if gs.Name != "" {
|
||
|
data.Set("name", gs.Name)
|
||
|
}
|
||
|
if gs.Description != "" {
|
||
|
data.Set("description", gs.Description)
|
||
|
}
|
||
|
if gs.ImageURL != "" {
|
||
|
data.Set("image_url", gs.ImageURL)
|
||
|
}
|
||
|
if gs.OfficeMode {
|
||
|
data.Set("office_mode", "true")
|
||
|
}
|
||
|
if gs.Share {
|
||
|
data.Set("share", "true")
|
||
|
}
|
||
|
httpReq.PostForm = data
|
||
|
|
||
|
var resp Group
|
||
|
err = c.do(httpReq, &resp)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return &resp, nil
|
||
|
}
|
||
|
|
||
|
///// Destroy /////
|
||
|
|
||
|
/*
|
||
|
DestroyGroup -
|
||
|
|
||
|
Disband a group
|
||
|
|
||
|
This action is only available to the group creator
|
||
|
|
||
|
Parameters:
|
||
|
groupID - required, ID(string)
|
||
|
*/
|
||
|
func (c *Client) DestroyGroup(groupID ID) error {
|
||
|
url := fmt.Sprintf(c.endpointBase+destroyGroupEndpoint, groupID)
|
||
|
|
||
|
httpReq, err := http.NewRequest("POST", url, nil)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
return c.do(httpReq, nil)
|
||
|
}
|
||
|
|
||
|
///// Join /////
|
||
|
|
||
|
/*
|
||
|
JoinGroup -
|
||
|
|
||
|
Join a shared group
|
||
|
|
||
|
Parameters:
|
||
|
groupID - required, ID(string)
|
||
|
shareToken - required, string
|
||
|
*/
|
||
|
func (c *Client) JoinGroup(groupID ID, shareToken string) (*Group, error) {
|
||
|
URL := fmt.Sprintf(c.endpointBase+joinGroupEndpoint, groupID, shareToken)
|
||
|
|
||
|
httpReq, err := http.NewRequest("POST", URL, nil)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
var resp Group
|
||
|
err = c.do(httpReq, &resp)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return &resp, nil
|
||
|
}
|
||
|
|
||
|
///// Rejoin /////
|
||
|
|
||
|
/*
|
||
|
RejoinGroup -
|
||
|
|
||
|
Rejoin a group. Only works if you previously removed yourself.
|
||
|
|
||
|
Parameters:
|
||
|
groupID - required, ID(string)
|
||
|
*/
|
||
|
func (c *Client) RejoinGroup(groupID ID) (*Group, error) {
|
||
|
httpReq, err := http.NewRequest("POST", c.endpointBase+rejoinGroupEndpoint, nil)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
data := url.Values{}
|
||
|
data.Set("group_id", string(groupID))
|
||
|
httpReq.PostForm = data
|
||
|
|
||
|
var resp Group
|
||
|
err = c.do(httpReq, &resp)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return &resp, nil
|
||
|
}
|
||
|
|
||
|
///// Change Owner /////
|
||
|
|
||
|
/*
|
||
|
ChangeGroupOwner-
|
||
|
|
||
|
Change owner of requested groups.
|
||
|
|
||
|
This action is only available to the group creator.
|
||
|
|
||
|
Response is a result object which contain status field,
|
||
|
the result of change owner action for the request
|
||
|
|
||
|
Parameters: See ChangeOwnerRequest
|
||
|
*/
|
||
|
func (c *Client) ChangeGroupOwner(reqs ChangeOwnerRequest) (ChangeOwnerResult, error) {
|
||
|
httpReq, err := http.NewRequest("POST", c.endpointBase+changeGroupOwnerEndpoint, nil)
|
||
|
if err != nil {
|
||
|
return ChangeOwnerResult{}, err
|
||
|
}
|
||
|
|
||
|
data := url.Values{}
|
||
|
data.Set("requests", marshal([]ChangeOwnerRequest{reqs}))
|
||
|
httpReq.PostForm = data
|
||
|
|
||
|
var resp struct {
|
||
|
Results []ChangeOwnerResult `json:"results"`
|
||
|
}
|
||
|
|
||
|
err = c.do(httpReq, &resp)
|
||
|
if err != nil {
|
||
|
return ChangeOwnerResult{}, err
|
||
|
}
|
||
|
|
||
|
if len(resp.Results) < 1 {
|
||
|
return ChangeOwnerResult{}, errors.New("failed to parse results")
|
||
|
}
|
||
|
|
||
|
return resp.Results[0], nil
|
||
|
}
|
||
|
|
||
|
type ChangeOwnerStatusCode string
|
||
|
|
||
|
const (
|
||
|
ChangeOwner_Ok ChangeOwnerStatusCode = "200"
|
||
|
ChangeOwner_RequesterNewOwner ChangeOwnerStatusCode = "400"
|
||
|
ChangeOwner_NotOwner ChangeOwnerStatusCode = "403"
|
||
|
ChangeOwner_BadGroupOrOwner ChangeOwnerStatusCode = "404"
|
||
|
ChangeOwner_BadRequest ChangeOwnerStatusCode = "405"
|
||
|
)
|
||
|
|
||
|
// String returns the description of the status code according to GroupMe
|
||
|
func (c ChangeOwnerStatusCode) String() string {
|
||
|
return map[ChangeOwnerStatusCode]string{
|
||
|
ChangeOwner_Ok: "success",
|
||
|
ChangeOwner_RequesterNewOwner: "requester is also a new owner",
|
||
|
ChangeOwner_NotOwner: "requester is not the owner of the group",
|
||
|
ChangeOwner_BadGroupOrOwner: "group or new owner not found or new owner is not memeber of the group",
|
||
|
ChangeOwner_BadRequest: "request object is missing required field or any of the required fields is not an ID",
|
||
|
}[c]
|
||
|
}
|
||
|
|
||
|
// ChangeOwnerRequest defines the new owner of a group
|
||
|
type ChangeOwnerRequest struct {
|
||
|
// Required
|
||
|
GroupID string `json:"group_id"`
|
||
|
// Required. UserId of the new owner of the group
|
||
|
// who must be an active member of the group
|
||
|
OwnerID string `json:"owner_id"`
|
||
|
}
|
||
|
|
||
|
func (r ChangeOwnerRequest) String() string {
|
||
|
return marshal(&r)
|
||
|
}
|
||
|
|
||
|
// ChangeOwnerResult holds the status of the group owner change
|
||
|
type ChangeOwnerResult struct {
|
||
|
GroupID string `json:"group_id"`
|
||
|
// UserId of the new owner of the group who is
|
||
|
// an active member of the group
|
||
|
OwnerID string `json:"owner_id"`
|
||
|
Status ChangeOwnerStatusCode `json:"status"`
|
||
|
}
|
||
|
|
||
|
func (r ChangeOwnerResult) String() string {
|
||
|
return marshal(&r)
|
||
|
}
|