diff --git a/README.md b/README.md new file mode 100644 index 0000000..91ad924 --- /dev/null +++ b/README.md @@ -0,0 +1,401 @@ +# WHOOP for Python + +Tools for acquiring and analyzing Whoop API data. + +WHOOP is a wearable strap for monitoring sleep, activity, and workouts. Learn more about WHOOP at https://www.whoop.com. + +WHOOP API documentation can be found at https://developer.whoop.com/api. + +## Contents + +- [Installation](#installation) +- [Getting Started](#getting-started) +- [API Requests](#api-requests) + - [Get Profile](#get-profile) + - [Get Body Measurement](#get-body-measurement) + - [Get Cycle By ID](#get-cycle-by-id) + - [Get Cycle Collection](#get-cycle-collection) + - [Get Recovery For Cycle](#get-recovery-for-cycle) + - [Get Recovery Collection](#get-recovery-collection) + - [Get Sleep By ID](#get-sleep-by-id) + - [Get Sleep Collection](#get-sleep-collection) + - [Get Workout By ID](#get-workout-by-id) + - [Get Workout Collection](#get-workout-collection) + +## Installation + +The `whoop` module can be installed via pip: + +`pip install whoop` + +## Getting Started + +In order to use the WHOOP client, you must have your WHOOP email and password. + +It is best practice to store these values in a `.env` file: + +```bash +# WHOOP credentials +USERNAME="" +PASSWORD="" +``` + +You can use [`python-dotenv`](https://github.com/theskumar/python-dotenv) to load the enviroment variables for use in code: + +```python +import os +from dotenv import load_dotenv + +load_dotenv() + +un = os.getenv("USERNAME") or "" +pw = os.getenv("PASSWORD") or "" +``` + +Once the environment variables are loaded, a `WhoopClient` object can be created: + +```python +from whoop import WhoopClient + +# Using a traditional constructor +client = WhoopClient(username, password) +... + +# Using a context manager +with WhoopClient(username, password) as client: + ... +``` + +The WHOOP client will authenticate the client upon construction by default. This involves fetching an access token from the API. If you don't want this request to happen automatically, pass `authenticate=False` into the object constructor. In order to make other requests, you will need to manually call the `authenticate()` method so that the other requests have the proper authorization headers: + +```python +client = WhoopClient( + client_id, client_secret, refresh_token, authenticate=False +) + +client.authenticate() +... +``` + +## API Requests + +There are ten different API requests that `WhoopClient` can make. Full WHOOP API documentation can be found on [WHOOP's website](https://developer.whoop.com/api). + +### Get Profile + +Get the user's basic profile. + +**Method**: `get_profile()` + +**Payload**: None + +**Example Response**: + +```python +{ + "user_id": 10129, + "email": "jsmith123@whoop.com", + "first_name": "John", + "last_name": "Smith" +} +``` + +### Get Body Measurement + +Get the user's body measurements. + +**Method**: `get_body_measurement()` + +**Payload**: None + +**Example Response**: + +```python +{ + "height_meter": 1.8288, + "weight_kilogram": 90.7185, + "max_heart_rate": 200 +} +``` + +### Get Cycle By ID + +Get the cycle for the specified ID. + +**Method**: `get_cycle_by_id(cycle_id: str)` + +**Payload**: + +- `cycle_id`: ID of the cycle to retrieve. Passed into the request path. + +**Example Response**: + +```python +{ + "id": 93845, + "user_id": 10129, + "created_at": "2022-04-24T11:25:44.774Z", + "updated_at": "2022-04-24T14:25:44.774Z", + "start": "2022-04-24T02:25:44.774Z", + "end": "2022-04-24T10:25:44.774Z", + "timezone_offset": "-05:00", + "score_state": "SCORED", + "score": { + "strain": 5.2951527, + "kilojoule": 8288.297, + "average_heart_rate": 68, + "max_heart_rate": 141 + } +} +``` + +### Get Cycle Collection + +Get all physiological cycles for a user. Results are sorted by start time in descending order. + +**Method**: `get_cycle_collection(start_date: str = None, end_date: str = )` + +**Payload**: + +- `startDate`: The earliest date for which to get data, pulled from the `start_date` parameter. Returns cycles that occurred after or during (inclusive) this time. Expected in ISO 8601 format (YYYY-MM-DD HH:MM:SS). Defaults to no start date. +- `endDate`: The latest date for which to get data, pulled from the `end_date` parameter. Returns cycles that intersect this time or ended before (exclusive) this time. Expected in ISO 8601 format (YYYY-MM-DD HH:MM:SS). Defaults to today's date. + +**Example Response**: + +```python +[ + { + "id": 93845, + "user_id": 10129, + "created_at": "2022-04-24T11:25:44.774Z", + "updated_at": "2022-04-24T14:25:44.774Z", + "start": "2022-04-24T02:25:44.774Z", + "end": "2022-04-24T10:25:44.774Z", + "timezone_offset": "-05:00", + "score_state": "SCORED", + "score": { + "strain": 5.2951527, + "kilojoule": 8288.297, + "average_heart_rate": 68, + "max_heart_rate": 141 + } + }, + ... +] +``` + +### Get Recovery For Cycle + +Get the recovery for a cycle + +**Method**: `get_recovery_for_cycle(cycle_id: str)` + +**Payload**: + +- `cycle_id`: ID of the cycle to retrieve. Passed into the request path. + +**Example Response**: + +```python + { + "cycle_id": 93845, + "sleep_id": 10235, + "user_id": 10129, + "created_at": "2022-04-24T11:25:44.774Z", + "updated_at": "2022-04-24T14:25:44.774Z", + "score_state": "SCORED", + "score": { + "user_calibrating": False, + "recovery_score": 44, + "resting_heart_rate": 64, + "hrv_rmssd_milli": 31.813562, + "spo2_percentage": 95.6875, + "skin_temp_celsius": 33.7 + } +} +``` + +### Get Recovery Collection + +Get all recoveries for a user. Results are sorted by start time of the related sleep in descending order. + +**Method**: `get_recovery_collection(start_date: str = None, end_date: str = )` + +**Payload**: + +- `startDate`: The earliest date for which to get data, pulled from the `start_date` parameter. Returns cycles that occurred after or during (inclusive) this time. Expected in ISO 8601 format (YYYY-MM-DD HH:MM:SS). Defaults to no start date. +- `endDate`: The latest date for which to get data, pulled from the `end_date` parameter. Returns cycles that intersect this time or ended before (exclusive) this time. Expected in ISO 8601 format (YYYY-MM-DD HH:MM:SS). Defaults to today's date. + +**Example Response**: + +```python +[ + { + "cycle_id": 93845, + "sleep_id": 10235, + "user_id": 10129, + "created_at": "2022-04-24T11:25:44.774Z", + "updated_at": "2022-04-24T14:25:44.774Z", + "score_state": "SCORED", + "score": { + "user_calibrating": False, + "recovery_score": 44, + "resting_heart_rate": 64, + "hrv_rmssd_milli": 31.813562, + "spo2_percentage": 95.6875, + "skin_temp_celsius": 33.7 + } + }, + ... +] +``` + +### Get Sleep By ID + +Get the sleep for the specified ID. + +**Method**: `get_sleep_by_id(sleep_id: str)` + +**Payload**: + +- `sleep`: ID of the sleep to retrieve. Passed into the request path. + +**Example Response**: + +```python +{ + "id": 93845, + "user_id": 10129, + "created_at": "2022-04-24T11:25:44.774Z", + "updated_at": "2022-04-24T14:25:44.774Z", + "start": "2022-04-24T02:25:44.774Z", + "end": "2022-04-24T10:25:44.774Z", + "timezone_offset": "-05:00", + "nap": False, + "score_state": "SCORED", + "score": { + "stage_summary": {}, + "sleep_needed": {}, + "respiratory_rate": 16.11328125, + "sleep_performance_percentage": 98, + "sleep_consistency_percentage": 90, + "sleep_efficiency_percentage": 91.69533848 + } +} +``` + +### Get Sleep Collection + +Get all sleeps for a user. Results are sorted by start time in descending order. + +**Method**: `get_sleep_collection(start_date: str = None, end_date: str = )` + +**Payload**: + +- `startDate`: The earliest date for which to get data, pulled from the `start_date` parameter. Returns sleeps that occurred after or during (inclusive) this time. Expected in ISO 8601 format (YYYY-MM-DD HH:MM:SS). Defaults to no start date. +- `endDate`: The latest date for which to get data, pulled from the `end_date` parameter. Returns sleeps that intersect this time or ended before (exclusive) this time. Expected in ISO 8601 format (YYYY-MM-DD HH:MM:SS). Defaults to today's date. + +**Example Response**: + +```python +[ + { + "id": 93845, + "user_id": 10129, + "created_at": "2022-04-24T11:25:44.774Z", + "updated_at": "2022-04-24T14:25:44.774Z", + "start": "2022-04-24T02:25:44.774Z", + "end": "2022-04-24T10:25:44.774Z", + "timezone_offset": "-05:00", + "nap": False, + "score_state": "SCORED", + "score": { + "stage_summary": {}, + "sleep_needed": {}, + "respiratory_rate": 16.11328125, + "sleep_performance_percentage": 98, + "sleep_consistency_percentage": 90, + "sleep_efficiency_percentage": 91.69533848 + } + }, + ... +] +``` + +### Get Workout By ID + +Get the workout for the specified ID. + +**Method**: `get_workout_by_id(workout_id: str)` + +**Payload**: + +- `workout_id`: ID of the workout to retrieve. Passed into the request path. + +**Example Response**: + +```python +{ + "id": 1043, + "user_id": 9012, + "created_at": "2022-04-24T11:25:44.774Z", + "updated_at": "2022-04-24T14:25:44.774Z", + "start": "2022-04-24T02:25:44.774Z", + "end": "2022-04-24T10:25:44.774Z", + "timezone_offset": "-05:00", + "sport_id": 1, + "score_state": "SCORED", + "score": { + "strain": 8.2463, + "average_heart_rate": 123, + "max_heart_rate": 146, + "kilojoule": 1569.34033203125, + "percent_recorded": 100, + "distance_meter": 1772.77035916, + "altitude_gain_meter": 46.64384460449, + "altitude_change_meter": -0.781372010707855, + "zone_duration": {} + } +} +``` + +### Get Workout Collection + +Get all workouts for a user. Results are sorted by start time in descending order. + +**Method**: `get_workout_collection(start_date: str = None, end_date: str = )` + +**Payload**: + +- `startDate`: The earliest date for which to get data, pulled from the `start_date` parameter. Returns workouts that occurred after or during (inclusive) this time. Expected in ISO 8601 format (YYYY-MM-DD HH:MM:SS). Defaults to no start date. +- `endDate`: The latest date for which to get data, pulled from the `end_date` parameter. Returns workouts that intersect this time or ended before (exclusive) this time. Expected in ISO 8601 format (YYYY-MM-DD HH:MM:SS). Defaults to today's date. + +**Example Response**: + +```python +[ + { + "id": 1043, + "user_id": 9012, + "created_at": "2022-04-24T11:25:44.774Z", + "updated_at": "2022-04-24T14:25:44.774Z", + "start": "2022-04-24T02:25:44.774Z", + "end": "2022-04-24T10:25:44.774Z", + "timezone_offset": "-05:00", + "sport_id": 1, + "score_state": "SCORED", + "score": { + "strain": 8.2463, + "average_heart_rate": 123, + "max_heart_rate": 146, + "kilojoule": 1569.34033203125, + "percent_recorded": 100, + "distance_meter": 1772.77035916, + "altitude_gain_meter": 46.64384460449, + "altitude_change_meter": -0.781372010707855, + "zone_duration": {} + } + }, + ... +] +``` diff --git a/whoop.py b/whoop.py new file mode 100644 index 0000000..c5953d5 --- /dev/null +++ b/whoop.py @@ -0,0 +1,554 @@ +"""Tools for acquiring and analyzing Whoop API data. + +WHOOP is a wearable strap for monitoring sleep, activity, and workouts. Learn more about +WHOOP at https://www.whoop.com. + +WHOOP API documentation can be found at https://developer.whoop.com/api. + +Examples: + Loading environment variables: + import os + from dotenv import load_dotenv + + load_dotenv() + + username = os.getenv("USERNAME") or "" + password = os.getenv("PASSWORD") or "" + + Creating a client: + import whoop as wh + + client = wh.WhoopClient(username, password) + ... + + with wh.WhoopClient(username, password) as client: + ... + + Making requests: + client = wh.WhoopClient(username, password) + + sleep = client.get_sleep_collection() + recovery = client.get_recovery_collectuon() + + print(sleep) + print(recovery) + +Attributes: + AUTH_URL (str): Base URL for authorization requests. + REQUEST_URL (str): Base URL for API requests. +""" + +from __future__ import annotations + +import json +from datetime import datetime, time, timedelta +from typing import Any + +from authlib.common.urls import extract_params +from authlib.integrations.requests_client import OAuth2Session + + +AUTH_URL = "https://api-7.whoop.com" +REQUEST_URL = "https://api.prod.whoop.com/developer" + + +def _auth_password_json(_client, _method, uri, headers, body): + body = json.dumps(dict(extract_params(body))) + headers["Content-Type"] = "application/json" + + return uri, headers, body + + +class WhoopClient: + """Make requests to the WHOOP API. + + Attributes: + session (authlib.OAuth2Session): Requests session for accessing the WHOOP API. + user_id (str): User ID of the owner of the session. Will default to an empty + string before the session is authenticated and then replaced by the correct + user ID once a token is fetched. + + Raises: + ValueError: If `start_date` is after `end_date`. + """ + + TOKEN_ENDPOINT_AUTH_METHOD = "password_json" # noqa + + #################################################################################### + # INIT STUFF + + def __init__( + self, + username: str, + password: str, + authenticate: bool = True, + ): + """Initialize an OAuth2 session for making API requests. + + Optionally makes a request to the WHOOP API to acquire an access token. + + Args: + username (str): WHOOP account email. + password (str): WHOOP account password. + authenticate (bool): Whether to fetch a token from the API upon + session creation. If false, `authenticate()` must be called manually. + Defaults to true. + kwargs (dict[str, Any], optional): Additional arguments for OAuth2Session. + """ + self._username = username + self._password = password + + self.session = OAuth2Session( + token_endpont=f"{AUTH_URL}/oauth/token", + token_endpoint_auth_method=self.TOKEN_ENDPOINT_AUTH_METHOD, + ) + + self.session.register_client_auth_method( + (self.TOKEN_ENDPOINT_AUTH_METHOD, _auth_password_json) + ) + + self.user_id = "" + + if authenticate: + self.authenticate() + + def __enter__(self) -> WhoopClient: + """Enter a context manager. + + Returns: + WhoopClient: A WHOOP client with an active OAuth2Session. + """ + return self + + def __exit__(self, *_) -> None: + """Exit a context manager by closing the OAuth2 session. + + Args: + _ (Any): Exception arguments passed when closing context manager. + """ + self.close() + + def __str__(self) -> str: + """Generate string representation of client. + + Returns: + str: String representation of client featuring user ID of the owner of the + session. + """ + return f"WhoopClient({self.user_id if self.user_id else ''})" + + def close(self) -> None: + """Close the OAuth2 Session.""" + self.session.close() + + #################################################################################### + # API ENDPOINTS + + def get_profile(self) -> dict[str, Any]: + """Make request to Get Profile endpoint. + + Get the user's basic profile. + + Returns: + dict[str, Any]: Response JSON data loaded into an object. Example: + { + "user_id": 10129, + "email": "jsmith123@whoop.com", + "first_name": "John", + "last_name": "Smith" + } + """ + return self._make_request(method="GET", url_slug="v1/user/profile/basic") + + def get_body_measurement(self) -> dict[str, Any]: + """Make request to Get Body Measurement endpoint. + + Get the user's body measurements. + + Returns: + dict[str, Any]: Response JSON data loaded into an object. Example: + { + "height_meter": 1.8288, + "weight_kilogram": 90.7185, + "max_heart_rate": 200 + } + """ + return self._make_request(method="GET", url_slug="v1/user/measurement/body") + + def get_cycle_by_id(self, cycle_id: str) -> dict[str, Any]: + """Make request to Get Cycle By ID endpoint. + + Get the cycle for the specified ID. + + Returns: + dict[str, Any]: Response JSON data loaded into an object. Example: + { + "id": 93845, + "user_id": 10129, + "created_at": "2022-04-24T11:25:44.774Z", + "updated_at": "2022-04-24T14:25:44.774Z", + "start": "2022-04-24T02:25:44.774Z", + "end": "2022-04-24T10:25:44.774Z", + "timezone_offset": "-05:00", + "score_state": "SCORED", + "score": { + "strain": 5.2951527, + "kilojoule": 8288.297, + "average_heart_rate": 68, + "max_heart_rate": 141 + } + } + """ + return self._make_request(method="GET", url_slug=f"v1/cycle/{cycle_id}") + + def get_cycle_collection( + self, + start_date: str | None = None, + end_date: str | None = None, + ) -> list[dict[str, Any]]: + """Make request to Get Cycle Collection endpoint. + + Get all physiological cycles for a user. Results are sorted by start time in + descending order. + + Returns: + list[dict[str, Any]]: Response JSON data loaded into an object. Example: + [ + { + "id": 93845, + "user_id": 10129, + "created_at": "2022-04-24T11:25:44.774Z", + "updated_at": "2022-04-24T14:25:44.774Z", + "start": "2022-04-24T02:25:44.774Z", + "end": "2022-04-24T10:25:44.774Z", + "timezone_offset": "-05:00", + "score_state": "SCORED", + "score": { + "strain": 5.2951527, + "kilojoule": 8288.297, + "average_heart_rate": 68, + "max_heart_rate": 141 + } + }, + ... + ] + """ + start, end = self._format_dates(start_date, end_date) + + return self._make_paginated_request( + method="GET", + url_slug="v1/cycle", + params={"start": start, "end": end, "limit": 25}, + ) + + def get_recovery_for_cycle(self, cycle_id: str) -> dict[str, Any]: + """Make request to Get Recovery For Cycle endpoint. + + Get the recovery for a cycle. + + Returns: + dict[str, Any]: Response JSON data loaded into an object. Example: + { + "cycle_id": 93845, + "sleep_id": 10235, + "user_id": 10129, + "created_at": "2022-04-24T11:25:44.774Z", + "updated_at": "2022-04-24T14:25:44.774Z", + "score_state": "SCORED", + "score": { + "user_calibrating": False, + "recovery_score": 44, + "resting_heart_rate": 64, + "hrv_rmssd_milli": 31.813562, + "spo2_percentage": 95.6875, + "skin_temp_celsius": 33.7 + } + } + """ + return self._make_request( + method="GET", url_slug=f"v1/cycle/{cycle_id}/recovery" + ) + + def get_recovery_collection( + self, + start_date: str | None = None, + end_date: str | None = None, + ) -> list[dict[str, Any]]: + """Make request to Get Recovery Collection endpoint. + + Get all recoveries for a user. Results are sorted by start time of the related + sleep in descending order. + + Returns: + list[dict[str, Any]]: Response JSON data loaded into an object. Example: + [ + { + "cycle_id": 93845, + "sleep_id": 10235, + "user_id": 10129, + "created_at": "2022-04-24T11:25:44.774Z", + "updated_at": "2022-04-24T14:25:44.774Z", + "score_state": "SCORED", + "score": { + "user_calibrating": False, + "recovery_score": 44, + "resting_heart_rate": 64, + "hrv_rmssd_milli": 31.813562, + "spo2_percentage": 95.6875, + "skin_temp_celsius": 33.7 + } + }, + ... + ] + """ + start, end = self._format_dates(start_date, end_date) + + return self._make_paginated_request( + method="GET", + url_slug="v1/recovery", + params={"start": start, "end": end, "limit": 25}, + ) + + def get_sleep_by_id(self, sleep_id: str) -> dict[str, Any]: + """Make request to Get Sleep By ID endpoint. + + Get the sleep for the specified ID. + + Returns: + dict[str, Any]: Response JSON data loaded into an object. Example: + { + "id": 93845, + "user_id": 10129, + "created_at": "2022-04-24T11:25:44.774Z", + "updated_at": "2022-04-24T14:25:44.774Z", + "start": "2022-04-24T02:25:44.774Z", + "end": "2022-04-24T10:25:44.774Z", + "timezone_offset": "-05:00", + "nap": False, + "score_state": "SCORED", + "score": { + "stage_summary": {}, + "sleep_needed": {}, + "respiratory_rate": 16.11328125, + "sleep_performance_percentage": 98, + "sleep_consistency_percentage": 90, + "sleep_efficiency_percentage": 91.69533848 + } + } + """ + return self._make_request( + method="GET", url_slug=f"v1/activity/sleep/{sleep_id}" + ) + + def get_sleep_collection( + self, + start_date: str | None = None, + end_date: str | None = None, + ) -> list[dict[str, Any]]: + """Make request to Get Sleep Collection endpoint. + + Get all sleeps for a user. Results are sorted by start time in descending order. + + Returns: + list[dict[str, Any]]: Response JSON data loaded into an object. Example: + [ + { + "id": 93845, + "user_id": 10129, + "created_at": "2022-04-24T11:25:44.774Z", + "updated_at": "2022-04-24T14:25:44.774Z", + "start": "2022-04-24T02:25:44.774Z", + "end": "2022-04-24T10:25:44.774Z", + "timezone_offset": "-05:00", + "nap": False, + "score_state": "SCORED", + "score": { + "stage_summary": {}, + "sleep_needed": {}, + "respiratory_rate": 16.11328125, + "sleep_performance_percentage": 98, + "sleep_consistency_percentage": 90, + "sleep_efficiency_percentage": 91.69533848 + } + }, + ... + ] + """ + start, end = self._format_dates(start_date, end_date) + + return self._make_paginated_request( + method="GET", + url_slug="v1/activity/sleep", + params={"start": start, "end": end, "limit": 25}, + ) + + def get_workout_by_id(self, workout_id: str) -> dict[str, Any]: + """Make request to Get Workout By ID endpoint. + + Get the workout for the specified ID. + + Returns: + dict[str, Any]: Response JSON data loaded into an object. Example: + { + "id": 1043, + "user_id": 9012, + "created_at": "2022-04-24T11:25:44.774Z", + "updated_at": "2022-04-24T14:25:44.774Z", + "start": "2022-04-24T02:25:44.774Z", + "end": "2022-04-24T10:25:44.774Z", + "timezone_offset": "-05:00", + "sport_id": 1, + "score_state": "SCORED", + "score": { + "strain": 8.2463, + "average_heart_rate": 123, + "max_heart_rate": 146, + "kilojoule": 1569.34033203125, + "percent_recorded": 100, + "distance_meter": 1772.77035916, + "altitude_gain_meter": 46.64384460449, + "altitude_change_meter": -0.781372010707855, + "zone_duration": {} + } + } + """ + return self._make_request( + method="GET", url_slug=f"v1/activity/workout/{workout_id}" + ) + + def get_workout_collection( + self, + start_date: str | None = None, + end_date: str | None = None, + ) -> list[dict[str, Any]]: + """Make request to Get Workout Collection endpoint. + + Get all workouts for a user. Results are sorted by start time in descending + order. + + Returns: + list[dict[str, Any]]: Response JSON data loaded into an object. Example: + [ + { + "id": 1043, + "user_id": 9012, + "created_at": "2022-04-24T11:25:44.774Z", + "updated_at": "2022-04-24T14:25:44.774Z", + "start": "2022-04-24T02:25:44.774Z", + "end": "2022-04-24T10:25:44.774Z", + "timezone_offset": "-05:00", + "sport_id": 1, + "score_state": "SCORED", + "score": { + "strain": 8.2463, + "average_heart_rate": 123, + "max_heart_rate": 146, + "kilojoule": 1569.34033203125, + "percent_recorded": 100, + "distance_meter": 1772.77035916, + "altitude_gain_meter": 46.64384460449, + "altitude_change_meter": -0.781372010707855, + "zone_duration": {} + } + }, + ... + ] + """ + start, end = self._format_dates(start_date, end_date) + + return self._make_paginated_request( + method="GET", + url_slug="v1/activity/workout", + params={"start": start, "end": end, "limit": 25}, + ) + + #################################################################################### + # API HELPER METHODS + + def authenticate(self, **kwargs) -> None: + """Authenticate OAuth2Session by fetching token. + + If `user_id` is `None`, it will be set according to the `user_id` returned with + the token. + + Args: + kwargs (dict[str, Any], optional): Additional arguments for `fetch_token()`. + """ + self.session.fetch_token( + url=f"{AUTH_URL}/oauth/token", + username=self._username, + password=self._password, + grant_type="password", + **kwargs, + ) + + if not self.user_id: + self.user_id = str(self.session.token.get("user", {}).get("id", "")) + + def is_authenticated(self) -> bool: + """Check if the OAuth2Session is authenticated. + + Returns: + bool: Whether the OAuth2Session has a token and is therefore authenticated. + """ + return self.session.token is not None + + def _make_paginated_request( + self, method, url_slug, **kwargs + ) -> list[dict[str, Any]]: + params = kwargs.pop("params", {}) + response_data: list[dict[str, Any]] = [] + + while True: + response = self._make_request( + method=method, + url_slug=url_slug, + params=params, + **kwargs, + ) + + response_data += response["records"] + + if next_token := response["next_token"]: + params["nextToken"] = next_token + + else: + break + + return response_data + + def _make_request( + self, method: str, url_slug: str, **kwargs: Any + ) -> dict[str, Any]: + response = self.session.request( + method=method, + url=f"{REQUEST_URL}/{url_slug}", + **kwargs, + ) + + response.raise_for_status() + + return response.json() + + def _format_dates( + self, start_date: str | None, end_date: str | None + ) -> tuple[str, str]: + end = datetime.combine( + datetime.fromisoformat(end_date) if end_date else datetime.today(), time.max + ) + start = datetime.combine( + datetime.fromisoformat(start_date) + if start_date + else datetime.today() - timedelta(days=6), + time.min, + ) + + if start > end: + raise ValueError( + f"Start datetime greater than end datetime: {start} > {end}" + ) + + return ( + start.isoformat() + "Z", + end.isoformat(timespec="seconds") + "Z", + )