2016-09-13 22:46:59 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# (c) 2015, Allen Sanabria <asanabria@linuxdynasty.org>
|
|
|
|
#
|
|
|
|
# This file is part of Ansible
|
2019-10-30 17:48:59 +01:00
|
|
|
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
|
|
|
|
|
|
|
|
from __future__ import (absolute_import, division, print_function)
|
|
|
|
__metaclass__ = type
|
2016-09-13 22:46:59 +02:00
|
|
|
|
2016-10-13 22:19:40 +02:00
|
|
|
try:
|
|
|
|
import boto3
|
|
|
|
import botocore
|
|
|
|
HAS_BOTO3 = True
|
2018-09-08 02:59:46 +02:00
|
|
|
except Exception:
|
2016-10-13 22:19:40 +02:00
|
|
|
HAS_BOTO3 = False
|
2016-09-13 22:46:59 +02:00
|
|
|
|
2018-11-20 22:58:46 +01:00
|
|
|
import pytest
|
2016-10-13 22:19:40 +02:00
|
|
|
|
2018-10-13 05:01:14 +02:00
|
|
|
from units.compat import unittest
|
2016-09-13 22:46:59 +02:00
|
|
|
from ansible.module_utils.ec2 import AWSRetry
|
|
|
|
|
2016-10-13 22:19:40 +02:00
|
|
|
if not HAS_BOTO3:
|
2018-11-20 22:58:46 +01:00
|
|
|
pytestmark = pytest.mark.skip("test_aws.py requires the python modules 'boto3' and 'botocore'")
|
2016-10-13 22:19:40 +02:00
|
|
|
|
|
|
|
|
2016-09-13 22:46:59 +02:00
|
|
|
class RetryTestCase(unittest.TestCase):
|
|
|
|
|
|
|
|
def test_no_failures(self):
|
|
|
|
self.counter = 0
|
|
|
|
|
|
|
|
@AWSRetry.backoff(tries=2, delay=0.1)
|
|
|
|
def no_failures():
|
|
|
|
self.counter += 1
|
|
|
|
|
|
|
|
r = no_failures()
|
|
|
|
self.assertEqual(self.counter, 1)
|
|
|
|
|
2017-08-22 21:31:20 +02:00
|
|
|
def test_extend_boto3_failures(self):
|
|
|
|
self.counter = 0
|
|
|
|
err_msg = {'Error': {'Code': 'MalformedPolicyDocument'}}
|
|
|
|
|
|
|
|
@AWSRetry.backoff(tries=2, delay=0.1, catch_extra_error_codes=['MalformedPolicyDocument'])
|
|
|
|
def extend_failures():
|
|
|
|
self.counter += 1
|
|
|
|
if self.counter < 2:
|
2020-02-13 22:59:00 +01:00
|
|
|
raise botocore.exceptions.ClientError(err_msg, 'You did something wrong.')
|
2017-08-22 21:31:20 +02:00
|
|
|
else:
|
|
|
|
return 'success'
|
|
|
|
|
|
|
|
r = extend_failures()
|
|
|
|
self.assertEqual(r, 'success')
|
|
|
|
self.assertEqual(self.counter, 2)
|
|
|
|
|
2016-09-13 22:46:59 +02:00
|
|
|
def test_retry_once(self):
|
|
|
|
self.counter = 0
|
2020-02-13 22:59:00 +01:00
|
|
|
err_msg = {'Error': {'Code': 'InternalFailure'}}
|
2016-09-13 22:46:59 +02:00
|
|
|
|
|
|
|
@AWSRetry.backoff(tries=2, delay=0.1)
|
|
|
|
def retry_once():
|
|
|
|
self.counter += 1
|
|
|
|
if self.counter < 2:
|
2020-02-13 22:59:00 +01:00
|
|
|
raise botocore.exceptions.ClientError(err_msg, 'Something went wrong!')
|
2016-09-13 22:46:59 +02:00
|
|
|
else:
|
|
|
|
return 'success'
|
|
|
|
|
|
|
|
r = retry_once()
|
|
|
|
self.assertEqual(r, 'success')
|
|
|
|
self.assertEqual(self.counter, 2)
|
|
|
|
|
|
|
|
def test_reached_limit(self):
|
|
|
|
self.counter = 0
|
|
|
|
err_msg = {'Error': {'Code': 'RequestLimitExceeded'}}
|
|
|
|
|
|
|
|
@AWSRetry.backoff(tries=4, delay=0.1)
|
|
|
|
def fail():
|
|
|
|
self.counter += 1
|
|
|
|
raise botocore.exceptions.ClientError(err_msg, 'toooo fast!!')
|
|
|
|
|
2017-05-30 19:05:19 +02:00
|
|
|
# with self.assertRaises(botocore.exceptions.ClientError):
|
2016-09-13 22:46:59 +02:00
|
|
|
try:
|
|
|
|
fail()
|
|
|
|
except Exception as e:
|
|
|
|
self.assertEqual(e.response['Error']['Code'], 'RequestLimitExceeded')
|
|
|
|
self.assertEqual(self.counter, 4)
|
|
|
|
|
|
|
|
def test_unexpected_exception_does_not_retry(self):
|
|
|
|
self.counter = 0
|
|
|
|
err_msg = {'Error': {'Code': 'AuthFailure'}}
|
|
|
|
|
|
|
|
@AWSRetry.backoff(tries=4, delay=0.1)
|
|
|
|
def raise_unexpected_error():
|
|
|
|
self.counter += 1
|
|
|
|
raise botocore.exceptions.ClientError(err_msg, 'unexpected error')
|
|
|
|
|
2017-05-30 19:05:19 +02:00
|
|
|
# with self.assertRaises(botocore.exceptions.ClientError):
|
2016-09-13 22:46:59 +02:00
|
|
|
try:
|
|
|
|
raise_unexpected_error()
|
|
|
|
except Exception as e:
|
|
|
|
self.assertEqual(e.response['Error']['Code'], 'AuthFailure')
|
|
|
|
|
|
|
|
self.assertEqual(self.counter, 1)
|