-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdiscord.go
122 lines (103 loc) · 2.96 KB
/
discord.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
package main
import (
"bytes"
"encoding/json"
"fmt"
"io/ioutil"
"net/http"
"time"
)
type Author struct {
Name string `json:"name"`
}
type Provider struct {
Name string `json:"name"`
}
type Field struct {
Name string `json:"name"`
Value string `json:"value"`
Inline bool `json:"inline,omitempty"`
}
// https://discord.com/developers/docs/resources/channel#embed-object
type Embed struct {
Title string `json:"title,omitempty"`
Description string `json:"description,omitempty"`
Author Author `json:"author,omitempty"`
Provider Provider `json:"provider,omitempty"`
Fields []Field `json:"fields,omitempty"`
}
type DiscordMessage struct {
Content string `json:"content,omitempty"`
Embeds []Embed `json:"embeds,omitempty"`
}
type DiscordWebhook struct {
Url string
MessageQueue chan DiscordMessage
}
type RateLimitResponse struct {
Global bool `json:"global"`
Message string `json:"message"`
RetryAfter int `json:"retry_after"`
}
func NewDiscordWebhook(url string) *DiscordWebhook {
// TODO This could be a weird edge case, though it hopefully isn't too likely
c := make(chan DiscordMessage, 1000)
d := DiscordWebhook{Url: url, MessageQueue: c}
go d.ProcessMessages()
return &d
}
func (d *DiscordWebhook) ProcessMessages() {
for m := range d.MessageQueue {
// Retry for any rate limits
// This could also be approached with re-queueing messages, but the naive
// solution to that (just doing c <- message here) results in a possible deadlock
for {
delay := d.sendMessage(m)
if delay == 0 {
break
}
fmt.Printf("Rate Limited: Delaying for %f seconds\n", delay.Seconds())
time.Sleep(delay)
}
}
}
func (d *DiscordWebhook) Close() {
close(d.MessageQueue)
}
// The design of this doesn't make a ton of sense, but it returns the duration
// that we need to wait until the next request. If 0, we can proceed. If we
// get rate-limited, this will be non-zero and tell us how long to wait (along
// with implicitly telling us that the sending failed).
func (d *DiscordWebhook) sendMessage(message DiscordMessage) time.Duration {
body, err := json.Marshal(message)
if err != nil {
fmt.Print("Failed to marshal JSON", err)
return 0
}
resp, err := http.Post(
d.Url,
"application/json",
bytes.NewReader(body),
)
if err != nil {
fmt.Println("ERROR", err)
return 0
}
body, _ = ioutil.ReadAll(resp.Body)
if resp.StatusCode != 204 {
fmt.Println("Request failed", resp.StatusCode, string(body))
}
if resp.StatusCode == 429 {
var content RateLimitResponse
err := json.Unmarshal(body, &content)
if err != nil {
fmt.Printf("ERROR: Could not parse delay time: %s. Defaulting to 10 seconds\n", err)
return 10 * time.Second
}
// The API docs say that this field is in seconds, but values that I have
// been getting are >1000, so I'm going to assume milliseconds. time.Duration
// takes nanoseconds, so we convert here.
return time.Duration(content.RetryAfter * 1e6)
}
return 0
}