1
0
Fork 0
mirror of https://github.com/diamondburned/arikawa.git synced 2024-12-10 23:45:42 +00:00
arikawa/api/guild.go

453 lines
16 KiB
Go
Raw Normal View History

2020-01-02 05:39:52 +00:00
package api
2020-01-04 04:19:24 +00:00
import (
2020-01-06 03:48:39 +00:00
"io"
2020-05-09 21:59:39 +00:00
"net/url"
2020-01-06 03:48:39 +00:00
2020-01-04 04:19:24 +00:00
"github.com/diamondburned/arikawa/discord" // for clarity
"github.com/diamondburned/arikawa/utils/httputil"
"github.com/diamondburned/arikawa/utils/json/option"
2020-01-04 04:19:24 +00:00
)
var EndpointGuilds = Endpoint + "guilds/"
2020-01-04 04:19:24 +00:00
// https://discordapp.com/developers/docs/resources/guild#create-guild-json-params
type CreateGuildData struct {
2020-05-11 22:06:19 +00:00
// Name is the name of the guild (2-100 characters)
2020-01-06 03:48:39 +00:00
Name string `json:"name"`
2020-05-11 22:06:19 +00:00
// VoiceRegion is the voice region id.
VoiceRegion string `json:"region,omitempty"`
// Icon is the base64 128x128 image for the guild icon.
Icon *Image `json:"image,omitempty"`
// Verification is the verification level.
Verification *discord.Verification `json:"verification_level,omitempty"`
// Notification is the default message notification level.
Notification *discord.Notification `json:"default_message_notifications,omitempty"`
// ExplicitFilter is the explicit content filter level.
ExplicitFilter *discord.ExplicitFilter `json:"explicit_content_filter,omitempty"`
// Roles are the new guild roles.
//
// When using the roles parameter, the first member of the array is used to
// change properties of the guild's @everyone role. If you are trying to
// bootstrap a guild with additional roles, keep this in mind.
//
// When using the roles parameter, the required id field within each role
// object is an integer placeholder, and will be replaced by the API upon
// consumption. Its purpose is to allow you to overwrite a role's
// permissions in a channel when also passing in channels with the channels
// array.
2020-01-06 03:48:39 +00:00
Roles []discord.Role `json:"roles,omitempty"`
2020-05-11 22:06:19 +00:00
// Channels are the new guild's channels.
// Assigning a channel to a channel category is not supported by this
// endpoint, i.e. a channel can't have the parent_id field.
//
// When using the channels parameter, the position field is ignored,
// and none of the default channels are created.
//
// When using the channels parameter, the id field within each channel
// object may be set to an integer placeholder, and will be replaced by the
// API upon consumption. Its purpose is to allow you to create
// GUILD_CATEGORY channels by setting the parent_id field on any children
// to the category's id field. Category channels must be listed before any
// children.
Channels []discord.Channel `json:"channels,omitempty"`
2020-01-06 03:48:39 +00:00
2020-05-11 22:06:19 +00:00
// AFKChannelID is the id for the afk channel.
AFKChannelID discord.Snowflake `json:"afk_channel_id,omitempty"`
// AFKTimeout is the afk timeout in seconds.
AFKTimeout option.Seconds `json:"afk_timeout,omitempty"`
2020-01-04 04:19:24 +00:00
2020-05-11 22:06:19 +00:00
// SystemChannelID is the id of the channel where guild notices such as
// welcome messages and boost events are posted.
SystemChannelID discord.Snowflake `json:"system_channel_id,omitempty"`
2020-01-04 04:19:24 +00:00
}
2020-05-11 22:06:19 +00:00
// CreateGuild creates a new guild. Returns a guild object on success.
// Fires a Guild Create Gateway event.
//
// This endpoint can be used only by bots in less than 10 guilds.
2020-01-04 04:19:24 +00:00
func (c *Client) CreateGuild(data CreateGuildData) (*discord.Guild, error) {
var g *discord.Guild
return g, c.RequestJSON(&g, "POST", Endpoint+"guilds", httputil.WithJSONBody(data))
2020-01-04 04:19:24 +00:00
}
2020-05-11 22:06:19 +00:00
// Guild returns the guild object for the given id.
// ApproximateMembers and ApproximatePresences will not be set.
2020-05-08 00:09:45 +00:00
func (c *Client) Guild(id discord.Snowflake) (*discord.Guild, error) {
2020-01-04 04:19:24 +00:00
var g *discord.Guild
2020-05-08 00:09:45 +00:00
return g, c.RequestJSON(&g, "GET", EndpointGuilds+id.String())
2020-01-04 04:19:24 +00:00
}
2020-05-11 22:06:19 +00:00
// GuildWithCount returns the guild object for the given id.
// This will also set the ApproximateMembers and ApproximatePresences fields
// of the guild struct.
2020-05-09 21:59:39 +00:00
func (c *Client) GuildWithCount(id discord.Snowflake) (*discord.Guild, error) {
var g *discord.Guild
return g, c.RequestJSON(
&g, "GET",
EndpointGuilds+id.String(),
httputil.WithSchema(c, url.Values{
"with_counts": {"true"},
}),
)
}
2020-01-16 04:27:57 +00:00
// Guilds returns all guilds, automatically paginating. Be careful, as this
// method may abuse the API by requesting thousands or millions of guilds. For
2020-05-11 22:06:19 +00:00
// lower-level access, use GuildsRange. Guilds returned have some fields
2020-01-16 04:27:57 +00:00
// filled only (ID, Name, Icon, Owner, Permissions).
2020-05-11 22:06:19 +00:00
//
// Max can be 0, in which case the function will try and fetch all guilds.
2020-01-16 04:27:57 +00:00
func (c *Client) Guilds(max uint) ([]discord.Guild, error) {
var guilds []discord.Guild
2020-01-06 05:22:26 +00:00
var after discord.Snowflake = 0
2020-01-16 04:27:57 +00:00
const hardLimit int = 100
2020-05-11 22:06:19 +00:00
unlimited := max == 0
for fetch := uint(hardLimit); max > 0 || unlimited; fetch = uint(hardLimit) {
2020-01-18 19:34:08 +00:00
if max > 0 {
if fetch > max {
fetch = max
}
max -= fetch
}
2020-01-16 04:27:57 +00:00
g, err := c.GuildsAfter(after, fetch)
2020-01-06 05:22:26 +00:00
if err != nil {
2020-01-16 04:27:57 +00:00
return guilds, err
2020-01-06 05:22:26 +00:00
}
2020-01-16 04:27:57 +00:00
guilds = append(guilds, g...)
2020-01-06 05:22:26 +00:00
2020-01-16 04:27:57 +00:00
if len(g) < hardLimit {
2020-01-06 05:22:26 +00:00
break
}
2020-01-16 04:27:57 +00:00
after = g[hardLimit-1].ID
2020-01-06 05:22:26 +00:00
}
2020-01-16 04:27:57 +00:00
return guilds, nil
2020-01-06 05:22:26 +00:00
}
2020-05-11 22:06:19 +00:00
// GuildsBefore fetches guilds before the specified ID. Check GuildsRange.
2020-04-19 21:53:53 +00:00
func (c *Client) GuildsBefore(before discord.Snowflake, limit uint) ([]discord.Guild, error) {
2020-01-16 04:27:57 +00:00
return c.GuildsRange(before, 0, limit)
2020-01-05 07:50:22 +00:00
}
2020-05-11 22:06:19 +00:00
// GuildsAfter fetches guilds after the specified ID. Check GuildsRange.
2020-04-19 21:53:53 +00:00
func (c *Client) GuildsAfter(after discord.Snowflake, limit uint) ([]discord.Guild, error) {
2020-01-16 04:27:57 +00:00
return c.GuildsRange(0, after, limit)
2020-01-05 07:50:22 +00:00
}
2020-05-11 22:06:19 +00:00
// GuildsRange returns a list of partial guild objects the current user is a
// member of. Requires the guilds OAuth2 scope.
//
// This endpoint returns 100 guilds by default, which is the maximum number
// of guilds a non-bot user can join. Therefore, pagination is not needed
// for integrations that need to get a list of the users' guilds.
2020-04-19 21:53:53 +00:00
func (c *Client) GuildsRange(before, after discord.Snowflake, limit uint) ([]discord.Guild, error) {
2020-05-11 22:06:19 +00:00
switch {
case limit == 0:
2020-01-16 04:27:57 +00:00
limit = 100
2020-05-11 22:06:19 +00:00
case limit > 100:
2020-01-16 04:27:57 +00:00
limit = 100
2020-01-05 07:50:22 +00:00
}
var param struct {
2020-05-11 18:24:28 +00:00
Before discord.Snowflake `schema:"before,omitempty"`
After discord.Snowflake `schema:"after,omitempty"`
2020-01-16 04:27:57 +00:00
Limit uint `schema:"limit"`
2020-01-05 07:50:22 +00:00
}
2020-01-16 04:27:57 +00:00
param.Before = before
param.After = after
param.Limit = limit
2020-01-05 07:50:22 +00:00
2020-01-16 04:27:57 +00:00
var gs []discord.Guild
return gs, c.RequestJSON(
&gs, "GET",
EndpointMe+"/guilds",
2020-01-06 03:48:39 +00:00
httputil.WithSchema(c, param),
2020-01-05 07:50:22 +00:00
)
}
2020-05-11 22:06:19 +00:00
// LeaveGuild leaves a guild.
2020-05-08 00:09:45 +00:00
func (c *Client) LeaveGuild(id discord.Snowflake) error {
return c.FastRequest("DELETE", EndpointMe+"/guilds/"+id.String())
2020-01-05 07:50:22 +00:00
}
2020-01-16 04:27:57 +00:00
// https://discordapp.com/developers/docs/resources/guild#modify-guild-json-params
type ModifyGuildData struct {
2020-05-11 22:06:19 +00:00
// Name is the guild's name.
Name string `json:"name,omitempty"`
// Region is the guild's voice region id.
Region option.NullableString `json:"region,omitempty"`
// Verification is the verification level.
//
// This field is nullable.
Verification *discord.Verification `json:"verification_level,omitempty"`
// Notification is the default message notification level.
//
// This field is nullable.
Notification *discord.Notification `json:"default_message_notifications,omitempty"`
// ExplicitFilter is the explicit content filter level.
//
// This field is nullable.
ExplicitFilter *discord.ExplicitFilter `json:"explicit_content_filter,omitempty"`
// AFKChannelID is the id for the afk channel.
//
// This field is nullable.
AFKChannelID discord.Snowflake `json:"afk_channel_id,string,omitempty"`
2020-05-11 22:06:19 +00:00
// AFKTimeout is the afk timeout in seconds.
AFKTimeout option.Seconds `json:"afk_timeout,omitempty"`
// Icon is the base64 1024x1024 png/jpeg/gif image for the guild icon
// (can be animated gif when the server has the ANIMATED_ICON feature).
Icon *Image `json:"icon,omitempty"`
// Splash is the base64 16:9 png/jpeg image for the guild splash
// (when the server has the INVITE_SPLASH feature).
Splash *Image `json:"splash,omitempty"`
2020-05-11 22:06:19 +00:00
// Banner is the base64 16:9 png/jpeg image for the guild banner (when the
// server has BANNER feature).
Banner *Image `json:"banner,omitempty"`
2020-01-05 07:50:22 +00:00
2020-05-11 22:06:19 +00:00
// OwnerID is the user id to transfer guild ownership to (must be owner).
OwnerID discord.Snowflake `json:"owner_id,omitempty"`
// SystemChannelID is the id of the channel where guild notices such as
// welcome messages and boost events are posted.
//
// This field is nullable.
SystemChannelID discord.Snowflake `json:"system_channel_id,omitempty"`
// RulesChannelID is the id of the channel where "PUBLIC" guilds display
// rules and/or guidelines.
//
// This field is nullable.
RulesChannelID discord.Snowflake `json:"rules_channel_id,omitempty"`
// PublicUpdatesChannelID is the id of the channel where admins and
// moderators of "PUBLIC" guilds receive notices from Discord.
//
// This field is nullable.
PublicUpdatesChannelID discord.Snowflake `json:"public_updates_channel_id,omitempty"`
2020-05-11 22:06:19 +00:00
// PreferredLocale is the preferred locale of a "PUBLIC" guild used in
// server discovery and notices from Discord.
//
// This defaults to "en-US".
PreferredLocale option.NullableString `json:"preferred_locale,omitempty"`
2020-01-05 07:50:22 +00:00
}
2020-05-11 22:06:19 +00:00
// ModifyGuild modifies a guild's settings. Requires the MANAGE_GUILD permission.
// Fires a Guild Update Gateway event.
2020-05-08 00:09:45 +00:00
func (c *Client) ModifyGuild(id discord.Snowflake, data ModifyGuildData) (*discord.Guild, error) {
2020-01-16 04:27:57 +00:00
var g *discord.Guild
return g, c.RequestJSON(
&g, "PATCH",
2020-05-08 00:09:45 +00:00
EndpointGuilds+id.String(),
httputil.WithJSONBody(data),
2020-01-05 07:50:22 +00:00
)
2020-05-11 18:24:28 +00:00
2020-01-05 07:50:22 +00:00
}
2020-05-11 22:06:19 +00:00
// DeleteGuild deletes a guild permanently. The User must be owner.
//
// Fires a Guild Delete Gateway event.
2020-05-08 00:09:45 +00:00
func (c *Client) DeleteGuild(id discord.Snowflake) error {
return c.FastRequest("DELETE", EndpointGuilds+id.String())
2020-01-06 03:48:39 +00:00
}
// GuildVoiceRegions is the same as /voice, but returns VIP ones as well if
// available.
2020-04-19 21:53:53 +00:00
func (c *Client) VoiceRegionsGuild(guildID discord.Snowflake) ([]discord.VoiceRegion, error) {
2020-01-06 03:48:39 +00:00
var vrs []discord.VoiceRegion
2020-04-19 21:53:53 +00:00
return vrs, c.RequestJSON(&vrs, "GET", EndpointGuilds+guildID.String()+"/regions")
2020-01-06 03:48:39 +00:00
}
2020-05-11 22:06:19 +00:00
// https://discord.com/developers/docs/resources/audit-log#get-guild-audit-log-query-string-parameters
2020-05-08 00:09:45 +00:00
type AuditLogData struct {
2020-05-11 22:06:19 +00:00
// UserID filters the log for actions made by a user.
UserID discord.Snowflake `schema:"user_id,omitempty"`
2020-05-11 22:06:19 +00:00
// ActionType is the type of audit log event.
ActionType discord.AuditLogEvent `schema:"action_type,omitempty"`
2020-05-11 22:06:19 +00:00
// Before filters the log before a certain entry ID.
Before discord.Snowflake `schema:"before,omitempty"`
2020-05-11 22:06:19 +00:00
// Limit limits how many entries are returned (default 50, minimum 1,
// maximum 100).
Limit uint `schema:"limit"`
2020-05-08 00:09:45 +00:00
}
2020-05-11 22:06:19 +00:00
// AuditLog returns an audit log object for the guild.
//
// Requires the VIEW_AUDIT_LOG permission.
2020-05-08 00:09:45 +00:00
func (c *Client) AuditLog(guildID discord.Snowflake, data AuditLogData) (*discord.AuditLog, error) {
switch {
case data.Limit == 0:
data.Limit = 50
case data.Limit > 100:
data.Limit = 100
}
var audit *discord.AuditLog
return audit, c.RequestJSON(
&audit, "GET",
EndpointGuilds+guildID.String()+"/audit-logs",
httputil.WithSchema(c, data),
2020-05-08 00:09:45 +00:00
)
}
2020-05-11 22:06:19 +00:00
// Integrations returns a list of integration objects for the guild.
//
// Requires the MANAGE_GUILD permission.
2020-04-19 21:53:53 +00:00
func (c *Client) Integrations(guildID discord.Snowflake) ([]discord.Integration, error) {
2020-01-06 03:48:39 +00:00
var ints []discord.Integration
2020-04-19 21:53:53 +00:00
return ints, c.RequestJSON(&ints, "GET", EndpointGuilds+guildID.String()+"/integrations")
2020-01-06 03:48:39 +00:00
}
2020-05-11 22:06:19 +00:00
// AttachIntegration attaches an integration object from the current user to
// the guild.
//
// Requires the MANAGE_GUILD permission.
// Fires a Guild Integrations Update Gateway event.
func (c *Client) AttachIntegration(guildID,
integrationID discord.Snowflake, integrationType discord.Service) error {
2020-01-06 03:48:39 +00:00
var param struct {
2020-01-15 07:34:18 +00:00
Type discord.Service `json:"type"`
ID discord.Snowflake `json:"id"`
2020-01-06 03:48:39 +00:00
}
2020-05-11 18:32:28 +00:00
param.Type = integrationType
param.ID = integrationID
2020-01-06 03:48:39 +00:00
return c.FastRequest(
"POST",
EndpointGuilds+guildID.String()+"/integrations",
httputil.WithJSONBody(param),
2020-01-06 03:48:39 +00:00
)
}
2020-05-10 23:48:26 +00:00
// https://discord.com/developers/docs/resources/guild#modify-guild-integration-json-params
type ModifyIntegrationData struct {
2020-05-11 22:06:19 +00:00
// ExpireBehavior is the behavior when an integration subscription lapses
// (see the integration expire behaviors documentation).
ExpireBehavior *discord.ExpireBehavior `json:"expire_behavior,omitempty"`
// ExpireGracePeriod is the period (in days) where the integration will
// ignore lapsed subscriptions.
ExpireGracePeriod option.NullableInt `json:"expire_grace_period,omitempty"`
// EnableEmoticons specifies whether emoticons should be synced for this
// integration (twitch only currently).
EnableEmoticons option.NullableBool `json:"enable_emoticons,omitempty"`
2020-05-10 23:48:26 +00:00
}
2020-01-06 03:48:39 +00:00
2020-05-11 22:06:19 +00:00
// ModifyIntegration modifies the behavior and settings of an integration
// object for the guild.
//
// Requires the MANAGE_GUILD permission.
// Fires a Guild Integrations Update Gateway event.
func (c *Client) ModifyIntegration(
guildID, integrationID discord.Snowflake, data ModifyIntegrationData) error {
2020-01-16 04:27:57 +00:00
return c.FastRequest(
2020-05-11 22:06:19 +00:00
"PATCH",
EndpointGuilds+guildID.String()+"/integrations/"+integrationID.String(),
2020-05-10 23:48:26 +00:00
httputil.WithJSONBody(data),
2020-01-06 03:48:39 +00:00
)
}
2020-05-11 22:06:19 +00:00
// Sync an integration. Requires the MANAGE_GUILD permission.
2020-04-19 21:53:53 +00:00
func (c *Client) SyncIntegration(guildID, integrationID discord.Snowflake) error {
2020-05-11 22:06:19 +00:00
return c.FastRequest(
"POST",
EndpointGuilds+guildID.String()+"/integrations/"+integrationID.String()+"/sync",
)
2020-01-06 03:48:39 +00:00
}
2020-05-11 22:06:19 +00:00
// GuildEmbed returns the guild embed object.
//
// Requires the MANAGE_GUILD permission.
2020-04-19 21:53:53 +00:00
func (c *Client) GuildEmbed(guildID discord.Snowflake) (*discord.GuildEmbed, error) {
2020-01-06 03:48:39 +00:00
var ge *discord.GuildEmbed
2020-04-19 21:53:53 +00:00
return ge, c.RequestJSON(&ge, "GET", EndpointGuilds+guildID.String()+"/embed")
2020-01-06 03:48:39 +00:00
}
2020-05-11 22:06:19 +00:00
// https://discord.com/developers/docs/resources/guild#guild-embed-object-guild-embed-structure
type ModifyGuildEmbedData struct {
Enabled option.Bool `json:"enabled,omitempty"`
ChannelID discord.Snowflake `json:"channel_id,omitempty"`
}
2020-04-19 21:53:53 +00:00
// ModifyGuildEmbed modifies the guild embed and updates the passed in
// GuildEmbed data.
//
// This method should be used with care: if you still want the embed enabled,
// you need to set the Enabled boolean, even if it's already enabled. If you
// don't, JSON will default it to false.
2020-05-11 22:06:19 +00:00
func (c *Client) ModifyGuildEmbed(guildID discord.Snowflake, data discord.GuildEmbed) error {
2020-04-19 21:53:53 +00:00
return c.RequestJSON(&data, "PATCH", EndpointGuilds+guildID.String()+"/embed")
2020-01-06 03:48:39 +00:00
}
2020-05-11 22:06:19 +00:00
// GuildVanityURL returns *Invite for guilds that have that feature enabled,
// but only Code and Uses are filled. Code will be "" if a vanity url for the
// guild is not set.
//
// Requires MANAGE_GUILD.
2020-04-19 21:53:53 +00:00
func (c *Client) GuildVanityURL(guildID discord.Snowflake) (*discord.Invite, error) {
2020-01-06 03:48:39 +00:00
var inv *discord.Invite
2020-04-19 21:53:53 +00:00
return inv, c.RequestJSON(&inv, "GET", EndpointGuilds+guildID.String()+"/vanity-url")
2020-01-06 03:48:39 +00:00
}
2020-05-11 22:06:19 +00:00
// https://discord.com/developers/docs/resources/guild#get-guild-widget-image-widget-style-options
type GuildImageStyle string
2020-01-06 03:48:39 +00:00
const (
2020-05-11 22:06:19 +00:00
// GuildShield is a shield style widget with Discord icon and guild members
// online count.
//
// Example: https://discordapp.com/api/guilds/81384788765712384/widget.png?style=shield
GuildShield GuildImageStyle = "shield"
// GuildBanner1 is a large image with guild icon, name and online count.
// "POWERED BY DISCORD" as the footer of the widget.
//
// Example: https://discordapp.com/api/guilds/81384788765712384/widget.png?style=banner1
GuildBanner1 GuildImageStyle = "banner1"
2020-05-11 22:06:19 +00:00
// GuildBanner2 is a smaller widget style with guild icon, name and online
// count. Split on the right with Discord logo.
//
// Example: https://discordapp.com/api/guilds/81384788765712384/widget.png?style=banner2
GuildBanner2 GuildImageStyle = "banner2"
2020-05-11 22:06:19 +00:00
// GuildBanner3 is a large image with guild icon, name and online count. In
// the footer, Discord logo on the left and "Chat Now" on the right.
//
// Example: https://discordapp.com/api/guilds/81384788765712384/widget.png?style=banner3
GuildBanner3 GuildImageStyle = "banner3"
2020-05-11 22:06:19 +00:00
// GuildBanner4 is a large Discord logo at the top of the widget.
// Guild icon, name and online count in the middle portion of the widget
// and a "JOIN MY SERVER" button at the bottom.
//
// Example: https://discordapp.com/api/guilds/81384788765712384/widget.png?style=banner4
GuildBanner4 GuildImageStyle = "banner4"
2020-01-06 03:48:39 +00:00
)
2020-05-11 22:06:19 +00:00
// GuildImageURL returns a link to the PNG image widget for the guild.
//
// Requires no permissions or authentication.
func (c *Client) GuildImageURL(guildID discord.Snowflake, img GuildImageStyle) string {
2020-04-19 21:53:53 +00:00
return EndpointGuilds + guildID.String() + "/widget.png?style=" + string(img)
2020-01-06 03:48:39 +00:00
}
2020-05-11 22:06:19 +00:00
// GuildImage returns a PNG image widget for the guild. Requires no permissions
// or authentication.
func (c *Client) GuildImage(guildID discord.Snowflake, img GuildImageStyle) (io.ReadCloser, error) {
2020-01-06 03:48:39 +00:00
r, err := c.Request("GET", c.GuildImageURL(guildID, img))
if err != nil {
return nil, err
}
2020-04-19 21:53:53 +00:00
return r.GetBody(), nil
2020-01-06 03:48:39 +00:00
}