-
Notifications
You must be signed in to change notification settings - Fork 71
/
Copy pathchannel_log.go
126 lines (101 loc) · 4.17 KB
/
channel_log.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
123
124
125
126
package courier
import (
"fmt"
"github.com/nyaruka/courier/utils/clogs"
"github.com/nyaruka/gocommon/httpx"
)
const (
ChannelLogTypeUnknown clogs.LogType = "unknown"
ChannelLogTypeMsgSend clogs.LogType = "msg_send"
ChannelLogTypeMsgStatus clogs.LogType = "msg_status"
ChannelLogTypeMsgReceive clogs.LogType = "msg_receive"
ChannelLogTypeEventReceive clogs.LogType = "event_receive"
ChannelLogTypeMultiReceive clogs.LogType = "multi_receive"
ChannelLogTypeAttachmentFetch clogs.LogType = "attachment_fetch"
ChannelLogTypeTokenRefresh clogs.LogType = "token_refresh"
ChannelLogTypePageSubscribe clogs.LogType = "page_subscribe"
ChannelLogTypeWebhookVerify clogs.LogType = "webhook_verify"
)
func ErrorResponseStatusCode() *clogs.LogError {
return clogs.NewLogError("response_status_code", "", "Unexpected response status code.")
}
func ErrorResponseUnparseable(format string) *clogs.LogError {
return clogs.NewLogError("response_unparseable", "", "Unable to parse response as %s.", format)
}
func ErrorResponseUnexpected(expected string) *clogs.LogError {
return clogs.NewLogError("response_unexpected", "", "Expected response to be '%s'.", expected)
}
func ErrorResponseValueMissing(key string) *clogs.LogError {
return clogs.NewLogError("response_value_missing", "", "Unable to find '%s' response.", key)
}
func ErrorMediaUnsupported(contentType string) *clogs.LogError {
return clogs.NewLogError("media_unsupported", "", "Unsupported attachment media type: %s.", contentType)
}
// ErrorMediaUnresolveable is used when media is unresolveable due to the channel's specific requirements
func ErrorMediaUnresolveable(contentType string) *clogs.LogError {
return clogs.NewLogError("media_unresolveable", "", "Unable to find version of %s attachment compatible with channel.", contentType)
}
func ErrorAttachmentNotDecodable() *clogs.LogError {
return clogs.NewLogError("attachment_not_decodable", "", "Unable to decode embedded attachment data.")
}
func ErrorExternal(code, message string) *clogs.LogError {
if message == "" {
message = fmt.Sprintf("Service specific error: %s.", code)
}
return clogs.NewLogError("external", code, message)
}
// ChannelLog stores the HTTP traces and errors generated by an interaction with a channel.
type ChannelLog struct {
*clogs.Log
channel Channel
attached bool
}
// NewChannelLogForIncoming creates a new channel log for an incoming request, the type of which won't be known
// until the handler completes.
func NewChannelLogForIncoming(logType clogs.LogType, ch Channel, r *httpx.Recorder, redactVals []string) *ChannelLog {
return newChannelLog(logType, ch, r, false, redactVals)
}
// NewChannelLogForSend creates a new channel log for a message send
func NewChannelLogForSend(msg MsgOut, redactVals []string) *ChannelLog {
return newChannelLog(ChannelLogTypeMsgSend, msg.Channel(), nil, true, redactVals)
}
// NewChannelLogForSend creates a new channel log for an attachment fetch
func NewChannelLogForAttachmentFetch(ch Channel, redactVals []string) *ChannelLog {
return newChannelLog(ChannelLogTypeAttachmentFetch, ch, nil, true, redactVals)
}
// NewChannelLog creates a new channel log with the given type and channel
func NewChannelLog(t clogs.LogType, ch Channel, redactVals []string) *ChannelLog {
return newChannelLog(t, ch, nil, false, redactVals)
}
func newChannelLog(t clogs.LogType, ch Channel, r *httpx.Recorder, attached bool, redactVals []string) *ChannelLog {
return &ChannelLog{
Log: clogs.NewLog(t, r, redactVals),
channel: ch,
attached: attached,
}
}
// Deprecated: channel handlers should add user-facing error messages via .Error() instead
func (l *ChannelLog) RawError(err error) {
l.Error(clogs.NewLogError("", "", err.Error()))
}
func (l *ChannelLog) Channel() Channel {
return l.channel
}
func (l *ChannelLog) Attached() bool {
return l.attached
}
func (l *ChannelLog) SetAttached(a bool) {
l.attached = a
}
// if we have an error or a non 2XX/3XX http response then log is considered an error
func (l *ChannelLog) IsError() bool {
if len(l.Errors) > 0 {
return true
}
for _, l := range l.HttpLogs {
if l.StatusCode < 200 || l.StatusCode >= 400 {
return true
}
}
return false
}