28
28
from ldclient .interfaces import EventProcessor
29
29
from ldclient .repeating_timer import RepeatingTimer
30
30
from ldclient .util import UnsuccessfulResponseException
31
- from ldclient .util import _headers , _retryable_statuses
32
31
from ldclient .util import log
33
- from ldclient .util import http_error_message , is_http_error_recoverable , stringify_attrs , throw_if_unsuccessful_response
32
+ from ldclient .util import check_if_error_is_recoverable_and_log , is_http_error_recoverable , stringify_attrs , throw_if_unsuccessful_response , _headers
34
33
from ldclient .diagnostics import create_diagnostic_init
35
34
36
35
__MAX_FLUSH_THREADS__ = 5
@@ -141,18 +140,6 @@ def _get_userkey(self, event):
141
140
return str (event ['user' ].get ('key' ))
142
141
143
142
144
- class _EventRetry (urllib3 .Retry ):
145
- def __init__ (self ):
146
- urllib3 .Retry .__init__ (self , total = 1 ,
147
- method_whitelist = False , # Enable retry on POST
148
- status_forcelist = _retryable_statuses ,
149
- raise_on_status = False )
150
-
151
- # Override backoff time to be flat 1 second
152
- def get_backoff_time (self ):
153
- return 1
154
-
155
-
156
143
class EventPayloadSendTask (object ):
157
144
def __init__ (self , http , config , formatter , payload , response_fn ):
158
145
self ._http = http
@@ -175,16 +162,17 @@ def _do_send(self, output_events):
175
162
try :
176
163
json_body = json .dumps (output_events )
177
164
log .debug ('Sending events payload: ' + json_body )
178
- hdrs = _headers (self ._config )
179
- hdrs ['X-LaunchDarkly-Event-Schema' ] = str (__CURRENT_EVENT_SCHEMA__ )
180
- hdrs ['X-LaunchDarkly-Payload-ID' ] = str (uuid .uuid4 ())
181
- uri = self ._config .events_uri
182
- r = self ._http .request ('POST' , uri ,
183
- headers = hdrs ,
184
- timeout = urllib3 .Timeout (connect = self ._config .connect_timeout , read = self ._config .read_timeout ),
185
- body = json_body ,
186
- retries = _EventRetry ())
187
- self ._response_fn (r )
165
+ payload_id = str (uuid .uuid4 ())
166
+ r = _post_events_with_retry (
167
+ self ._http ,
168
+ self ._config ,
169
+ self ._config .events_uri ,
170
+ payload_id ,
171
+ json_body ,
172
+ "%d events" % len (self ._payload .events )
173
+ )
174
+ if r :
175
+ self ._response_fn (r )
188
176
return r
189
177
except Exception as e :
190
178
log .warning (
@@ -202,13 +190,14 @@ def run(self):
202
190
try :
203
191
json_body = json .dumps (self ._event_body )
204
192
log .debug ('Sending diagnostic event: ' + json_body )
205
- hdrs = _headers (self ._config )
206
- uri = self ._config .events_base_uri + '/diagnostic'
207
- r = self ._http .request ('POST' , uri ,
208
- headers = hdrs ,
209
- timeout = urllib3 .Timeout (connect = self ._config .connect_timeout , read = self ._config .read_timeout ),
210
- body = json_body ,
211
- retries = 1 )
193
+ _post_events_with_retry (
194
+ self ._http ,
195
+ self ._config ,
196
+ self ._config .events_base_uri + '/diagnostic' ,
197
+ None ,
198
+ json_body ,
199
+ "diagnostic event"
200
+ )
212
201
except Exception as e :
213
202
log .warning (
214
203
'Unhandled exception in event processor. Diagnostic event was not sent. [%s]' , e )
@@ -381,11 +370,9 @@ def _handle_response(self, r):
381
370
if server_date is not None :
382
371
timestamp = int (time .mktime (server_date ) * 1000 )
383
372
self ._last_known_past_time = timestamp
384
- if r .status > 299 :
385
- log .error (http_error_message (r .status , "event delivery" , "some events were dropped" ))
386
- if not is_http_error_recoverable (r .status ):
387
- self ._disabled = True
388
- return
373
+ if r .status > 299 and not is_http_error_recoverable (r .status ):
374
+ self ._disabled = True
375
+ return
389
376
390
377
def _send_and_reset_diagnostics (self ):
391
378
if self ._diagnostic_accumulator is not None :
@@ -472,3 +459,43 @@ def __enter__(self):
472
459
473
460
def __exit__ (self , type , value , traceback ):
474
461
self .stop ()
462
+
463
+
464
+ def _post_events_with_retry (
465
+ http_client ,
466
+ config ,
467
+ uri ,
468
+ payload_id ,
469
+ body ,
470
+ events_description
471
+ ):
472
+ hdrs = _headers (config )
473
+ hdrs ['Content-Type' ] = 'application/json'
474
+ if payload_id :
475
+ hdrs ['X-LaunchDarkly-Event-Schema' ] = str (__CURRENT_EVENT_SCHEMA__ )
476
+ hdrs ['X-LaunchDarkly-Payload-ID' ] = payload_id
477
+ can_retry = True
478
+ context = "posting %s" % events_description
479
+ while True :
480
+ next_action_message = "will retry" if can_retry else "some events were dropped"
481
+ try :
482
+ r = http_client .request (
483
+ 'POST' ,
484
+ uri ,
485
+ headers = hdrs ,
486
+ body = body ,
487
+ timeout = urllib3 .Timeout (connect = config .connect_timeout , read = config .read_timeout ),
488
+ retries = 0
489
+ )
490
+ if r .status < 300 :
491
+ return r
492
+ recoverable = check_if_error_is_recoverable_and_log (context , r .status , None , next_action_message )
493
+ if not recoverable :
494
+ return r
495
+ except Exception as e :
496
+ check_if_error_is_recoverable_and_log (context , None , str (e ), next_action_message )
497
+ if not can_retry :
498
+ return None
499
+ can_retry = False
500
+ # fixed delay of 1 second for event retries
501
+ time .sleep (1 )
0 commit comments