blob: c522bf61be2da4b2ef47e5256cba0353ac3ea361 [file] [log] [blame]
# Licensed to the Apache Software Foundation (ASF) under one or more
# contributor license agreements. See the NOTICE file distributed with
# this work for additional information regarding copyright ownership.
# The ASF licenses this file to You under the Apache License, Version 2.0
# (the "License"); you may not use this file except in compliance with
# the License. You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
import unittest
import sys
import mock
from libcloud.common.base import LazyObject, Response
from libcloud.common.exceptions import BaseHTTPError, RateLimitReachedError
from libcloud.test import LibcloudTestCase
class LazyObjectTest(LibcloudTestCase):
class A(LazyObject):
def __init__(self, x, y=None):
self.x = x
self.y = y
def test_lazy_init(self):
# Test normal init
a = self.A(1, y=2)
self.assertTrue(isinstance(a, self.A))
# Test lazy init
with mock.patch.object(self.A, "__init__", return_value=None) as mock_init:
a = self.A.lazy(3, y=4)
self.assertTrue(isinstance(a, self.A)) # Proxy is a subclass of A
mock_init.assert_not_called()
# Since we have a mock init, an A object doesn't actually get
# created. But, we can still call __dict__ on the proxy, which will
# init the lazy object.
self.assertEqual(a.__dict__, {})
mock_init.assert_called_once_with(3, y=4)
def test_setattr(self):
a = self.A.lazy("foo", y="bar")
a.z = "baz"
wrapped_lazy_obj = object.__getattribute__(a, "_lazy_obj")
self.assertEqual(a.z, "baz")
self.assertEqual(wrapped_lazy_obj.z, "baz")
class ErrorResponseTest(LibcloudTestCase):
def mock_response(self, code, headers={}):
m = mock.MagicMock()
m.request = mock.Mock()
m.headers = headers
m.status_code = code
m.text = None
return m
def test_rate_limit_response(self):
resp_mock = self.mock_response(429, {"Retry-After": "120"})
try:
Response(resp_mock, mock.MagicMock())
except RateLimitReachedError as e:
self.assertEqual(e.retry_after, 120)
except Exception:
# We should have got a RateLimitReachedError
self.fail("Catched exception should have been RateLimitReachedError")
else:
# We should have got an exception
self.fail("HTTP Status 429 response didn't raised an exception")
def test_error_with_retry_after(self):
# 503 Service Unavailable may include Retry-After header
resp_mock = self.mock_response(503, {"Retry-After": "300"})
try:
Response(resp_mock, mock.MagicMock())
except BaseHTTPError as e:
self.assertIn("retry-after", e.headers)
self.assertEqual(e.headers["retry-after"], "300")
else:
# We should have got an exception
self.fail("HTTP Status 503 response didn't raised an exception")
@mock.patch("time.time", return_value=1231006505)
def test_error_with_retry_after_http_date_format(self, time_mock):
retry_after = "Sat, 03 Jan 2009 18:20:05 -0000"
# 503 Service Unavailable may include Retry-After header
resp_mock = self.mock_response(503, {"Retry-After": retry_after})
try:
Response(resp_mock, mock.MagicMock())
except BaseHTTPError as e:
self.assertIn("retry-after", e.headers)
# HTTP-date got translated to delay-secs
self.assertEqual(e.headers["retry-after"], "300")
else:
# We should have got an exception
self.fail("HTTP Status 503 response didn't raised an exception")
if __name__ == "__main__":
sys.exit(unittest.main())