|
| 1 | +# coding=utf-8 |
| 2 | + |
| 3 | +import datetime as dt |
| 4 | +import time |
| 5 | + |
| 6 | +import pytest |
| 7 | + |
| 8 | +from scout_apm.compat import datetime_to_timestamp |
| 9 | +from scout_apm.core.queue_time import ( |
| 10 | + CUTOFF_EPOCH_S, |
| 11 | + _convert_ambiguous_timestamp_to_ns, |
| 12 | + track_job_queue_time, |
| 13 | + track_request_queue_time, |
| 14 | +) |
| 15 | + |
| 16 | + |
| 17 | +@pytest.mark.parametrize("with_t", [True, False]) |
| 18 | +def test_track_request_queue_time_valid(with_t, tracked_request): |
| 19 | + queue_start = int(datetime_to_timestamp(dt.datetime.utcnow())) - 2 |
| 20 | + if with_t: |
| 21 | + header_value = str("t=") + str(queue_start) |
| 22 | + else: |
| 23 | + header_value = str(queue_start) |
| 24 | + |
| 25 | + result = track_request_queue_time(header_value, tracked_request) |
| 26 | + |
| 27 | + assert result is True |
| 28 | + queue_time_ns = tracked_request.tags["scout.queue_time_ns"] |
| 29 | + assert isinstance(queue_time_ns, int) and queue_time_ns > 0 |
| 30 | + |
| 31 | + |
| 32 | +@pytest.mark.parametrize( |
| 33 | + "header_value", |
| 34 | + [ |
| 35 | + str(""), |
| 36 | + str("t=X"), # first character not a digit |
| 37 | + str("t=0.3f"), # raises ValueError on float() conversion |
| 38 | + str(datetime_to_timestamp(dt.datetime.utcnow()) + 3600.0), # one hour in future |
| 39 | + str(datetime_to_timestamp(dt.datetime(2009, 1, 1))), # before ambig cutoff |
| 40 | + ], |
| 41 | +) |
| 42 | +def test_track_request_queue_time_invalid(header_value, tracked_request): |
| 43 | + result = track_request_queue_time(header_value, tracked_request) |
| 44 | + |
| 45 | + assert result is False |
| 46 | + assert "scout.queue_time_ns" not in tracked_request.tags |
| 47 | + |
| 48 | + |
| 49 | +@pytest.mark.parametrize("with_t", [True, False]) |
| 50 | +def test_track_job_queue_time_valid(with_t, tracked_request): |
| 51 | + queue_start = datetime_to_timestamp(dt.datetime.utcnow()) - 2.0 |
| 52 | + result = track_job_queue_time(queue_start, tracked_request) |
| 53 | + |
| 54 | + assert result is True |
| 55 | + queue_time_ns = tracked_request.tags["scout.job_queue_time_ns"] |
| 56 | + assert isinstance(queue_time_ns, int) and queue_time_ns > 0 |
| 57 | + |
| 58 | + |
| 59 | +@pytest.mark.parametrize( |
| 60 | + "header_value", |
| 61 | + [ |
| 62 | + str(""), |
| 63 | + str("123"), |
| 64 | + str(datetime_to_timestamp(dt.datetime.utcnow()) + 3600.0), # one hour in future |
| 65 | + str(datetime_to_timestamp(dt.datetime(2009, 1, 1))), # before ambig cutoff |
| 66 | + ], |
| 67 | +) |
| 68 | +def test_track_job_queue_time_invalid(header_value, tracked_request): |
| 69 | + result = track_job_queue_time(header_value, tracked_request) |
| 70 | + |
| 71 | + assert result is False |
| 72 | + assert "scout.job_queue_time_ns" not in tracked_request.tags |
| 73 | + |
| 74 | + |
| 75 | +ref_time_s = time.mktime((2019, 6, 1, 0, 0, 0, 0, 0, 0)) |
| 76 | + |
| 77 | + |
| 78 | +@pytest.mark.parametrize( |
| 79 | + "given,expected", |
| 80 | + [ |
| 81 | + (ref_time_s, ref_time_s * 1e9), |
| 82 | + (ref_time_s * 1e3, ref_time_s * 1e9), |
| 83 | + (ref_time_s * 1e6, ref_time_s * 1e9), |
| 84 | + (CUTOFF_EPOCH_S + 10, (CUTOFF_EPOCH_S + 10) * 1e9), |
| 85 | + (0.0, 0.0), |
| 86 | + (1000.0, 0.0), |
| 87 | + (float("inf"), float("inf")), |
| 88 | + (float("-inf"), 0.0), |
| 89 | + (float("nan"), 0.0), |
| 90 | + ], |
| 91 | +) |
| 92 | +def test_convert_ambiguous_timestamp_to_ns(given, expected): |
| 93 | + assert _convert_ambiguous_timestamp_to_ns(given) == expected |
0 commit comments