Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions Ubika/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,12 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0

## Unreleased

## 2025-12-09 - 1.0.4

### Added

- Add timestepper to avoid timeouts on large data fetches

## 2025-11-28 - 1.0.3

### Fixed
Expand Down
2 changes: 1 addition & 1 deletion Ubika/manifest.json
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@
"name": "Ubika",
"uuid": "0c82ee9b-f645-47f9-8e16-a689cfc246c4",
"slug": "ubika",
"version": "1.0.3",
"version": "1.0.4",
"categories": [
"Network"
]
Expand Down
244 changes: 244 additions & 0 deletions Ubika/tests/test_timestepper.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,244 @@
import datetime
from unittest.mock import MagicMock, patch

import pytest

from ubika_modules.timestepper import TimeStepper


@pytest.fixture
def mock_trigger():
trigger = MagicMock()
trigger.log = MagicMock()
trigger.configuration.intake_key = "test_intake_key"
return trigger


@pytest.fixture
def fixed_time():
return datetime.datetime(2024, 1, 15, 12, 0, 0, tzinfo=datetime.UTC)


def test_timestepper_initialization(mock_trigger):
start = datetime.datetime(2024, 1, 15, 10, 0, 0, tzinfo=datetime.UTC)
end = datetime.datetime(2024, 1, 15, 11, 0, 0, tzinfo=datetime.UTC)
frequency = datetime.timedelta(seconds=60)
timedelta = datetime.timedelta(minutes=1)

stepper = TimeStepper(mock_trigger, start, end, frequency, timedelta)

assert stepper.trigger == mock_trigger
assert stepper.start == start
assert stepper.end == end
assert stepper.frequency == frequency
assert stepper.timedelta == timedelta


def test_ranges_yields_current_time_range_first(mock_trigger):
start = datetime.datetime(2024, 1, 15, 10, 0, 0, tzinfo=datetime.UTC)
end = datetime.datetime(2024, 1, 15, 11, 0, 0, tzinfo=datetime.UTC)
frequency = datetime.timedelta(seconds=60)
timedelta = datetime.timedelta(minutes=1)

stepper = TimeStepper(mock_trigger, start, end, frequency, timedelta)

with patch("ubika_modules.timestepper.datetime") as mock_datetime, patch("ubika_modules.timestepper.time"):
mock_datetime.datetime.now.return_value = datetime.datetime(2024, 1, 15, 12, 0, 0, tzinfo=datetime.UTC)
mock_datetime.UTC = datetime.UTC
mock_datetime.timedelta = datetime.timedelta

generator = stepper.ranges()
first_range = next(generator)

assert first_range == (start, end)


def test_ranges_updates_start_and_end_after_iteration(mock_trigger):
start = datetime.datetime(2024, 1, 15, 10, 0, 0, tzinfo=datetime.UTC)
end = datetime.datetime(2024, 1, 15, 10, 1, 0, tzinfo=datetime.UTC)
frequency = datetime.timedelta(seconds=60)
timedelta = datetime.timedelta(minutes=1)

stepper = TimeStepper(mock_trigger, start, end, frequency, timedelta)

with patch("ubika_modules.timestepper.datetime") as mock_datetime, patch("ubika_modules.timestepper.time"):
mock_datetime.datetime.now.return_value = datetime.datetime(2024, 1, 15, 12, 0, 0, tzinfo=datetime.UTC)
mock_datetime.UTC = datetime.UTC
mock_datetime.timedelta = datetime.timedelta

generator = stepper.ranges()
next(generator)
second_range = next(generator)

assert second_range[0] == end
assert second_range[1] > end


def test_ranges_sleeps_when_next_end_is_in_future(mock_trigger):
start = datetime.datetime(2024, 1, 15, 11, 58, 0, tzinfo=datetime.UTC)
end = datetime.datetime(2024, 1, 15, 11, 59, 0, tzinfo=datetime.UTC)
frequency = datetime.timedelta(seconds=60)
timedelta = datetime.timedelta(minutes=1)

stepper = TimeStepper(mock_trigger, start, end, frequency, timedelta)

with patch("ubika_modules.timestepper.datetime") as mock_datetime, patch(
"ubika_modules.timestepper.time"
) as mock_time:
now = datetime.datetime(2024, 1, 15, 11, 59, 30, tzinfo=datetime.UTC)
mock_datetime.datetime.now.return_value = now
mock_datetime.UTC = datetime.UTC
mock_datetime.timedelta = datetime.timedelta
mock_time.sleep = MagicMock()

generator = stepper.ranges()
next(generator)
next(generator)

assert mock_time.sleep.call_count == 1
sleep_duration = mock_time.sleep.call_args[0][0]
assert sleep_duration > 0


def test_ranges_does_not_sleep_when_next_end_is_in_past(mock_trigger):
start = datetime.datetime(2024, 1, 15, 10, 0, 0, tzinfo=datetime.UTC)
end = datetime.datetime(2024, 1, 15, 10, 1, 0, tzinfo=datetime.UTC)
frequency = datetime.timedelta(seconds=60)
timedelta = datetime.timedelta(minutes=1)

stepper = TimeStepper(mock_trigger, start, end, frequency, timedelta)

with patch("ubika_modules.timestepper.datetime") as mock_datetime, patch(
"ubika_modules.timestepper.time"
) as mock_time:
mock_datetime.datetime.now.return_value = datetime.datetime(2024, 1, 15, 12, 0, 0, tzinfo=datetime.UTC)
mock_datetime.UTC = datetime.UTC
mock_datetime.timedelta = datetime.timedelta
mock_time.sleep = MagicMock()

generator = stepper.ranges()
next(generator)
next(generator)

assert mock_time.sleep.call_count == 0


def test_create_with_default_parameters(mock_trigger, fixed_time):
with patch("ubika_modules.timestepper.datetime") as mock_datetime:
mock_datetime.datetime.now.return_value = fixed_time
mock_datetime.UTC = datetime.UTC
mock_datetime.timedelta = datetime.timedelta

stepper = TimeStepper.create(mock_trigger)

assert stepper.trigger == mock_trigger
assert stepper.frequency == datetime.timedelta(seconds=60)
assert stepper.timedelta == datetime.timedelta(minutes=1)
assert stepper.end < fixed_time
assert stepper.start < stepper.end


def test_create_with_start_time_zero(mock_trigger, fixed_time):
with patch("ubika_modules.timestepper.datetime") as mock_datetime:
mock_datetime.datetime.now.return_value = fixed_time
mock_datetime.UTC = datetime.UTC
mock_datetime.timedelta = datetime.timedelta

stepper = TimeStepper.create(mock_trigger, start_time=0)

expected_end = fixed_time - datetime.timedelta(minutes=1)
assert stepper.end == expected_end


def test_create_with_custom_start_time(mock_trigger, fixed_time):
with patch("ubika_modules.timestepper.datetime") as mock_datetime:
mock_datetime.datetime.now.return_value = fixed_time
mock_datetime.UTC = datetime.UTC
mock_datetime.timedelta = datetime.timedelta

stepper = TimeStepper.create(mock_trigger, start_time=2)

expected_end = fixed_time - datetime.timedelta(hours=2)
assert stepper.end == expected_end


def test_create_with_custom_frequency_and_timedelta(mock_trigger, fixed_time):
with patch("ubika_modules.timestepper.datetime") as mock_datetime:
mock_datetime.datetime.now.return_value = fixed_time
mock_datetime.UTC = datetime.UTC
mock_datetime.timedelta = datetime.timedelta

stepper = TimeStepper.create(mock_trigger, frequency=120, timedelta=5)

assert stepper.frequency == datetime.timedelta(seconds=120)
assert stepper.timedelta == datetime.timedelta(minutes=5)


def test_create_from_time_with_default_parameters(mock_trigger):
start = datetime.datetime(2024, 1, 15, 10, 0, 0, tzinfo=datetime.UTC)

stepper = TimeStepper.create_from_time(mock_trigger, start)

assert stepper.trigger == mock_trigger
assert stepper.start == start
assert stepper.end == start + datetime.timedelta(seconds=60)
assert stepper.frequency == datetime.timedelta(seconds=60)
assert stepper.timedelta == datetime.timedelta(minutes=1)


def test_create_from_time_with_custom_parameters(mock_trigger):
start = datetime.datetime(2024, 1, 15, 10, 0, 0, tzinfo=datetime.UTC)

stepper = TimeStepper.create_from_time(mock_trigger, start, frequency=300, timedelta=10)

assert stepper.start == start
assert stepper.end == start + datetime.timedelta(seconds=300)
assert stepper.frequency == datetime.timedelta(seconds=300)
assert stepper.timedelta == datetime.timedelta(minutes=10)


def test_ranges_logs_current_lag(mock_trigger):
start = datetime.datetime(2024, 1, 15, 10, 0, 0, tzinfo=datetime.UTC)
end = datetime.datetime(2024, 1, 15, 10, 1, 0, tzinfo=datetime.UTC)
frequency = datetime.timedelta(seconds=60)
timedelta = datetime.timedelta(minutes=1)

stepper = TimeStepper(mock_trigger, start, end, frequency, timedelta)

with patch("ubika_modules.timestepper.datetime") as mock_datetime, patch("ubika_modules.timestepper.time"):
mock_datetime.datetime.now.return_value = datetime.datetime(2024, 1, 15, 12, 0, 0, tzinfo=datetime.UTC)
mock_datetime.UTC = datetime.UTC
mock_datetime.timedelta = datetime.timedelta

generator = stepper.ranges()
next(generator)
next(generator)

assert mock_trigger.log.called
log_calls = [call for call in mock_trigger.log.call_args_list if "Current lag" in str(call)]
assert len(log_calls) > 0


def test_ranges_logs_waiting_message_when_sleeping(mock_trigger):
start = datetime.datetime(2024, 1, 15, 11, 58, 0, tzinfo=datetime.UTC)
end = datetime.datetime(2024, 1, 15, 11, 59, 0, tzinfo=datetime.UTC)
frequency = datetime.timedelta(seconds=60)
timedelta = datetime.timedelta(minutes=1)

stepper = TimeStepper(mock_trigger, start, end, frequency, timedelta)

with patch("ubika_modules.timestepper.datetime") as mock_datetime, patch(
"ubika_modules.timestepper.time"
) as mock_time:
now = datetime.datetime(2024, 1, 15, 11, 59, 30, tzinfo=datetime.UTC)
mock_datetime.datetime.now.return_value = now
mock_datetime.UTC = datetime.UTC
mock_datetime.timedelta = datetime.timedelta
mock_time.sleep = MagicMock()

generator = stepper.ranges()
next(generator)
next(generator)

log_calls = [call for call in mock_trigger.log.call_args_list if "Waiting" in str(call)]
assert len(log_calls) > 0
16 changes: 11 additions & 5 deletions Ubika/tests/test_ubika_cloud_protector_next_gen.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
from unittest.mock import MagicMock, patch
from datetime import datetime, timezone

import pytest
import requests_mock
Expand Down Expand Up @@ -151,7 +152,7 @@ def test_fetch_events_with_pagination(trigger, message1, message2):
)

trigger.from_timestamp = 1747326567845
events = trigger.fetch_events()
events = trigger._UbikaCloudProtectorNextGenConnector__fetch_next_events(1747326567845, 1747326667845)
Copy link

Copilot AI Dec 9, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The test URL mock on line 141 is outdated and doesn't match the actual API call being made. The __fetch_next_events method now includes filters.toDate parameter (line 155 passes both start and end timestamps), but the mocked URL doesn't include this parameter. This test will likely fail. Update the mock URL to:

"https://api.ubika.io/rest/logs.ubika.io/v1/ns/sekoia/security-events?filters.fromDate=1747326567845&filters.toDate=1747326667845&pagination.realtime=True&pagination.pageSize=100"

Copilot uses AI. Check for mistakes.

assert list(events) == [message1["spec"]["items"]]

Expand All @@ -172,8 +173,7 @@ def test_next_batch_sleep_until_next_round(trigger, message1, message2):
)

mock_requests.get(
"https://api.ubika.io/rest/logs.ubika.io/v1/ns/sekoia/security-events?filters.fromDate=1747326567845&"
"pagination.realtime=True&pagination.pageSize=100",
"https://api.ubika.io/rest/logs.ubika.io/v1/ns/sekoia/security-events?filters.fromDate=1747326560000&filters.toDate=1747326560000&pagination.realtime=True&pagination.pageSize=100",
status_code=200,
json=message1,
)
Expand All @@ -192,7 +192,10 @@ def test_next_batch_sleep_until_next_round(trigger, message1, message2):
end_time = start_time + batch_duration
mock_time.time.side_effect = [start_time, end_time, end_time]

trigger.next_batch()
start = datetime.fromtimestamp(1747326560, tz=timezone.utc)
end = datetime.fromtimestamp(1747326560, tz=timezone.utc)

trigger.next_batch(start, end)

assert trigger.push_events_to_intakes.call_count == 1
assert mock_time.sleep.call_count == 0
Expand All @@ -206,5 +209,8 @@ def test_authorization_error(trigger):
status_code=400,
)

start = datetime.fromtimestamp(1747326560, tz=timezone.utc)
end = datetime.fromtimestamp(1747326560, tz=timezone.utc)

with pytest.raises(AuthorizationError):
trigger.next_batch()
trigger.next_batch(start=start, end=end)
Loading