1
1
#
2
2
# Copyright (c) 2023 Airbyte, Inc., all rights reserved.
3
3
#
4
-
4
+ from copy import deepcopy
5
5
from datetime import datetime , timedelta , timezone
6
6
from functools import partial
7
7
from typing import Any , Mapping , Optional
@@ -84,7 +84,7 @@ def _cursor_with_slice_boundary_fields(
84
84
return ConcurrentCursor (
85
85
_A_STREAM_NAME ,
86
86
_A_STREAM_NAMESPACE ,
87
- {} ,
87
+ deepcopy ( _NO_STATE ) ,
88
88
self ._message_repository ,
89
89
self ._state_manager ,
90
90
EpochValueConcurrentStreamStateConverter (is_sequential_state ),
@@ -99,7 +99,7 @@ def _cursor_without_slice_boundary_fields(self) -> ConcurrentCursor:
99
99
return ConcurrentCursor (
100
100
_A_STREAM_NAME ,
101
101
_A_STREAM_NAMESPACE ,
102
- {} ,
102
+ deepcopy ( _NO_STATE ) ,
103
103
self ._message_repository ,
104
104
self ._state_manager ,
105
105
EpochValueConcurrentStreamStateConverter (is_sequential_state = True ),
@@ -265,7 +265,7 @@ def test_given_no_state_when_generate_slices_then_create_slice_from_start_to_end
265
265
cursor = ConcurrentCursor (
266
266
_A_STREAM_NAME ,
267
267
_A_STREAM_NAMESPACE ,
268
- _NO_STATE ,
268
+ deepcopy ( _NO_STATE ) ,
269
269
self ._message_repository ,
270
270
self ._state_manager ,
271
271
EpochValueConcurrentStreamStateConverter (is_sequential_state = False ),
@@ -950,6 +950,60 @@ def test_given_initial_state_is_sequential_and_start_provided_when_generate_slic
950
950
}, # State message is updated to the legacy format before being emitted
951
951
)
952
952
953
+ @freezegun .freeze_time (time_to_freeze = datetime .fromtimestamp (50 , timezone .utc ))
954
+ def test_given_most_recent_cursor_value_in_input_state_when_emit_state_then_serialize_state_properly (
955
+ self ,
956
+ ) -> None :
957
+ cursor = ConcurrentCursor (
958
+ _A_STREAM_NAME ,
959
+ _A_STREAM_NAMESPACE ,
960
+ {
961
+ "state_type" : ConcurrencyCompatibleStateType .date_range .value ,
962
+ "slices" : [
963
+ {
964
+ EpochValueConcurrentStreamStateConverter .START_KEY : 0 ,
965
+ EpochValueConcurrentStreamStateConverter .END_KEY : 20 ,
966
+ EpochValueConcurrentStreamStateConverter .MOST_RECENT_RECORD_KEY : 15 ,
967
+ },
968
+ ],
969
+ },
970
+ self ._message_repository ,
971
+ self ._state_manager ,
972
+ EpochValueConcurrentStreamStateConverter (is_sequential_state = False ),
973
+ CursorField (_A_CURSOR_FIELD_KEY ),
974
+ _SLICE_BOUNDARY_FIELDS ,
975
+ datetime .fromtimestamp (0 , timezone .utc ),
976
+ EpochValueConcurrentStreamStateConverter .get_end_provider (),
977
+ _NO_LOOKBACK_WINDOW ,
978
+ )
979
+
980
+ cursor .close_partition (
981
+ _partition (
982
+ StreamSlice (
983
+ partition = {},
984
+ cursor_slice = {
985
+ _LOWER_SLICE_BOUNDARY_FIELD : 20 ,
986
+ _UPPER_SLICE_BOUNDARY_FIELD : 50 ,
987
+ },
988
+ ),
989
+ _stream_name = _A_STREAM_NAME ,
990
+ )
991
+ )
992
+
993
+ expected_state = {
994
+ "state_type" : ConcurrencyCompatibleStateType .date_range .value ,
995
+ "slices" : [
996
+ {
997
+ EpochValueConcurrentStreamStateConverter .START_KEY : 0 ,
998
+ EpochValueConcurrentStreamStateConverter .END_KEY : 50 ,
999
+ EpochValueConcurrentStreamStateConverter .MOST_RECENT_RECORD_KEY : 15 ,
1000
+ },
1001
+ ],
1002
+ }
1003
+ self ._state_manager .update_state_for_stream .assert_called_once_with (
1004
+ _A_STREAM_NAME , _A_STREAM_NAMESPACE , expected_state
1005
+ )
1006
+
953
1007
954
1008
class ClampingIntegrationTest (TestCase ):
955
1009
def setUp (self ) -> None :
0 commit comments