test_chrono.py 5.59 KB
Newer Older
1
# -*- coding: utf-8 -*-
2
import datetime
3

4
5
6
import pytest

import env  # noqa: F401
7
from pybind11_tests import chrono as m
8
9
10
11
12


def test_chrono_system_clock():

    # Get the time from both c++ and datetime
13
    date0 = datetime.datetime.today()
14
    date1 = m.test_chrono1()
15
16
17
18
19
20
    date2 = datetime.datetime.today()

    # The returned value should be a datetime
    assert isinstance(date1, datetime.datetime)

    # The numbers should vary by a very small amount (time it took to execute)
21
    diff_python = abs(date2 - date0)
22
23
    diff = abs(date1 - date2)

24
    # There should never be a days difference
25
26
    assert diff.days == 0

27
    # Since datetime.datetime.today() calls time.time(), and on some platforms
28
29
    # that has 1 second accuracy, we compare this way
    assert diff.seconds <= diff_python.seconds
30
31
32
33
34
35


def test_chrono_system_clock_roundtrip():
    date1 = datetime.datetime.today()

    # Roundtrip the time
36
    date2 = m.test_chrono2(date1)
37
38
39
40
41
42

    # The returned value should be a datetime
    assert isinstance(date2, datetime.datetime)

    # They should be identical (no information lost on roundtrip)
    diff = abs(date1 - date2)
43
    assert diff == datetime.timedelta(0)
44
45


46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
def test_chrono_system_clock_roundtrip_date():
    date1 = datetime.date.today()

    # Roundtrip the time
    datetime2 = m.test_chrono2(date1)
    date2 = datetime2.date()
    time2 = datetime2.time()

    # The returned value should be a datetime
    assert isinstance(datetime2, datetime.datetime)
    assert isinstance(date2, datetime.date)
    assert isinstance(time2, datetime.time)

    # They should be identical (no information lost on roundtrip)
    diff = abs(date1 - date2)
    assert diff.days == 0
    assert diff.seconds == 0
    assert diff.microseconds == 0

    # Year, Month & Day should be the same after the round trip
66
    assert date1 == date2
67
68
69
70
71
72
73
74

    # There should be no time information
    assert time2.hour == 0
    assert time2.minute == 0
    assert time2.second == 0
    assert time2.microsecond == 0


75
76
77
78
79
SKIP_TZ_ENV_ON_WIN = pytest.mark.skipif(
    "env.WIN", reason="TZ environment variable only supported on POSIX"
)


80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
@pytest.mark.parametrize(
    "time1",
    [
        datetime.datetime.today().time(),
        datetime.time(0, 0, 0),
        datetime.time(0, 0, 0, 1),
        datetime.time(0, 28, 45, 109827),
        datetime.time(0, 59, 59, 999999),
        datetime.time(1, 0, 0),
        datetime.time(5, 59, 59, 0),
        datetime.time(5, 59, 59, 1),
    ],
)
@pytest.mark.parametrize(
    "tz",
    [
        None,
        pytest.param("Europe/Brussels", marks=SKIP_TZ_ENV_ON_WIN),
        pytest.param("Asia/Pyongyang", marks=SKIP_TZ_ENV_ON_WIN),
        pytest.param("America/New_York", marks=SKIP_TZ_ENV_ON_WIN),
    ],
)
102
103
104
def test_chrono_system_clock_roundtrip_time(time1, tz, monkeypatch):
    if tz is not None:
        monkeypatch.setenv("TZ", "/usr/share/zoneinfo/{}".format(tz))
105
106
107
108
109
110
111
112
113
114
115
116

    # Roundtrip the time
    datetime2 = m.test_chrono2(time1)
    date2 = datetime2.date()
    time2 = datetime2.time()

    # The returned value should be a datetime
    assert isinstance(datetime2, datetime.datetime)
    assert isinstance(date2, datetime.date)
    assert isinstance(time2, datetime.time)

    # Hour, Minute, Second & Microsecond should be the same after the round trip
117
    assert time1 == time2
118
119
120
121
122
123
124

    # There should be no date information (i.e. date = python base date)
    assert date2.year == 1970
    assert date2.month == 1
    assert date2.day == 1


125
126
def test_chrono_duration_roundtrip():

127
    # Get the difference between two times (a timedelta)
128
129
130
131
132
133
134
    date1 = datetime.datetime.today()
    date2 = datetime.datetime.today()
    diff = date2 - date1

    # Make sure this is a timedelta
    assert isinstance(diff, datetime.timedelta)

135
    cpp_diff = m.test_chrono3(diff)
136

137
138
139
140
141
142
143
    assert cpp_diff == diff

    # Negative timedelta roundtrip
    diff = datetime.timedelta(microseconds=-1)
    cpp_diff = m.test_chrono3(diff)

    assert cpp_diff == diff
144
145
146
147
148
149
150
151


def test_chrono_duration_subtraction_equivalence():

    date1 = datetime.datetime.today()
    date2 = datetime.datetime.today()

    diff = date2 - date1
152
    cpp_diff = m.test_chrono4(date2, date1)
153

154
    assert cpp_diff == diff
155
156


157
158
159
160
161
162
163
164
def test_chrono_duration_subtraction_equivalence_date():

    date1 = datetime.date.today()
    date2 = datetime.date.today()

    diff = date2 - date1
    cpp_diff = m.test_chrono4(date2, date1)

165
    assert cpp_diff == diff
166
167


168
def test_chrono_steady_clock():
169
    time1 = m.test_chrono5()
170
    assert isinstance(time1, datetime.timedelta)
171
172
173


def test_chrono_steady_clock_roundtrip():
174
    time1 = datetime.timedelta(days=10, seconds=10, microseconds=100)
175
    time2 = m.test_chrono6(time1)
176

177
    assert isinstance(time2, datetime.timedelta)
178
179

    # They should be identical (no information lost on roundtrip)
180
    assert time1 == time2
181
182
183


def test_floating_point_duration():
184
185
    # Test using a floating point number in seconds
    time = m.test_chrono7(35.525123)
186
187
188
189

    assert isinstance(time, datetime.timedelta)

    assert time.seconds == 35
190
    assert 525122 <= time.microseconds <= 525123
191

192
    diff = m.test_chrono_float_diff(43.789012, 1.123456)
193
194
    assert diff.seconds == 42
    assert 665556 <= diff.microseconds <= 665557
195
196
197
198
199


def test_nano_timepoint():
    time = datetime.datetime.now()
    time1 = m.test_nano_timepoint(time, datetime.timedelta(seconds=60))
200
    assert time1 == time + datetime.timedelta(seconds=60)
201
202
203
204
205
206
207
208
209
210


def test_chrono_different_resolutions():
    resolutions = m.different_resolutions()
    time = datetime.datetime.now()
    resolutions.timestamp_h = time
    resolutions.timestamp_m = time
    resolutions.timestamp_s = time
    resolutions.timestamp_ms = time
    resolutions.timestamp_us = time