client.go 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257
  1. /******************************************************************************
  2. * Discordgo by Bruce Marriner <bruce@sqls.net>
  3. * A Discord API for Golang.
  4. * See discord.go for more information.
  5. *
  6. * This file contains functions for interacting with the Discord API
  7. * at the lowest level. See other files for easier methods of access.
  8. */
  9. package discordgo
  10. import (
  11. "bytes"
  12. "encoding/json"
  13. "errors"
  14. "fmt"
  15. "io/ioutil"
  16. "net/http"
  17. "time"
  18. )
  19. // Request makes a REST API GET Request with Discord.
  20. // TODO make this handle GET, POST, DELETE, etc
  21. func Request(session *Session, urlStr string) (body []byte, err error) {
  22. req, err := http.NewRequest("GET", urlStr, bytes.NewBuffer([]byte(fmt.Sprintf(``))))
  23. if err != nil {
  24. return
  25. }
  26. req.Header.Set("authorization", session.Token)
  27. req.Header.Set("Content-Type", "application/json")
  28. client := &http.Client{Timeout: (20 * time.Second)}
  29. resp, err := client.Do(req)
  30. if err != nil {
  31. return
  32. }
  33. body, err = ioutil.ReadAll(resp.Body)
  34. resp.Body.Close()
  35. if err != nil {
  36. return
  37. }
  38. if resp.StatusCode != 200 {
  39. err = errors.New(fmt.Sprintf("StatusCode: %d, %s", resp.StatusCode, string(body)))
  40. return
  41. }
  42. if session.Debug {
  43. var prettyJSON bytes.Buffer
  44. error := json.Indent(&prettyJSON, body, "", "\t")
  45. if error != nil {
  46. fmt.Print("JSON parse error: ", error)
  47. return
  48. }
  49. fmt.Println(urlStr+" Response:\n", string(prettyJSON.Bytes()))
  50. }
  51. return
  52. }
  53. // Login asks the Discord server for an authentication token
  54. func Login(session *Session, email string, password string) (token string, err error) {
  55. var urlStr string = fmt.Sprintf("%s/%s", discordApi, "auth/login")
  56. req, err := http.NewRequest("POST", urlStr, bytes.NewBuffer([]byte(fmt.Sprintf(`{"email":"%s", "password":"%s"}`, email, password))))
  57. if err != nil {
  58. return
  59. }
  60. req.Header.Set("Content-Type", "application/json")
  61. client := &http.Client{Timeout: (20 * time.Second)}
  62. resp, err := client.Do(req)
  63. if err != nil {
  64. return
  65. }
  66. defer resp.Body.Close()
  67. body, _ := ioutil.ReadAll(resp.Body)
  68. if resp.StatusCode != 200 {
  69. err = errors.New(fmt.Sprintf("StatusCode: %d, %s", resp.StatusCode, string(body)))
  70. return
  71. }
  72. if session.Debug {
  73. var prettyJSON bytes.Buffer
  74. error := json.Indent(&prettyJSON, body, "", "\t")
  75. if error != nil {
  76. fmt.Print("JSON parse error: ", error)
  77. return
  78. }
  79. fmt.Println("requestToken Response:\n", string(prettyJSON.Bytes()))
  80. }
  81. temp := &Session{} // TODO Must be a better way
  82. err = json.Unmarshal(body, &temp)
  83. token = temp.Token
  84. return
  85. }
  86. // Returns the user details of the given userId
  87. // session : An active session connection to Discord
  88. // user : A user Id or name
  89. func Users(session *Session, userId string) (user User, err error) {
  90. body, err := Request(session, fmt.Sprintf("%s/users/%s", discordApi, userId))
  91. err = json.Unmarshal(body, &user)
  92. return
  93. }
  94. // PrivateChannels returns an array of Channel structures for all private
  95. // channels for a user
  96. func PrivateChannels(session *Session, userId string) (channels []Channel, err error) {
  97. body, err := Request(session, fmt.Sprintf("%s/%s", discordApi, fmt.Sprintf("users/%s/channels", userId)))
  98. err = json.Unmarshal(body, &channels)
  99. return
  100. }
  101. // Servers returns an array of Server structures for all servers for a user
  102. func Servers(session *Session, userId string) (servers []Server, err error) {
  103. body, err := Request(session, fmt.Sprintf("%s/users/%s/guilds", discordApi, userId))
  104. err = json.Unmarshal(body, &servers)
  105. return
  106. }
  107. // Members returns an array of Member structures for all members of a given
  108. // server.
  109. func Members(session *Session, serverId int) (members []Member, err error) {
  110. body, err := Request(session, fmt.Sprintf("%s/guilds/%d/members", discordApi, serverId))
  111. err = json.Unmarshal(body, &members)
  112. return
  113. }
  114. // Channels returns an array of Channel structures for all channels of a given
  115. // server.
  116. func Channels(session *Session, serverId int) (channels []Channel, err error) {
  117. body, err := Request(session, fmt.Sprintf("%s/guilds/%d/channels", discordApi, serverId))
  118. err = json.Unmarshal(body, &channels)
  119. return
  120. }
  121. // Messages returns an array of Message structures for messaages within a given
  122. // channel. limit, beforeId, and afterId can be used to control what messages
  123. // are returned.
  124. func Messages(session *Session, channelId int, limit int, afterId int, beforeId int) (messages []Message, err error) {
  125. var urlStr string
  126. if limit > 0 {
  127. urlStr = fmt.Sprintf("%s/channels/%d/messages?limit=%d", discordApi, channelId, limit)
  128. }
  129. if afterId > 0 {
  130. if urlStr != "" {
  131. urlStr = urlStr + fmt.Sprintf("&after=%d", afterId)
  132. } else {
  133. urlStr = fmt.Sprintf("%s/channels/%d/messages?after=%d", discordApi, channelId, afterId)
  134. }
  135. }
  136. if beforeId > 0 {
  137. if urlStr != "" {
  138. urlStr = urlStr + fmt.Sprintf("&before=%d", beforeId)
  139. } else {
  140. urlStr = fmt.Sprintf("%s/channels/%d/messages?after=%d", discordApi, channelId, beforeId)
  141. }
  142. }
  143. if urlStr == "" {
  144. urlStr = fmt.Sprintf("%s/channels/%d/messages", discordApi, channelId)
  145. }
  146. fmt.Println(urlStr)
  147. body, err := Request(session, urlStr)
  148. err = json.Unmarshal(body, &messages)
  149. return
  150. }
  151. // SendMessage sends a message to the given channel.
  152. func SendMessage(session *Session, channelId int, message string) (response Message, err error) {
  153. var urlStr string = fmt.Sprintf("%s/channels/%d/messages", discordApi, channelId)
  154. req, err := http.NewRequest("POST", urlStr, bytes.NewBuffer([]byte(fmt.Sprintf(`{"content":"%s"}`, message))))
  155. if err != nil {
  156. return
  157. }
  158. req.Header.Set("authorization", session.Token)
  159. req.Header.Set("Content-Type", "application/json")
  160. client := &http.Client{Timeout: (20 * time.Second)}
  161. resp, err := client.Do(req)
  162. if err != nil {
  163. return
  164. }
  165. body, err := ioutil.ReadAll(resp.Body)
  166. if err != nil {
  167. return
  168. }
  169. resp.Body.Close()
  170. if resp.StatusCode != 200 {
  171. err = errors.New(fmt.Sprintf("StatusCode: %d, %s", resp.StatusCode, string(body)))
  172. return
  173. }
  174. if session.Debug {
  175. var prettyJSON bytes.Buffer
  176. error := json.Indent(&prettyJSON, body, "", "\t")
  177. if error != nil {
  178. fmt.Print("JSON parse error: ", error)
  179. return
  180. }
  181. fmt.Println(urlStr+" Response:\n", string(prettyJSON.Bytes()))
  182. }
  183. err = json.Unmarshal(body, &response)
  184. return
  185. }
  186. // Close ends a session and logs out from the Discord REST API.
  187. func Logout(session *Session) (err error) {
  188. req, err := http.NewRequest("POST", fmt.Sprintf("%s/%s", discordApi, fmt.Sprintf("auth/logout")), bytes.NewBuffer([]byte(fmt.Sprintf(``))))
  189. if err != nil {
  190. return
  191. }
  192. req.Header.Set("authorization", session.Token)
  193. req.Header.Set("Content-Type", "application/json")
  194. client := &http.Client{Timeout: (20 * time.Second)}
  195. resp, err := client.Do(req)
  196. if err != nil {
  197. return
  198. }
  199. body, err := ioutil.ReadAll(resp.Body)
  200. if err != nil {
  201. return
  202. }
  203. resp.Body.Close()
  204. if resp.StatusCode != 204 && resp.StatusCode != 200 {
  205. err = errors.New(fmt.Sprintf("StatusCode: %d, %s", resp.StatusCode, string(body)))
  206. return
  207. }
  208. return
  209. }