Skip to content

Commit e1a7e30

Browse files
committed
Add special sentry field support.
1 parent e42a087 commit e1a7e30

File tree

4 files changed

+203
-30
lines changed

4 files changed

+203
-30
lines changed

README.md

+15-8
Original file line numberDiff line numberDiff line change
@@ -87,14 +87,21 @@ hook, err := NewWithClientSentryHook(client, []logrus.Level{
8787

8888
## Special fields
8989

90-
Some logrus fields have a special meaning in this hook,
91-
these are `server_name`, `logger` and `http_request`.
92-
When logs are sent to sentry these fields are treated differently.
93-
- `server_name` (also known as hostname) is the name of the server which
94-
is logging the event (hostname.example.com)
95-
- `logger` is the part of the application which is logging the event.
96-
In go this usually means setting it to the name of the package.
97-
- `http_request` is the in-coming request(*http.Request). The detailed request data are sent to Sentry.
90+
Some logrus fields have a special meaning in this hook, and they will be especially processed by Sentry.
91+
92+
93+
| Field key | Description |
94+
| ------------- | ------------- |
95+
| `event_id` | Each logged event is identified by the `event_id`, which is hexadecimal string representing a UUID4 value. You can manually specify the identifier of a log event by supplying this field. The `event_id` string should be in one of the following UUID format: `xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx` `xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx` and `urn:uuid:xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx`)|
96+
| `user_name` | Name of the user who is in the context of the event |
97+
| `user_email` | Email of the user who is in the context of the event |
98+
| `user_id` | ID of the user who is in the context of the event |
99+
| `user_ip` | IP of the user who is in the context of the event |
100+
| `server_name` | Also known as hostname, it is the name of the server which is logging the event (hostname.example.com) |
101+
| `logger` | `logger` is the part of the application which is logging the event. In go this usually means setting it to the name of the package. |
102+
| `http_request` | `http_request` is the in-coming request(*http.Request). The detailed request data are sent to Sentry. |
103+
104+
98105

99106
## Timeout
100107

sentry.go

+51-20
Original file line numberDiff line numberDiff line change
@@ -20,37 +20,62 @@ var (
2020
}
2121
)
2222

23-
func getAndDel(d logrus.Fields, key string) (string, bool) {
24-
var (
25-
ok bool
26-
v interface{}
27-
val string
28-
)
29-
if v, ok = d[key]; !ok {
23+
func getEventID(d logrus.Fields) (string, bool) {
24+
eventID, ok := d["event_id"].(string)
25+
26+
if !ok {
3027
return "", false
3128
}
3229

33-
if val, ok = v.(string); !ok {
30+
//verify eventID is 32 characters hexadecimal string (UUID4)
31+
uuid := parseUUID(eventID)
32+
33+
if uuid == nil {
3434
return "", false
3535
}
36-
delete(d, key)
37-
return val, true
36+
37+
return uuid.noDashString(), true
3838
}
3939

40-
func getAndDelRequest(d logrus.Fields, key string) (*http.Request, bool) {
41-
var (
42-
ok bool
43-
v interface{}
44-
req *http.Request
45-
)
46-
if v, ok = d[key]; !ok {
40+
func getUserContext(d logrus.Fields) (*raven.User, bool) {
41+
if v, ok := d["user"]; ok {
42+
switch val := v.(type) {
43+
case *raven.User:
44+
return val, true
45+
46+
case raven.User:
47+
return &val, true
48+
}
49+
}
50+
51+
username, _ := d["user_name"].(string)
52+
email, _ := d["user_email"].(string)
53+
id, _ := d["user_id"].(string)
54+
ip, _ := d["user_ip"].(string)
55+
56+
if username == "" && email == "" && id == "" && ip == "" {
4757
return nil, false
4858
}
49-
if req, ok = v.(*http.Request); !ok || req == nil {
59+
60+
return &raven.User{id, username, email, ip}, true
61+
}
62+
63+
func getAndDel(d logrus.Fields, key string) (string, bool) {
64+
if value, ok := d[key].(string); ok {
65+
delete(d, key)
66+
return value, true
67+
} else {
68+
return "", false
69+
}
70+
}
71+
72+
func getAndDelRequest(d logrus.Fields, key string) (*http.Request, bool) {
73+
if value, ok := d[key].(*http.Request); ok {
74+
delete(d, key)
75+
return value, true
76+
} else {
5077
return nil, false
5178
}
52-
delete(d, key)
53-
return req, true
5479
}
5580

5681
// SentryHook delivers logs to a sentry server.
@@ -143,6 +168,12 @@ func (hook *SentryHook) Fire(entry *logrus.Entry) error {
143168
if req, ok := getAndDelRequest(d, "http_request"); ok {
144169
packet.Interfaces = append(packet.Interfaces, raven.NewHttp(req))
145170
}
171+
if user, ok := getUserContext(d); ok {
172+
packet.Interfaces = append(packet.Interfaces, user)
173+
}
174+
if eventID, ok := getEventID(d); ok {
175+
packet.EventID = eventID
176+
}
146177
stConfig := &hook.StacktraceConfiguration
147178
if stConfig.Enable && entry.Level <= stConfig.Level {
148179
currentStacktrace := raven.NewStacktrace(stConfig.Skip, stConfig.Context, stConfig.InAppPrefixes)

sentry_test.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,7 @@ func getTestLogger() *logrus.Logger {
3333
// so need to explicitly construct one for purpose of test
3434
type resultPacket struct {
3535
raven.Packet
36-
Stacktrace raven.Stacktrace `json:stacktrace`
36+
Stacktrace raven.Stacktrace `json:"stacktrace"`
3737
}
3838

3939
func WithTestDSN(t *testing.T, tf func(string, <-chan *resultPacket)) {
@@ -206,7 +206,7 @@ func TestSentryStacktrace(t *testing.T) {
206206
t.Errorf("File name should have ended with %s, was %s", expectedSuffix, lastFrame.Filename)
207207
}
208208
if lastFrame.Lineno != expectedLineno {
209-
t.Errorf("Line number should have been %s, was %s", expectedLineno, lastFrame.Lineno)
209+
t.Errorf("Line number should have been %d, was %d", expectedLineno, lastFrame.Lineno)
210210
}
211211
if lastFrame.InApp {
212212
t.Error("Frame should not be identified as in_app without prefixes")

utils.go

+135
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,135 @@
1+
package logrus_sentry
2+
3+
import (
4+
"fmt"
5+
"strings"
6+
)
7+
8+
/*
9+
Copyright (c) 2009,2014 Google Inc. All rights reserved.
10+
11+
Redistribution and use in source and binary forms, with or without
12+
modification, are permitted provided that the following conditions are
13+
met:
14+
15+
* Redistributions of source code must retain the above copyright
16+
notice, this list of conditions and the following disclaimer.
17+
* Redistributions in binary form must reproduce the above
18+
copyright notice, this list of conditions and the following disclaimer
19+
in the documentation and/or other materials provided with the
20+
distribution.
21+
* Neither the name of Google Inc. nor the names of its
22+
contributors may be used to endorse or promote products derived from
23+
this software without specific prior written permission.
24+
25+
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
26+
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
27+
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
28+
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
29+
OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
30+
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
31+
LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
32+
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
33+
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
34+
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
35+
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
36+
*/
37+
38+
// A UUID is a 128 bit (16 byte) Universal Unique IDentifier as defined in RFC
39+
// 4122.
40+
type uuid []byte
41+
42+
// parseUUID decodes s into a UUID or returns nil. Both the UUID form of
43+
// xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx and
44+
// xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx and
45+
// urn:uuid:xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx are decoded.
46+
func parseUUID(s string) uuid {
47+
//If it is in no dash format "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"
48+
if len(s) == 32 {
49+
uuid := make([]byte, 16)
50+
for i, x := range []int{
51+
0, 2, 4, 6, 8, 10,
52+
12, 14, 16, 18, 20,
53+
22, 24, 26, 28, 30} {
54+
if v, ok := xtob(s[x:]); !ok {
55+
return nil
56+
} else {
57+
uuid[i] = v
58+
}
59+
}
60+
return uuid
61+
}
62+
63+
if len(s) == 36+9 {
64+
if strings.ToLower(s[:9]) != "urn:uuid:" {
65+
return nil
66+
}
67+
s = s[9:]
68+
} else if len(s) != 36 {
69+
return nil
70+
}
71+
if s[8] != '-' || s[13] != '-' || s[18] != '-' || s[23] != '-' {
72+
return nil
73+
}
74+
uuid := make([]byte, 16)
75+
for i, x := range []int{
76+
0, 2, 4, 6,
77+
9, 11,
78+
14, 16,
79+
19, 21,
80+
24, 26, 28, 30, 32, 34} {
81+
if v, ok := xtob(s[x:]); !ok {
82+
return nil
83+
} else {
84+
uuid[i] = v
85+
}
86+
}
87+
return uuid
88+
}
89+
90+
// String returns the string form of uuid, xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx
91+
// , or "" if uuid is invalid.
92+
func (uuid uuid) string() string {
93+
if uuid == nil || len(uuid) != 16 {
94+
return ""
95+
}
96+
b := []byte(uuid)
97+
return fmt.Sprintf("%08x-%04x-%04x-%04x-%012x",
98+
b[:4], b[4:6], b[6:8], b[8:10], b[10:])
99+
}
100+
101+
func (uuid uuid) noDashString() string {
102+
if uuid == nil || len(uuid) != 16 {
103+
return ""
104+
}
105+
b := []byte(uuid)
106+
return fmt.Sprintf("%08x%04x%04x%04x%012x",
107+
b[:4], b[4:6], b[6:8], b[8:10], b[10:])
108+
}
109+
110+
// xvalues returns the value of a byte as a hexadecimal digit or 255.
111+
var xvalues = []byte{
112+
255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
113+
255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
114+
255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
115+
0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 255, 255, 255, 255, 255, 255,
116+
255, 10, 11, 12, 13, 14, 15, 255, 255, 255, 255, 255, 255, 255, 255, 255,
117+
255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
118+
255, 10, 11, 12, 13, 14, 15, 255, 255, 255, 255, 255, 255, 255, 255, 255,
119+
255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
120+
255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
121+
255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
122+
255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
123+
255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
124+
255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
125+
255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
126+
255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
127+
255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,
128+
}
129+
130+
// xtob converts the the first two hex bytes of x into a byte.
131+
func xtob(x string) (byte, bool) {
132+
b1 := xvalues[x[0]]
133+
b2 := xvalues[x[1]]
134+
return (b1 << 4) | b2, b1 != 255 && b2 != 255
135+
}

0 commit comments

Comments
 (0)