0bc4de65e3
The db truncates the microseconds so this can end up being negative. Closes-Bug: #1493166 Change-Id: I5169821e7465e483b8379ee7f614ca08877973e7
175 lines
5.1 KiB
Python
175 lines
5.1 KiB
Python
#
|
|
# Licensed 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 datetime
|
|
from testtools import matchers
|
|
|
|
from heat.common import timeutils as util
|
|
from heat.tests import common
|
|
|
|
|
|
class ISO8601UtilityTest(common.HeatTestCase):
|
|
|
|
def setUp(self):
|
|
super(ISO8601UtilityTest, self).setUp()
|
|
|
|
def test_valid_durations(self):
|
|
self.assertEqual(0, util.parse_isoduration('PT'))
|
|
self.assertEqual(3600, util.parse_isoduration('PT1H'))
|
|
self.assertEqual(120, util.parse_isoduration('PT2M'))
|
|
self.assertEqual(3, util.parse_isoduration('PT3S'))
|
|
self.assertEqual(3900, util.parse_isoduration('PT1H5M'))
|
|
self.assertEqual(3605, util.parse_isoduration('PT1H5S'))
|
|
self.assertEqual(303, util.parse_isoduration('PT5M3S'))
|
|
self.assertEqual(3903, util.parse_isoduration('PT1H5M3S'))
|
|
self.assertEqual(24 * 3600, util.parse_isoduration('PT24H'))
|
|
|
|
def test_invalid_durations(self):
|
|
self.assertRaises(ValueError, util.parse_isoduration, 'P1Y')
|
|
self.assertRaises(ValueError, util.parse_isoduration, 'P1DT12H')
|
|
self.assertRaises(ValueError, util.parse_isoduration, 'PT1Y1D')
|
|
self.assertRaises(ValueError, util.parse_isoduration, 'PTAH1M0S')
|
|
self.assertRaises(ValueError, util.parse_isoduration, 'PT1HBM0S')
|
|
self.assertRaises(ValueError, util.parse_isoduration, 'PT1H1MCS')
|
|
self.assertRaises(ValueError, util.parse_isoduration, 'PT1H1H')
|
|
self.assertRaises(ValueError, util.parse_isoduration, 'PT1MM')
|
|
self.assertRaises(ValueError, util.parse_isoduration, 'PT1S0S')
|
|
self.assertRaises(ValueError, util.parse_isoduration, 'ABCDEFGH')
|
|
|
|
|
|
class DurationTest(common.HeatTestCase):
|
|
|
|
def setUp(self):
|
|
super(DurationTest, self).setUp()
|
|
st = util.wallclock()
|
|
mock_clock = self.patchobject(util, 'wallclock')
|
|
mock_clock.side_effect = [st, st + 0.5]
|
|
|
|
def test_duration_not_expired(self):
|
|
self.assertFalse(util.Duration(1.0).expired())
|
|
|
|
def test_duration_expired(self):
|
|
self.assertTrue(util.Duration(0.1).expired())
|
|
|
|
|
|
class RetryBackoffExponentialTest(common.HeatTestCase):
|
|
|
|
scenarios = [(
|
|
'0_0',
|
|
dict(
|
|
attempt=0,
|
|
scale_factor=0.0,
|
|
delay=0.0,
|
|
)
|
|
), (
|
|
'0_1',
|
|
dict(
|
|
attempt=0,
|
|
scale_factor=1.0,
|
|
delay=1.0,
|
|
)
|
|
), (
|
|
'1_1',
|
|
dict(
|
|
attempt=1,
|
|
scale_factor=1.0,
|
|
delay=2.0,
|
|
)
|
|
), (
|
|
'2_1',
|
|
dict(
|
|
attempt=2,
|
|
scale_factor=1.0,
|
|
delay=4.0,
|
|
)
|
|
), (
|
|
'3_1',
|
|
dict(
|
|
attempt=3,
|
|
scale_factor=1.0,
|
|
delay=8.0,
|
|
)
|
|
), (
|
|
'4_1',
|
|
dict(
|
|
attempt=4,
|
|
scale_factor=1.0,
|
|
delay=16.0,
|
|
)
|
|
), (
|
|
'4_4',
|
|
dict(
|
|
attempt=4,
|
|
scale_factor=4.0,
|
|
delay=64.0,
|
|
)
|
|
)]
|
|
|
|
def test_backoff_delay(self):
|
|
delay = util.retry_backoff_delay(
|
|
self.attempt, self.scale_factor)
|
|
self.assertEqual(self.delay, delay)
|
|
|
|
|
|
class RetryBackoffJitterTest(common.HeatTestCase):
|
|
|
|
scenarios = [(
|
|
'0_0_1',
|
|
dict(
|
|
attempt=0,
|
|
scale_factor=0.0,
|
|
jitter_max=1.0,
|
|
delay_from=0.0,
|
|
delay_to=1.0
|
|
)
|
|
), (
|
|
'1_1_1',
|
|
dict(
|
|
attempt=1,
|
|
scale_factor=1.0,
|
|
jitter_max=1.0,
|
|
delay_from=2.0,
|
|
delay_to=3.0
|
|
)
|
|
), (
|
|
'1_1_5',
|
|
dict(
|
|
attempt=1,
|
|
scale_factor=1.0,
|
|
jitter_max=5.0,
|
|
delay_from=2.0,
|
|
delay_to=7.0
|
|
)
|
|
)]
|
|
|
|
def test_backoff_delay(self):
|
|
for _ in range(100):
|
|
delay = util.retry_backoff_delay(
|
|
self.attempt, self.scale_factor, self.jitter_max)
|
|
self.assertThat(delay, matchers.GreaterThan(self.delay_from))
|
|
self.assertThat(delay, matchers.LessThan(self.delay_to))
|
|
|
|
|
|
class RoundToSecondsTest(common.HeatTestCase):
|
|
|
|
scenarios = [('down', dict(in_secs=5, in_ms=12345, out_secs=5)),
|
|
('up', dict(in_secs=5, in_ms=501000, out_secs=6)),
|
|
('same', dict(in_secs=5, in_ms=0, out_secs=5))]
|
|
|
|
def test_rounding(self):
|
|
inp = datetime.datetime(2015, 7, 27, 4,
|
|
34, self.in_secs, self.in_ms)
|
|
exp = datetime.datetime(2015, 7, 27, 4,
|
|
34, self.out_secs, 0)
|
|
self.assertEqual(exp, util.round_to_seconds(inp))
|