Improve ansible-test HttpClient error handling. (#30301)
- Automatic retries on DNS lookup failures.
- Handle API errors in shippable.py.
(cherry picked from commit 1a602096f5
)
This commit is contained in:
parent
04cc8a9468
commit
57683caa44
2 changed files with 29 additions and 1 deletions
|
@ -6,6 +6,7 @@ Avoids use of urllib2 due to lack of SNI support.
|
||||||
from __future__ import absolute_import, print_function
|
from __future__ import absolute_import, print_function
|
||||||
|
|
||||||
import json
|
import json
|
||||||
|
import time
|
||||||
|
|
||||||
try:
|
try:
|
||||||
from urllib import urlencode
|
from urllib import urlencode
|
||||||
|
@ -24,6 +25,8 @@ from lib.util import (
|
||||||
CommonConfig,
|
CommonConfig,
|
||||||
ApplicationError,
|
ApplicationError,
|
||||||
run_command,
|
run_command,
|
||||||
|
SubprocessError,
|
||||||
|
display,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@ -83,7 +86,28 @@ class HttpClient(object):
|
||||||
|
|
||||||
cmd += [url]
|
cmd += [url]
|
||||||
|
|
||||||
|
attempts = 0
|
||||||
|
max_attempts = 3
|
||||||
|
sleep_seconds = 3
|
||||||
|
|
||||||
|
# curl error codes which are safe to retry (request never sent to server)
|
||||||
|
retry_on_status = (
|
||||||
|
6, # CURLE_COULDNT_RESOLVE_HOST
|
||||||
|
)
|
||||||
|
|
||||||
|
while True:
|
||||||
|
attempts += 1
|
||||||
|
|
||||||
|
try:
|
||||||
stdout, _ = run_command(self.args, cmd, capture=True, always=self.always, cmd_verbosity=2)
|
stdout, _ = run_command(self.args, cmd, capture=True, always=self.always, cmd_verbosity=2)
|
||||||
|
break
|
||||||
|
except SubprocessError as ex:
|
||||||
|
if ex.status in retry_on_status and attempts < max_attempts:
|
||||||
|
display.warning(u'%s' % ex)
|
||||||
|
time.sleep(sleep_seconds)
|
||||||
|
continue
|
||||||
|
|
||||||
|
raise
|
||||||
|
|
||||||
if self.args.explain and not self.always:
|
if self.args.explain and not self.always:
|
||||||
return HttpResponse(method, url, 200, '')
|
return HttpResponse(method, url, 200, '')
|
||||||
|
|
|
@ -7,6 +7,7 @@ from __future__ import absolute_import, print_function
|
||||||
# noinspection PyCompatibility
|
# noinspection PyCompatibility
|
||||||
import argparse
|
import argparse
|
||||||
import errno
|
import errno
|
||||||
|
import json
|
||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
|
|
||||||
|
@ -44,6 +45,9 @@ def main():
|
||||||
response = client.get('https://api.shippable.com/jobs?runIds=%s' % run_id)
|
response = client.get('https://api.shippable.com/jobs?runIds=%s' % run_id)
|
||||||
jobs = response.json()
|
jobs = response.json()
|
||||||
|
|
||||||
|
if not isinstance(jobs, list):
|
||||||
|
raise ApplicationError(json.dumps(jobs, indent=4, sort_keys=True))
|
||||||
|
|
||||||
if len(jobs) == 1:
|
if len(jobs) == 1:
|
||||||
raise ApplicationError('Shippable run %s has only one job. Did you use the "Rebuild with SSH" option?' % run_id)
|
raise ApplicationError('Shippable run %s has only one job. Did you use the "Rebuild with SSH" option?' % run_id)
|
||||||
except ApplicationWarning as ex:
|
except ApplicationWarning as ex:
|
||||||
|
|
Loading…
Reference in a new issue