From eeb15bf1ce7a04c274ea7428c5eabc24ccf84ff9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Nina=20=C3=98deg=C3=A5rd?= Date: Sun, 1 Dec 2024 23:11:31 +0100 Subject: [PATCH] feat: add future time shift strings (#2048) --- CHANGELOG.md | 6 +++++ cognite/client/_api/datapoint_tasks.py | 4 +-- cognite/client/_api/datapoints.py | 15 ++++++++--- cognite/client/_api/iam.py | 23 ++++++++++++---- cognite/client/_api/synthetic_time_series.py | 2 +- cognite/client/_api_client.py | 3 ++- cognite/client/_version.py | 2 +- cognite/client/utils/_time.py | 27 ++++++++++++++----- pyproject.toml | 2 +- tests/tests_integration/conftest.py | 12 +++++++++ .../test_transformations/test_schema.py | 2 +- .../test_api/test_workflows.py | 24 +++++++++++++++++ tests/tests_unit/test_utils/test_time.py | 26 ++++++++++++------ 13 files changed, 118 insertions(+), 30 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 686438ff6..d6e6e6f6b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -17,6 +17,12 @@ Changes are grouped as follows - `Fixed` for any bug fixes. - `Security` in case of vulnerabilities. +## [7.69.3] - 2024-12-02 +### Added +- API endpoints currently accepting relative time strings like `2d-ago` now support a forward-looking syntax, e.g. `2w-ahead` or `15m-ahead`. +### Fixed +- Revoking sessions through `client.iam.sessions.revoke` no longer raises an API error for very large payloads + ## [7.69.2] - 2024-11-28 ### Improved - Handle conversion of instance lists like NodeList to pandas DataFrame in scenarios where: a) properties are expanded diff --git a/cognite/client/_api/datapoint_tasks.py b/cognite/client/_api/datapoint_tasks.py index 6dcea2c87..f77227d93 100644 --- a/cognite/client/_api/datapoint_tasks.py +++ b/cognite/client/_api/datapoint_tasks.py @@ -48,7 +48,7 @@ parse_str_timezone, split_granularity_into_quantity_and_normalized_unit, split_time_range, - time_ago_to_ms, + time_shift_to_ms, timestamp_to_ms, ) from cognite.client.utils.useful_types import SequenceNotStr @@ -329,7 +329,7 @@ def _ts_to_ms_frozen_now(ts: int | str | datetime.datetime | None, frozen_time_n if ts is None: return default elif isinstance(ts, str): - return frozen_time_now - time_ago_to_ms(ts) + return frozen_time_now - time_shift_to_ms(ts) else: return timestamp_to_ms(ts) diff --git a/cognite/client/_api/datapoints.py b/cognite/client/_api/datapoints.py index 01d40fbaa..785910783 100644 --- a/cognite/client/_api/datapoints.py +++ b/cognite/client/_api/datapoints.py @@ -771,7 +771,8 @@ def retrieve( Examples: You can specify the identifiers of the datapoints you wish to retrieve in a number of ways. In this example - we are using the time-ago format to get raw data for the time series with id=42 from 2 weeks ago up until now. + we are using the time-ago format, ``"2w-ago"`` to get raw data for the time series with id=42 from 2 weeks ago up until now. + You can also use the time-ahead format, like ``"3d-ahead"``, to specify a relative time in the future. >>> from cognite.client import CogniteClient >>> client = CogniteClient() @@ -821,7 +822,7 @@ def retrieve( >>> dps_lst = client.time_series.data.retrieve( ... id=[ ... DatapointsQuery(id=42, end="1d-ago", aggregates="average"), - ... DatapointsQuery(id=69, end="2d-ago", aggregates=["average"]), + ... DatapointsQuery(id=69, end="2d-ahead", aggregates=["average"]), ... DatapointsQuery(id=96, end="3d-ago", aggregates=["min", "max", "count"]), ... ], ... external_id=DatapointsQuery(external_id="foo", aggregates="max"), @@ -1417,6 +1418,10 @@ def retrieve_latest( >>> res = client.time_series.data.retrieve_latest(id=1, before="2d-ago")[0] + You can also get the first datapoint before a specific time in the future e.g. forecast data: + + >>> res = client.time_series.data.retrieve_latest(id=1, before="2d-ahead")[0] + You can also retrieve the datapoint in a different unit or unit system: >>> res = client.time_series.data.retrieve_latest(id=1, target_unit="temperature:deg_f")[0] @@ -1661,6 +1666,10 @@ def delete_range( >>> from cognite.client import CogniteClient >>> client = CogniteClient() >>> client.time_series.data.delete_range(start="1w-ago", end="now", id=1) + + Deleting the data from now until 2 days in the future from a time series containing e.g. forecasted data: + + >>> client.time_series.data.delete_range(start="now", end="2d-ahead", id=1) """ start_ms = timestamp_to_ms(start) end_ms = timestamp_to_ms(end) @@ -1684,7 +1693,7 @@ def delete_ranges(self, ranges: list[dict[str, Any]]) -> None: >>> from cognite.client import CogniteClient >>> client = CogniteClient() >>> ranges = [{"id": 1, "start": "2d-ago", "end": "now"}, - ... {"external_id": "abc", "start": "2d-ago", "end": "now"}] + ... {"external_id": "abc", "start": "2d-ago", "end": "2d-ahead"}] >>> client.time_series.data.delete_ranges(ranges) """ valid_ranges = [] diff --git a/cognite/client/_api/iam.py b/cognite/client/_api/iam.py index 101481118..9fbb280be 100644 --- a/cognite/client/_api/iam.py +++ b/cognite/client/_api/iam.py @@ -4,7 +4,7 @@ from collections.abc import Iterable, Sequence from itertools import groupby from operator import itemgetter -from typing import TYPE_CHECKING, Any, Literal, TypeAlias, overload +from typing import TYPE_CHECKING, Any, Literal, TypeAlias, cast, overload from cognite.client._api.user_profiles import UserProfilesAPI from cognite.client._api_client import APIClient @@ -532,6 +532,9 @@ class SessionsAPI(APIClient): def __init__(self, config: ClientConfig, api_version: str | None, cognite_client: CogniteClient) -> None: super().__init__(config, api_version, cognite_client) self._LIST_LIMIT = 100 + self._DELETE_LIMIT = ( + 100 # There isn't an API limit so this is a self-inflicted limit due to no support for large payloads + ) def create( self, @@ -597,11 +600,21 @@ def revoke(self, id: int | Sequence[int]) -> Session | SessionList: Returns: Session | SessionList: List of revoked sessions. If the user does not have the sessionsAcl:LIST capability, then only the session IDs will be present in the response. """ - identifiers = IdentifierSequence.load(ids=id, external_ids=None) - items = {"items": identifiers.as_dicts()} - result = SessionList._load(self._post(self._RESOURCE_PATH + "/revoke", items).json()["items"]) - return result[0] if isinstance(id, int) else result + ident_sequence = IdentifierSequence.load(ids=id, external_ids=None) + + revoked_sessions_res = cast( + list, + self._delete_multiple( + identifiers=ident_sequence, + wrap_ids=True, + returns_items=True, + delete_endpoint="/revoke", + ), + ) + + revoked_sessions = SessionList._load(revoked_sessions_res) + return revoked_sessions[0] if ident_sequence.is_singleton() else revoked_sessions @overload def retrieve(self, id: int) -> Session: ... diff --git a/cognite/client/_api/synthetic_time_series.py b/cognite/client/_api/synthetic_time_series.py index de9116152..23b18acfe 100644 --- a/cognite/client/_api/synthetic_time_series.py +++ b/cognite/client/_api/synthetic_time_series.py @@ -103,7 +103,7 @@ def query( ... "C": NodeId("my-space", "my-ts-xid"), ... } >>> dps = client.time_series.data.synthetic.query( - ... expressions="A+B+C", start="2w-ago", end="now", variables=variables) + ... expressions="A+B+C", start="2w-ago", end="2w-ahead", variables=variables) Use sympy to build complex expressions: diff --git a/cognite/client/_api_client.py b/cognite/client/_api_client.py index df93f600f..135c2bf7f 100644 --- a/cognite/client/_api_client.py +++ b/cognite/client/_api_client.py @@ -967,8 +967,9 @@ def _delete_multiple( extra_body_fields: dict[str, Any] | None = None, returns_items: bool = False, executor: TaskExecutor | None = None, + delete_endpoint: str = "/delete", ) -> list | None: - resource_path = (resource_path or self._RESOURCE_PATH) + "/delete" + resource_path = (resource_path or self._RESOURCE_PATH) + delete_endpoint tasks = [ { "url_path": resource_path, diff --git a/cognite/client/_version.py b/cognite/client/_version.py index d394953f3..5e35c2bdf 100644 --- a/cognite/client/_version.py +++ b/cognite/client/_version.py @@ -1,4 +1,4 @@ from __future__ import annotations -__version__ = "7.69.2" +__version__ = "7.69.3" __api_subversion__ = "20230101" diff --git a/cognite/client/utils/_time.py b/cognite/client/utils/_time.py index fd3a6bcec..b56225e85 100644 --- a/cognite/client/utils/_time.py +++ b/cognite/client/utils/_time.py @@ -228,32 +228,45 @@ def granularity_to_ms(granularity: str, as_unit: bool = False) -> int: return ms -def time_ago_to_ms(time_ago_string: str) -> int: - """Returns millisecond representation of time-ago string""" +def time_shift_to_ms(time_ago_string: str) -> int: + """Returns millisecond representation of time-shift string""" if time_ago_string == "now": return 0 - ms = time_string_to_ms(r"(\d+)({})-ago", time_ago_string, UNIT_IN_MS) + ms = time_string_to_ms(r"(\d+)({})-(?:ago|ahead)", time_ago_string, UNIT_IN_MS) if ms is None: raise ValueError( - f"Invalid time-ago format: `{time_ago_string}`. Must be on format (s|m|h|d|w)-ago or 'now'. " - "E.g. '3d-ago' or '1w-ago'." + f"Invalid time-shift format: `{time_ago_string}`. Must be on format (s|m|h|d|w)-(ago|ahead) or 'now'. " + "E.g. '3d-ago' or '1w-ahead'." ) + if "ahead" in time_ago_string: + return -ms return ms def timestamp_to_ms(timestamp: int | float | str | datetime) -> int: - """Returns the ms representation of some timestamp given by milliseconds, time-ago format or datetime object + """Returns the ms representation of some timestamp given by milliseconds, time-shift format or datetime object Args: timestamp (int | float | str | datetime): Convert this timestamp to ms. Returns: int: Milliseconds since epoch representation of timestamp + + Examples: + + Gets the millisecond representation of a timestamp: + + >>> from cognite.client.utils import timestamp_to_ms + >>> from datetime import datetime + >>> timestamp_to_ms(datetime(2021, 1, 7, 12, 0, 0)) + >>> timestamp_to_ms("now") + >>> timestamp_to_ms("2w-ago") # 2 weeks ago + >>> timestamp_to_ms("3d-ahead") # 3 days ahead from now """ if isinstance(timestamp, numbers.Number): # float, int, int64 etc ms = int(timestamp) # type: ignore[arg-type] elif isinstance(timestamp, str): - ms = int(round(time.time() * 1000)) - time_ago_to_ms(timestamp) + ms = int(round(time.time() * 1000)) - time_shift_to_ms(timestamp) elif isinstance(timestamp, datetime): ms = datetime_to_ms(timestamp) else: diff --git a/pyproject.toml b/pyproject.toml index bff80801b..85caa9882 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,7 +1,7 @@ [tool.poetry] name = "cognite-sdk" -version = "7.69.2" +version = "7.69.3" description = "Cognite Python SDK" readme = "README.md" documentation = "https://cognite-sdk-python.readthedocs-hosted.com" diff --git a/tests/tests_integration/conftest.py b/tests/tests_integration/conftest.py index 81165363b..31299427c 100644 --- a/tests/tests_integration/conftest.py +++ b/tests/tests_integration/conftest.py @@ -9,6 +9,7 @@ from cognite.client.credentials import OAuthClientCertificate, OAuthClientCredentials, OAuthInteractive from cognite.client.data_classes import DataSet, DataSetWrite from cognite.client.data_classes.data_modeling import SpaceApply +from cognite.client.utils import timestamp_to_ms from tests.utils import REPO_ROOT @@ -17,6 +18,17 @@ def cognite_client() -> CogniteClient: return make_cognite_client(beta=False) +@pytest.fixture(autouse=True, scope="session") +def session_cleanup(cognite_client: CogniteClient): + resource_age = timestamp_to_ms("30m-ago") + + active_sessions = cognite_client.iam.sessions.list(status="ACTIVE", limit=-1) + sessions_to_revoke = [session.id for session in active_sessions if session.creation_time < resource_age] + + if sessions_to_revoke: + cognite_client.iam.sessions.revoke(sessions_to_revoke) + + @pytest.fixture(scope="session") def cognite_client_alpha() -> CogniteClient: load_dotenv(REPO_ROOT / "alpha.env") diff --git a/tests/tests_integration/test_api/test_transformations/test_schema.py b/tests/tests_integration/test_api/test_transformations/test_schema.py index db168257d..65f7751ba 100644 --- a/tests/tests_integration/test_api/test_transformations/test_schema.py +++ b/tests/tests_integration/test_api/test_transformations/test_schema.py @@ -21,7 +21,7 @@ def test_assets_delete(self, cognite_client): assert asset_columns[0].type.type == "long" assert asset_columns[0].sql_type == "BIGINT" assert asset_columns[0].name == "id" - assert asset_columns[0].nullable is True + # assert asset_columns[0].nullable is True # TODO: revert when schema is fixed, @silvavelosa def test_raw(self, cognite_client): asset_columns = cognite_client.transformations.schema.retrieve(destination=TransformationDestination.raw()) diff --git a/tests/tests_integration/test_api/test_workflows.py b/tests/tests_integration/test_api/test_workflows.py index 3432bb308..3f46dc812 100644 --- a/tests/tests_integration/test_api/test_workflows.py +++ b/tests/tests_integration/test_api/test_workflows.py @@ -33,9 +33,33 @@ WorkflowVersionUpsert, ) from cognite.client.exceptions import CogniteAPIError +from cognite.client.utils import timestamp_to_ms from cognite.client.utils._text import random_string +@pytest.fixture(autouse=True, scope="module") +def wf_setup_module(cognite_client: CogniteClient) -> None: + """setup any state specific to the execution of the given module.""" + resource_age = timestamp_to_ms("30m-ago") + + wf_triggers = cognite_client.workflows.triggers.list(limit=None) + wf_triggers_to_delete = [wf.external_id for wf in wf_triggers if wf.created_time < resource_age] + if wf_triggers_to_delete: + cognite_client.workflows.triggers.delete(wf_triggers_to_delete) + + wf_versions = cognite_client.workflows.versions.list(limit=None) + wf_versions_to_delete = [ + (wf.workflow_external_id, wf.version) for wf in wf_versions if wf.created_time < resource_age + ] + if wf_versions_to_delete: + cognite_client.workflows.versions.delete(wf_versions_to_delete) + + wfs = cognite_client.workflows.list(limit=None) + wfs_to_delete = [wf.external_id for wf in wfs if wf.created_time < resource_age] + if wfs_to_delete: + cognite_client.workflows.delete(wfs_to_delete) + + @pytest.fixture(scope="session") def data_set(cognite_client: CogniteClient) -> DataSet: return cognite_client.data_sets.list(limit=1)[0] diff --git a/tests/tests_unit/test_utils/test_time.py b/tests/tests_unit/test_utils/test_time.py index 44d4ea156..772c76e46 100644 --- a/tests/tests_unit/test_utils/test_time.py +++ b/tests/tests_unit/test_utils/test_time.py @@ -191,7 +191,7 @@ def test_float(self): @mock.patch("cognite.client.utils._time.time.time") @pytest.mark.parametrize( - "time_ago_string, expected_timestamp", + "time_shift_string, expected_timestamp", [ ("now", 10**12), ("1s-ago", 10**12 - 1 * 1000), @@ -204,19 +204,29 @@ def test_float(self): ("13d-ago", 10**12 - 13 * 24 * 60 * 60 * 1000), ("1w-ago", 10**12 - 1 * 7 * 24 * 60 * 60 * 1000), ("13w-ago", 10**12 - 13 * 7 * 24 * 60 * 60 * 1000), + ("1s-ahead", 10**12 + 1 * 1000), + ("13s-ahead", 10**12 + 13 * 1000), + ("1m-ahead", 10**12 + 1 * 60 * 1000), + ("13m-ahead", 10**12 + 13 * 60 * 1000), + ("1h-ahead", 10**12 + 1 * 60 * 60 * 1000), + ("13h-ahead", 10**12 + 13 * 60 * 60 * 1000), + ("1d-ahead", 10**12 + 1 * 24 * 60 * 60 * 1000), + ("13d-ahead", 10**12 + 13 * 24 * 60 * 60 * 1000), + ("1w-ahead", 10**12 + 1 * 7 * 24 * 60 * 60 * 1000), + ("13w-ahead", 10**12 + 13 * 7 * 24 * 60 * 60 * 1000), ], ) - def test_time_ago(self, time_mock, time_ago_string, expected_timestamp): + def test_time_shift(self, time_mock, time_shift_string, expected_timestamp): time_mock.return_value = 10**9 - assert timestamp_to_ms(time_ago_string) == expected_timestamp + assert timestamp_to_ms(time_shift_string) == expected_timestamp - @pytest.mark.parametrize("time_ago_string", ["1s", "4h", "13m-ag", "13m ago", "bla"]) - def test_invalid(self, time_ago_string): - with pytest.raises(ValueError, match=time_ago_string): - timestamp_to_ms(time_ago_string) + @pytest.mark.parametrize("time_shift_string", ["1s", "4h", "13m-ag", "13m-ahe", "13m ago", "13m ahead", "bla"]) + def test_invalid(self, time_shift_string): + with pytest.raises(ValueError, match=time_shift_string): + timestamp_to_ms(time_shift_string) - def test_time_ago_real_time(self): + def test_time_shift_real_time(self): expected_time_now = datetime.now().timestamp() * 1000 time_now = timestamp_to_ms("now") assert abs(expected_time_now - time_now) < 15