@@ -15,7 +15,8 @@ import (
15
15
16
16
// RequestLogOptions configures the retry policy's behavior.
17
17
type RequestLogOptions struct {
18
- // LogWarningIfTryOverThreshold logs a warning if a tried operation takes longer than the specified duration (0=no logging).
18
+ // LogWarningIfTryOverThreshold logs a warning if a tried operation takes longer than the specified
19
+ // duration (-1=no logging; 0=default threshold).
19
20
LogWarningIfTryOverThreshold time.Duration
20
21
}
21
22
@@ -40,45 +41,50 @@ func (f *requestLogPolicyFactory) New(node pipeline.Node) pipeline.Policy {
40
41
type requestLogPolicy struct {
41
42
node pipeline.Node
42
43
o RequestLogOptions
43
- try int
44
+ try int32
44
45
operationStart time.Time
45
46
}
46
47
47
48
func redactSigQueryParam (rawQuery string ) (bool , string ) {
48
- sigFound := strings .EqualFold (rawQuery , "?sig=" )
49
+ rawQuery = strings .ToLower (rawQuery ) // lowercase the string so we can look for ?sig= and &sig=
50
+ sigFound := strings .Contains (rawQuery , "?sig=" )
49
51
if ! sigFound {
50
- sigFound = strings .EqualFold (rawQuery , "&sig=" )
52
+ sigFound = strings .Contains (rawQuery , "&sig=" )
51
53
if ! sigFound {
52
54
return sigFound , rawQuery // [?|&]sig= not found; return same rawQuery passed in (no memory allocation)
53
55
}
54
56
}
55
- // [?|&]sig= was found, redact its value
57
+ // [?|&]sig= found, redact its value
56
58
values , _ := url .ParseQuery (rawQuery )
57
59
for name := range values {
58
60
if strings .EqualFold (name , "sig" ) {
59
- values [name ] = []string {"(redacted) " }
61
+ values [name ] = []string {"REDACTED " }
60
62
}
61
63
}
62
64
return sigFound , values .Encode ()
63
65
}
64
66
67
+ func prepareRequestForLogging (request pipeline.Request ) * http.Request {
68
+ req := request
69
+ if sigFound , rawQuery := redactSigQueryParam (req .URL .RawQuery ); sigFound {
70
+ // Make copy so we don't destroy the query parameters we actually need to send in the request
71
+ req = request .Copy ()
72
+ req .Request .URL .RawQuery = rawQuery
73
+ }
74
+ return req .Request
75
+ }
76
+
65
77
func (p * requestLogPolicy ) Do (ctx context.Context , request pipeline.Request ) (response pipeline.Response , err error ) {
66
78
p .try ++ // The first try is #1 (not #0)
67
79
if p .try == 1 {
68
80
p .operationStart = time .Now () // If this is the 1st try, record the operation state time
69
81
}
70
82
71
83
// Log the outgoing request as informational
72
- if p .node .WouldLog (pipeline .LogInfo ) {
84
+ if p .node .ShouldLog (pipeline .LogInfo ) {
73
85
b := & bytes.Buffer {}
74
86
fmt .Fprintf (b , "==> OUTGOING REQUEST (Try=%d)\n " , p .try )
75
- req := request
76
- if sigFound , rawQuery := redactSigQueryParam (req .URL .RawQuery ); sigFound {
77
- // TODO: Make copy so we dont' destroy the query parameters we actually need to send in the request
78
- req = request .Copy ()
79
- req .Request .URL .RawQuery = rawQuery
80
- }
81
- pipeline .WriteRequest (b , req .Request )
87
+ pipeline .WriteRequest (b , prepareRequestForLogging (request ))
82
88
p .node .Log (pipeline .LogInfo , b .String ())
83
89
}
84
90
@@ -92,16 +98,18 @@ func (p *requestLogPolicy) Do(ctx context.Context, request pipeline.Request) (re
92
98
severity := pipeline .LogInfo // Assume success and default to informational logging
93
99
logMsg := func (b * bytes.Buffer ) {
94
100
b .WriteString ("SUCCESSFUL OPERATION\n " )
95
- pipeline .WriteResponseWithRequest ( b , response .Response ())
101
+ pipeline .WriteRequestWithResponse ( b , prepareRequestForLogging ( request ) , response .Response ())
96
102
}
97
103
104
+ forceLog := false
98
105
// If the response took too long, we'll upgrade to warning.
99
106
if p .o .LogWarningIfTryOverThreshold > 0 && tryDuration > p .o .LogWarningIfTryOverThreshold {
100
107
// Log a warning if the try duration exceeded the specified threshold
101
108
severity = pipeline .LogWarning
102
109
logMsg = func (b * bytes.Buffer ) {
103
110
fmt .Fprintf (b , "SLOW OPERATION [tryDuration > %v]\n " , p .o .LogWarningIfTryOverThreshold )
104
- pipeline .WriteResponseWithRequest (b , response .Response ())
111
+ pipeline .WriteRequestWithResponse (b , prepareRequestForLogging (request ), response .Response ())
112
+ forceLog = true // For CSS (Customer Support Services), we always log these to help diagnose latency issues
105
113
}
106
114
}
107
115
@@ -111,9 +119,10 @@ func (p *requestLogPolicy) Do(ctx context.Context, request pipeline.Request) (re
111
119
severity = pipeline .LogError // Promote to Error any 4xx (except those listed is an error) or any 5xx
112
120
logMsg = func (b * bytes.Buffer ) {
113
121
// Write the error, the originating request and the stack
114
- fmt .Fprintf (b , "OPERATION ERROR:\n %v \n " , err )
115
- pipeline .WriteResponseWithRequest ( b , response .Response ())
122
+ fmt .Fprintf (b , "OPERATION ERROR:\n " )
123
+ pipeline .WriteRequestWithResponse ( b , prepareRequestForLogging ( request ) , response .Response ())
116
124
b .Write (stack ()) // For errors, we append the stack trace (an expensive operation)
125
+ forceLog = true // TODO: Do we really want this here?
117
126
}
118
127
} else {
119
128
// For other status codes, we leave the severity as is.
@@ -123,17 +132,25 @@ func (p *requestLogPolicy) Do(ctx context.Context, request pipeline.Request) (re
123
132
logMsg = func (b * bytes.Buffer ) {
124
133
// Write the error, the originating request and the stack
125
134
fmt .Fprintf (b , "NETWORK ERROR:\n %v\n " , err )
126
- pipeline .WriteRequest (b , request . Request )
135
+ pipeline .WriteRequest (b , prepareRequestForLogging ( request ) )
127
136
b .Write (stack ()) // For errors, we append the stack trace (an expensive operation)
137
+ forceLog = true
128
138
}
129
139
}
130
140
131
- if p .node .WouldLog (severity ) || false { // Change false to true for testing
141
+ if shouldLog := p .node .ShouldLog (severity ); forceLog || shouldLog {
132
142
// We're going to log this; build the string to log
133
143
b := & bytes.Buffer {}
134
144
fmt .Fprintf (b , "==> REQUEST/RESPONSE (Try=%d, TryDuration=%v, OpDuration=%v) -- " , p .try , tryDuration , opDuration )
135
145
logMsg (b )
136
- p .node .Log (severity , b .String ())
146
+ msg := b .String ()
147
+
148
+ if forceLog {
149
+ pipeline .ForceLog (severity , msg )
150
+ }
151
+ if shouldLog {
152
+ p .node .Log (severity , msg )
153
+ }
137
154
}
138
155
return response , err
139
156
}
0 commit comments