Module redvox.cloud.client
This module contains the RedVox Cloud API client.
This client provides convenient access to RedVox metadata and data. This client manages the tedious business of keeping an up-to-date authentication token for making authenticated API requests.
Expand source code
"""
This module contains the RedVox Cloud API client.
This client provides convenient access to RedVox metadata and data. This client manages the tedious business of keeping
an up-to-date authentication token for making authenticated API requests.
"""
import contextlib
import threading
from multiprocessing import Queue
from typing import List, Optional, Tuple, TYPE_CHECKING, Iterator
import requests
import redvox.cloud.api as api
import redvox.cloud.auth_api as auth_api
from redvox.cloud.config import RedVoxConfig
import redvox.cloud.errors as cloud_errors
import redvox.common.constants as constants
from redvox.cloud.query_timing_correction import (
    correct_query_timing as do_correct_query_timing,
)
import redvox.cloud.data_api as data_api
import redvox.cloud.metadata_api as metadata_api
import redvox.cloud.station_stats as station_stats_api
import redvox.cloud.session_model_api as session_model_api
if TYPE_CHECKING:
    from redvox.cloud.query_timing_correction import CorrectedQuery
def chunk_time_range(
    start_ts: int, end_ts: int, max_chunk: int
) -> List[Tuple[int, int]]:
    """
    Chunks the given request window into smaller windows.
    :param start_ts: Start of the request window.
    :param end_ts: End of the request window.
    :param max_chunk: Max chunk size.
    :return: A list of window chunks.
    """
    if end_ts <= start_ts:
        raise cloud_errors.CloudApiError("start_ts must be < end_ts")
    if max_chunk <= 0:
        raise cloud_errors.CloudApiError("max_chunk must be > 0")
    chunks: List[Tuple[int, int]] = []
    start: int = start_ts
    while start + max_chunk < end_ts:
        chunks.append((start, start + max_chunk))
        start += max_chunk
    if start < end_ts:
        chunks.append((start, end_ts))
    return chunks
class CloudClient:
    """
    The RedVox Cloud API client.
    """
    def __init__(
        self,
        redvox_config: Optional[RedVoxConfig] = RedVoxConfig.find(),
        refresh_token_interval: float = 600.0,
        timeout: Optional[float] = 10.0,
    ):
        """
        Instantiates this client.
        :param redvox_config: The Redvox endpoint configuration.
        :param refresh_token_interval: An optional interval in seconds that the auth token should be refreshed.
        :param timeout: An optional timeout
        """
        if redvox_config is None:
            raise cloud_errors.CloudApiError(
                "A RedVoxConfig was not found in the environment and one wasn't provided"
            )
        if refresh_token_interval <= 0:
            raise cloud_errors.CloudApiError(
                "refresh_token_interval must be strictly > 0"
            )
        if timeout is not None and (timeout <= 0):
            raise cloud_errors.CloudApiError("timeout must be strictly > 0")
        self.redvox_config: RedVoxConfig = redvox_config
        self.refresh_token_interval: float = refresh_token_interval
        self.timeout: Optional[float] = timeout
        self.__session = (
            requests.Session()
        )  # This must be initialized before the auth req!
        self.__refresh_timer = None
        self.__authenticate()
        self.__refresh_timer = threading.Timer(
            self.refresh_token_interval, self.__refresh_token
        )
        self.__refresh_timer.start()
    def __authenticate(self) -> None:
        # Token provided
        if self.redvox_config.username == "auth_token":
            self.auth_token = self.redvox_config.password
        # Username/Password provided
        else:
            try:
                auth_resp: auth_api.AuthResp = self.authenticate_user(
                    self.redvox_config.username, self.redvox_config.password
                )
            except cloud_errors.ApiConnectionError as ex:
                self.close()
                raise ex
            if (
                auth_resp.status != 200
                or auth_resp.auth_token is None
                or len(auth_resp.auth_token) == 0
            ):
                self.close()
                raise cloud_errors.AuthenticationError()
            self.auth_token: str = auth_resp.auth_token
    def __refresh_token(self):
        """
        Automatically refreshes this client's auth token in the background once per minute.
        """
        # pylint: disable=W0702
        # noinspection PyBroadException
        try:
            self.auth_token = self.refresh_own_auth_token().auth_token
            self.__refresh_timer = threading.Timer(
                self.refresh_token_interval, self.__refresh_token
            )
            self.__refresh_timer.start()
        except:
            self.close()
    def close(self):
        """
        Terminates this client process by cancelling the refresh token timer.
        """
        # pylint: disable=W0702
        # noinspection PyBroadException
        try:
            if self.__refresh_timer is not None:
                self.__refresh_timer.cancel()
        except:
            pass
        # pylint: disable=W0702
        # noinspection PyBroadException
        try:
            if self.__session is not None:
                self.__session.close()
        except:
            pass
    def health_check(self) -> bool:
        """
        An API call that returns True if the API Cloud server is up and running or False otherwise.
        :return: True if the API Cloud server is up and running or False otherwise.
        """
        return api.health_check(
            self.redvox_config, session=self.__session, timeout=self.timeout
        )
    def authenticate_user(self, username: str, password: str) -> auth_api.AuthResp:
        """
        Attempts to authenticate the given RedVox user.
        :param username: The RedVox username.
        :param password: The RedVox password.
        :return: An authenticate response.
        """
        if len(username) == 0:
            raise cloud_errors.CloudApiError("Username must be provided")
        if len(password) == 0:
            raise cloud_errors.CloudApiError("Password must be provided")
        auth_req: auth_api.AuthReq = auth_api.AuthReq(username, password)
        return auth_api.authenticate_user(
            self.redvox_config, auth_req, session=self.__session, timeout=self.timeout
        )
    def validate_auth_token(
        self, auth_token: str
    ) -> Optional[auth_api.ValidateTokenResp]:
        """
        Validates the provided authentication token with the cloud API.
        :param auth_token: Authentication token to validate.
        :return: An authentication response with token details or None if token in invalid
        """
        if len(auth_token) == 0:
            raise cloud_errors.CloudApiError("auth_token must be provided")
        token_req: auth_api.ValidateTokenReq = auth_api.ValidateTokenReq(auth_token)
        return auth_api.validate_token(
            self.redvox_config, token_req, session=self.__session, timeout=self.timeout
        )
    def validate_own_auth_token(self) -> Optional[auth_api.ValidateTokenResp]:
        """
        Validates the current token held by this client.
        :return: An authentication response with token details or None if token in invalid
        """
        return self.validate_auth_token(self.auth_token)
    def refresh_auth_token(
        self, auth_token: str
    ) -> Optional[auth_api.RefreshTokenResp]:
        """
        Retrieves a new authentication token from a given valid authentication token.
        :param auth_token: The authentication token to verify.
        :return: A new authentication token or None if the provide auth token is not valid.
        """
        if len(auth_token) == 0:
            raise cloud_errors.CloudApiError("auth_token must be provided")
        refresh_token_req: auth_api.RefreshTokenReq = auth_api.RefreshTokenReq(
            auth_token
        )
        return auth_api.refresh_token(
            self.redvox_config,
            refresh_token_req,
            session=self.__session,
            timeout=self.timeout,
        )
    def refresh_own_auth_token(self) -> Optional[auth_api.RefreshTokenResp]:
        """
        Retrieves a new authentication token using the token held by this client as a reference.
        :return: A new authentication token or None if the provide auth token is not valid.
        """
        return self.refresh_auth_token(self.auth_token)
    def request_metadata(
        self,
        start_ts_s: int,
        end_ts_s: int,
        station_ids: List[str],
        metadata_to_include: List[str],
        chunk_by_seconds: int = constants.SECONDS_PER_DAY,
    ) -> Optional[metadata_api.MetadataResp]:
        """
        Requests RedVox packet metadata.
        :param start_ts_s: Start epoch of request window.
        :param end_ts_s: End epoch of request window.
        :param station_ids: A list of station ids.
        :param metadata_to_include: A list of metadata fields to include (see: redvox.cloud.metadata.AvailableMetadata)
        :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d)
        :return: A metadata result containing the requested metadata or None on error.
        """
        if end_ts_s <= start_ts_s:
            raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s")
        if len(station_ids) == 0:
            raise cloud_errors.CloudApiError("At least one station_id must be included")
        if len(metadata_to_include) == 0:
            raise cloud_errors.CloudApiError(
                "At least one metadata field must be included"
            )
        if chunk_by_seconds <= 0:
            raise cloud_errors.CloudApiError("chunk_by_seconds must be > 0")
        time_chunks: List[Tuple[int, int]] = chunk_time_range(
            start_ts_s, end_ts_s, chunk_by_seconds
        )
        metadata_resp: metadata_api.MetadataResp = metadata_api.MetadataResp([])
        for start_ts, end_ts in time_chunks:
            metadata_req: metadata_api.MetadataReq = metadata_api.MetadataReq(
                self.auth_token,
                start_ts,
                end_ts,
                station_ids,
                metadata_to_include,
                self.redvox_config.secret_token,
            )
            chunked_resp: Optional[
                metadata_api.MetadataResp
            ] = metadata_api.request_metadata(
                self.redvox_config,
                metadata_req,
                session=self.__session,
                timeout=self.timeout,
            )
            if chunked_resp:
                metadata_resp.metadata.extend(chunked_resp.metadata)
        return metadata_resp
    def request_metadata_m(
        self,
        start_ts_s: int,
        end_ts_s: int,
        station_ids: List[str],
        metadata_to_include: List[str],
        chunk_by_seconds: int = constants.SECONDS_PER_HOUR,
    ) -> Optional[metadata_api.MetadataRespM]:
        """
        Requests RedVox packet metadata.
        :param start_ts_s: Start epoch of request window.
        :param end_ts_s: End epoch of request window.
        :param station_ids: A list of station ids.
        :param metadata_to_include: A list of metadata fields to include (see: redvox.cloud.metadata.AvailableMetadata)
        :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d)
        :return: A metadata result containing the requested metadata or None on error.
        """
        metadata_resp: metadata_api.MetadataRespM = metadata_api.MetadataRespM([])
        resp: Optional[metadata_api.MetadataRespM]
        for resp in self.request_metadata_m_stream(
            start_ts_s, end_ts_s, station_ids, metadata_to_include, chunk_by_seconds
        ):
            if resp is not None:
                metadata_resp.db_packets.extend(resp.db_packets)
        return metadata_resp
    def request_metadata_m_stream(
        self,
        start_ts_s: int,
        end_ts_s: int,
        station_ids: List[str],
        metadata_to_include: List[str],
        chunk_by_seconds: int = constants.SECONDS_PER_HOUR,
    ) -> Iterator[Optional[metadata_api.MetadataRespM]]:
        """
        Requests RedVox packet metadata.
        :param start_ts_s: Start epoch of request window.
        :param end_ts_s: End epoch of request window.
        :param station_ids: A list of station ids.
        :param metadata_to_include: A list of metadata fields to include (see: redvox.cloud.metadata.AvailableMetadata)
        :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d)
        :return: A metadata result containing the requested metadata or None on error.
        """
        if end_ts_s <= start_ts_s:
            raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s")
        if len(station_ids) == 0:
            raise cloud_errors.CloudApiError("At least one station_id must be included")
        if len(metadata_to_include) == 0:
            raise cloud_errors.CloudApiError(
                "At least one metadata field must be included"
            )
        if chunk_by_seconds <= 0:
            raise cloud_errors.CloudApiError("chunk_by_seconds must be > 0")
        time_chunks: List[Tuple[int, int]] = chunk_time_range(
            start_ts_s, end_ts_s, chunk_by_seconds
        )
        for start_ts, end_ts in time_chunks:
            metadata_req: metadata_api.MetadataReq = metadata_api.MetadataReq(
                self.auth_token,
                start_ts,
                end_ts,
                station_ids,
                metadata_to_include,
                self.redvox_config.secret_token,
            )
            chunked_resp: Optional[
                metadata_api.MetadataRespM
            ] = metadata_api.request_metadata_m(
                self.redvox_config,
                metadata_req,
                session=self.__session,
                timeout=self.timeout,
            )
            yield chunked_resp
    def request_geo_metadata_stream(
        self,
        start_ts_s: int,
        end_ts_s: int,
        bounding_box: Optional[metadata_api.BoundingBox],
        bounding_circle: Optional[metadata_api.BoundingCircle],
        metadata_to_include: List[str],
        chunk_by_seconds: int = constants.SECONDS_PER_HOUR,
    ) -> Iterator[Optional[metadata_api.GeoMetadataResp]]:
        """
        Requests RedVox packet metadata from geo bounds.
        :param start_ts_s: Start epoch of request window.
        :param end_ts_s: End epoch of request window.
        :param bounding_box: A bounding box given by the SW and NE coordinates. Mutually exclusive with bounding_circle.
        :param bounding_circle: A bounding circle given by a center point and radius in meter. Mutually exclusive with bounding_box.
        :param metadata_to_include: A list of metadata fields to include (see: redvox.cloud.metadata.AvailableMetadata)
        :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d)
        :return: A geo metadata result containing the requested metadata or None on error.
        """
        if end_ts_s <= start_ts_s:
            raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s")
        if bounding_box is None and bounding_circle is None:
            raise cloud_errors.CloudApiError(
                "Exactly one of bounding_box or bounding_circle must be set"
            )
        if bounding_box is not None and bounding_circle is not None:
            raise cloud_errors.CloudApiError(
                "Exactly one of bounding_box or bounding_circle must be set"
            )
        if len(metadata_to_include) == 0:
            raise cloud_errors.CloudApiError(
                "At least one metadata field must be included"
            )
        if chunk_by_seconds <= 0:
            raise cloud_errors.CloudApiError("chunk_by_seconds must be > 0")
        time_chunks: List[Tuple[int, int]] = chunk_time_range(
            start_ts_s, end_ts_s, chunk_by_seconds
        )
        for start_ts, end_ts in time_chunks:
            geo_metadata_req: metadata_api.GeoMetadataReq = metadata_api.GeoMetadataReq(
                self.auth_token,
                start_ts,
                end_ts,
                bounding_box,
                bounding_circle,
                metadata_to_include,
            )
            chunked_resp: Optional[
                metadata_api.GeoMetadataResp
            ] = metadata_api.request_geo_metadata(
                self.redvox_config,
                geo_metadata_req,
                session=self.__session,
                timeout=self.timeout,
            )
            yield chunked_resp
    def request_timing_metadata(
        self,
        start_ts_s: int,
        end_ts_s: int,
        station_ids: List[str],
        chunk_by_seconds: int = constants.SECONDS_PER_DAY,
    ) -> metadata_api.TimingMetaResponse:
        """
        Requests timing metadata from RedVox packets.
        :param start_ts_s: Start epoch of the request.
        :param end_ts_s: End epoch of the request.
        :param station_ids: A list of station ids.
        :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d)
        :return: A response containing the requested metadata.
        """
        if end_ts_s <= start_ts_s:
            raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s")
        if len(station_ids) == 0:
            raise cloud_errors.CloudApiError("At least one station_id must be included")
        if chunk_by_seconds <= 0:
            raise cloud_errors.CloudApiError("chunk_by_seconds must be > 0")
        time_chunks: List[Tuple[int, int]] = chunk_time_range(
            start_ts_s, end_ts_s, chunk_by_seconds
        )
        metadata_resp: metadata_api.TimingMetaResponse = (
            metadata_api.TimingMetaResponse([])
        )
        for start_ts, end_ts in time_chunks:
            timing_req: metadata_api.TimingMetaRequest = metadata_api.TimingMetaRequest(
                self.auth_token,
                start_ts,
                end_ts,
                station_ids,
                self.redvox_config.secret_token,
            )
            chunked_resp: metadata_api.TimingMetaResponse = (
                metadata_api.request_timing_metadata(
                    self.redvox_config,
                    timing_req,
                    session=self.__session,
                    timeout=self.timeout,
                )
            )
            if chunked_resp:
                metadata_resp.items.extend(chunked_resp.items)
        return metadata_resp
    def request_report_data(self, report_id: str) -> Optional[data_api.ReportDataResp]:
        """
        Requests a signed URL for a given report ID.
        :param report_id: The report ID to request data for.
        :return: A response containing a signed URL of the report data.
        """
        if len(report_id) == 0:
            raise cloud_errors.CloudApiError("report_id must be included")
        report_data_req: data_api.ReportDataReq = data_api.ReportDataReq(
            self.auth_token, report_id, self.redvox_config.secret_token
        )
        return data_api.request_report_data(
            self.redvox_config,
            report_data_req,
            session=self.__session,
            timeout=self.timeout,
        )
    def request_station_statuses(
        self,
        start_ts_s: int,
        end_ts_s: int,
        station_ids: List[str],
    ) -> Optional[metadata_api.StationStatusResp]:
        """
        Requests station timing information from the cloud services.
        :param start_ts_s: The start of the request data window.
        :param end_ts_s: The end of the request data window.
        :param station_ids: A list of station IDs.
        :return: A StationStatsResp.
        """
        if end_ts_s <= start_ts_s:
            raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s")
        station_status_req: metadata_api.StationStatusReq = (
            metadata_api.StationStatusReq(
                self.redvox_config.secret_token,
                self.auth_token,
                start_ts_s,
                end_ts_s,
                station_ids,
            )
        )
        return metadata_api.request_station_statuses(
            self.redvox_config,
            station_status_req,
            session=self.__session,
            timeout=self.timeout,
        )
    def request_data_range(
        self,
        start_ts_s: int,
        end_ts_s: int,
        station_ids: List[str],
        req_type: data_api.DataRangeReqType = data_api.DataRangeReqType.API_900_1000,
        correct_query_timing: bool = True,
        out_queue: Optional[Queue] = None,
    ) -> data_api.DataRangeResp:
        """
        Request signed URLs for RedVox packets.
        :param start_ts_s: The start epoch of the window.
        :param end_ts_s:  The end epoch of the window.
        :param station_ids: A list of station ids.
        :param req_type: The type of data to request.
        :param correct_query_timing: If set to true, timing correction will be applied to each station before the data is
                                    correct_timing queried.
        :return: A response containing a list of signed URLs for the RedVox packets.
        """
        def _make_req(
            _start_ts_s: int, _end_ts_s: int, _station_ids: List[str]
        ) -> data_api.DataRangeResp:
            """
            Makes the actual data request after timing correction was or was not applied.
            :param _start_ts_s: The start epoch of the window.
            :param _end_ts_s: The end epoch of the window.
            :param _station_ids: A list of station IDs.
            :return: A response containing a list of signed URLs for the RedVox packets.
            """
            data_range_req: data_api.DataRangeReq = data_api.DataRangeReq(
                self.auth_token,
                _start_ts_s,
                _end_ts_s,
                _station_ids,
                self.redvox_config.secret_token,
            )
            req_resp: data_api.DataRangeResp = data_api.request_range_data(
                self.redvox_config,
                data_range_req,
                session=self.__session,
                timeout=self.timeout,
                req_type=req_type,
            )
            return req_resp
        if end_ts_s <= start_ts_s:
            raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s")
        if len(station_ids) == 0:
            raise cloud_errors.CloudApiError("At least one station_id must be provided")
        # If timing correction was requested, we want to find the corrected start and end offsets in order to
        # request the correct range of data for each station's particular clock offset
        if correct_query_timing:
            corrected_queries: Optional[
                List["CorrectedQuery"]
            ] = do_correct_query_timing(self, start_ts_s, end_ts_s, station_ids)
            # If we get nothing back, we can't apply a correction
            if corrected_queries is None or len(corrected_queries) == 0:
                print("No timing corrections returned, running original query.")
                return _make_req(start_ts_s, end_ts_s, station_ids)
            # Make a request for each corrected query, aggregating the results of each request
            resp: data_api.DataRangeResp = data_api.DataRangeResp([])
            corrected_query: "CorrectedQuery"
            for corrected_query in corrected_queries:
                correction_msg: str = (
                    f"Running timing corrected query for {corrected_query.station_id} "
                    f"start offset={corrected_query.start_offset()} "
                    f"end offset={corrected_query.end_offset()}"
                )
                if out_queue is None:
                    print(correction_msg)
                else:
                    out_queue.put(correction_msg, block=False)
                resp.append(
                    _make_req(
                        round(corrected_query.corrected_start_ts),
                        round(corrected_query.corrected_end_ts),
                        [corrected_query.station_id],
                    )
                )
            return resp
        else:
            # No timing correction requested, go ahead just make the original uncorrected request
            return _make_req(start_ts_s, end_ts_s, station_ids)
    def request_station_stats(
        self,
        start_ts_s: int,
        end_ts_s: int,
        station_ids: List[str],
    ) -> Optional[station_stats_api.StationStatsResp]:
        """
        Request signed URLs for RedVox packets.
        :param start_ts_s: The start epoch of the window.
        :param end_ts_s:  The end epoch of the window.
        :param station_ids: A list of station ids.
        :return: A response containing a list of signed URLs for the RedVox packets.
        """
        if end_ts_s <= start_ts_s:
            raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s")
        if len(station_ids) == 0:
            raise cloud_errors.CloudApiError("At least one station_id must be provided")
        station_stats_req: station_stats_api.StationStatReq = (
            station_stats_api.StationStatReq(
                self.auth_token,
                start_ts_s,
                end_ts_s,
                station_ids,
                self.redvox_config.secret_token,
            )
        )
        return station_stats_api.request_station_stats(
            self.redvox_config, station_stats_req, self.__session, self.timeout
        )
    def request_session_model(
        self, session_key: str
    ) -> session_model_api.SessionModelResp:
        """
        Requests a session model.
        :param session_key: The session key.
        :return: An instance of a SessionModelResp.
        """
        req: session_model_api.SessionModelReq = session_model_api.SessionModelReq(
            self.auth_token, session_key
        )
        return session_model_api.request_session(
            self.redvox_config, req, self.__session, self.timeout
        )
    def request_session_models(
        self,
        id_uuids: Optional[List[str]] = None,
        owner: Optional[str] = None,
        start_ts: Optional[int] = None,
        end_ts: Optional[int] = None,
        include_public: bool = False,
    ) -> session_model_api.SessionModelsResp:
        """
        Requests a range of session models.
        :param id_uuids: An optional list of IDs or ID:UUIDs.
        :param owner: An optional owner.
        :param start_ts: An optional start timestamp (microseconds).
        :param end_ts: An optional end timestamp (microseconds).
        :param include_public: Additionally include public sessions that may not be the same as the owner.
        :return: An instance of SessionModelsResp.
        """
        req: session_model_api.SessionModelsReq = session_model_api.SessionModelsReq(
            self.auth_token, id_uuids, owner, start_ts, end_ts, include_public
        )
        return session_model_api.request_sessions(
            self.redvox_config, req, self.__session, self.timeout
        )
    def request_dynamic_session_model(
        self,
        session_key: str,
        start_ts: int,
        end_ts: int,
    ) -> session_model_api.DynamicSessionModelResp:
        """
        Requests a dynamic session given the fields that make up its key.
        :param session_key: The parent session model key.
        :param start_ts: The start of the dynamic session (microseconds).
        :param end_ts: The end of the dynamic session (microseconds).
        :return: An instance of DynamicSessionModelResp.
        """
        req: session_model_api.DynamicSessionModelReq = (
            session_model_api.DynamicSessionModelReq(
                self.auth_token, session_key, start_ts, end_ts
            )
        )
        return session_model_api.request_dynamic_session(
            self.redvox_config, req, self.__session, self.timeout
        )
@contextlib.contextmanager
def cloud_client(
    redvox_config: Optional[RedVoxConfig] = RedVoxConfig.find(),
    refresh_token_interval: float = 600.0,
    timeout: float = 10.0,
):
    """
    Function that can be used within a "with" block to automatically handle the closing of open resources.
    Creates and returns a CloudClient that will automatically be closed when exiting the with block or if an error
    occurs.
    See https://docs.python.org/3/library/contextlib.html for more info.
    :param redvox_config: The Redvox endpoint configuration.
    :param refresh_token_interval: An optional token refresh interval
    :param timeout: An optional timeout.
    :return: A CloudClient.
    """
    if redvox_config is None:
        raise cloud_errors.CloudApiError(
            "A RedVoxConfig was not found in the environment and one wasn't provided"
        )
    client: CloudClient = CloudClient(redvox_config, refresh_token_interval, timeout)
    try:
        yield client
    finally:
        if client is not None:
            client.close()Functions
- def chunk_time_range(start_ts: int, end_ts: int, max_chunk: int) ‑> List[Tuple[int, int]]
- 
Chunks the given request window into smaller windows. :param start_ts: Start of the request window. :param end_ts: End of the request window. :param max_chunk: Max chunk size. :return: A list of window chunks. Expand source codedef chunk_time_range( start_ts: int, end_ts: int, max_chunk: int ) -> List[Tuple[int, int]]: """ Chunks the given request window into smaller windows. :param start_ts: Start of the request window. :param end_ts: End of the request window. :param max_chunk: Max chunk size. :return: A list of window chunks. """ if end_ts <= start_ts: raise cloud_errors.CloudApiError("start_ts must be < end_ts") if max_chunk <= 0: raise cloud_errors.CloudApiError("max_chunk must be > 0") chunks: List[Tuple[int, int]] = [] start: int = start_ts while start + max_chunk < end_ts: chunks.append((start, start + max_chunk)) start += max_chunk if start < end_ts: chunks.append((start, end_ts)) return chunks
- def cloud_client(redvox_config: Optional[RedVoxConfig] = None, refresh_token_interval: float = 600.0, timeout: float = 10.0)
- 
Function that can be used within a "with" block to automatically handle the closing of open resources. Creates and returns a CloudClient that will automatically be closed when exiting the with block or if an error occurs. See https://docs.python.org/3/library/contextlib.html for more info. :param redvox_config: The Redvox endpoint configuration. :param refresh_token_interval: An optional token refresh interval :param timeout: An optional timeout. :return: A CloudClient. Expand source code@contextlib.contextmanager def cloud_client( redvox_config: Optional[RedVoxConfig] = RedVoxConfig.find(), refresh_token_interval: float = 600.0, timeout: float = 10.0, ): """ Function that can be used within a "with" block to automatically handle the closing of open resources. Creates and returns a CloudClient that will automatically be closed when exiting the with block or if an error occurs. See https://docs.python.org/3/library/contextlib.html for more info. :param redvox_config: The Redvox endpoint configuration. :param refresh_token_interval: An optional token refresh interval :param timeout: An optional timeout. :return: A CloudClient. """ if redvox_config is None: raise cloud_errors.CloudApiError( "A RedVoxConfig was not found in the environment and one wasn't provided" ) client: CloudClient = CloudClient(redvox_config, refresh_token_interval, timeout) try: yield client finally: if client is not None: client.close()
Classes
- class CloudClient (redvox_config: Optional[RedVoxConfig] = None, refresh_token_interval: float = 600.0, timeout: Optional[float] = 10.0)
- 
The RedVox Cloud API client. Instantiates this client. :param redvox_config: The Redvox endpoint configuration. :param refresh_token_interval: An optional interval in seconds that the auth token should be refreshed. :param timeout: An optional timeout Expand source codeclass CloudClient: """ The RedVox Cloud API client. """ def __init__( self, redvox_config: Optional[RedVoxConfig] = RedVoxConfig.find(), refresh_token_interval: float = 600.0, timeout: Optional[float] = 10.0, ): """ Instantiates this client. :param redvox_config: The Redvox endpoint configuration. :param refresh_token_interval: An optional interval in seconds that the auth token should be refreshed. :param timeout: An optional timeout """ if redvox_config is None: raise cloud_errors.CloudApiError( "A RedVoxConfig was not found in the environment and one wasn't provided" ) if refresh_token_interval <= 0: raise cloud_errors.CloudApiError( "refresh_token_interval must be strictly > 0" ) if timeout is not None and (timeout <= 0): raise cloud_errors.CloudApiError("timeout must be strictly > 0") self.redvox_config: RedVoxConfig = redvox_config self.refresh_token_interval: float = refresh_token_interval self.timeout: Optional[float] = timeout self.__session = ( requests.Session() ) # This must be initialized before the auth req! self.__refresh_timer = None self.__authenticate() self.__refresh_timer = threading.Timer( self.refresh_token_interval, self.__refresh_token ) self.__refresh_timer.start() def __authenticate(self) -> None: # Token provided if self.redvox_config.username == "auth_token": self.auth_token = self.redvox_config.password # Username/Password provided else: try: auth_resp: auth_api.AuthResp = self.authenticate_user( self.redvox_config.username, self.redvox_config.password ) except cloud_errors.ApiConnectionError as ex: self.close() raise ex if ( auth_resp.status != 200 or auth_resp.auth_token is None or len(auth_resp.auth_token) == 0 ): self.close() raise cloud_errors.AuthenticationError() self.auth_token: str = auth_resp.auth_token def __refresh_token(self): """ Automatically refreshes this client's auth token in the background once per minute. """ # pylint: disable=W0702 # noinspection PyBroadException try: self.auth_token = self.refresh_own_auth_token().auth_token self.__refresh_timer = threading.Timer( self.refresh_token_interval, self.__refresh_token ) self.__refresh_timer.start() except: self.close() def close(self): """ Terminates this client process by cancelling the refresh token timer. """ # pylint: disable=W0702 # noinspection PyBroadException try: if self.__refresh_timer is not None: self.__refresh_timer.cancel() except: pass # pylint: disable=W0702 # noinspection PyBroadException try: if self.__session is not None: self.__session.close() except: pass def health_check(self) -> bool: """ An API call that returns True if the API Cloud server is up and running or False otherwise. :return: True if the API Cloud server is up and running or False otherwise. """ return api.health_check( self.redvox_config, session=self.__session, timeout=self.timeout ) def authenticate_user(self, username: str, password: str) -> auth_api.AuthResp: """ Attempts to authenticate the given RedVox user. :param username: The RedVox username. :param password: The RedVox password. :return: An authenticate response. """ if len(username) == 0: raise cloud_errors.CloudApiError("Username must be provided") if len(password) == 0: raise cloud_errors.CloudApiError("Password must be provided") auth_req: auth_api.AuthReq = auth_api.AuthReq(username, password) return auth_api.authenticate_user( self.redvox_config, auth_req, session=self.__session, timeout=self.timeout ) def validate_auth_token( self, auth_token: str ) -> Optional[auth_api.ValidateTokenResp]: """ Validates the provided authentication token with the cloud API. :param auth_token: Authentication token to validate. :return: An authentication response with token details or None if token in invalid """ if len(auth_token) == 0: raise cloud_errors.CloudApiError("auth_token must be provided") token_req: auth_api.ValidateTokenReq = auth_api.ValidateTokenReq(auth_token) return auth_api.validate_token( self.redvox_config, token_req, session=self.__session, timeout=self.timeout ) def validate_own_auth_token(self) -> Optional[auth_api.ValidateTokenResp]: """ Validates the current token held by this client. :return: An authentication response with token details or None if token in invalid """ return self.validate_auth_token(self.auth_token) def refresh_auth_token( self, auth_token: str ) -> Optional[auth_api.RefreshTokenResp]: """ Retrieves a new authentication token from a given valid authentication token. :param auth_token: The authentication token to verify. :return: A new authentication token or None if the provide auth token is not valid. """ if len(auth_token) == 0: raise cloud_errors.CloudApiError("auth_token must be provided") refresh_token_req: auth_api.RefreshTokenReq = auth_api.RefreshTokenReq( auth_token ) return auth_api.refresh_token( self.redvox_config, refresh_token_req, session=self.__session, timeout=self.timeout, ) def refresh_own_auth_token(self) -> Optional[auth_api.RefreshTokenResp]: """ Retrieves a new authentication token using the token held by this client as a reference. :return: A new authentication token or None if the provide auth token is not valid. """ return self.refresh_auth_token(self.auth_token) def request_metadata( self, start_ts_s: int, end_ts_s: int, station_ids: List[str], metadata_to_include: List[str], chunk_by_seconds: int = constants.SECONDS_PER_DAY, ) -> Optional[metadata_api.MetadataResp]: """ Requests RedVox packet metadata. :param start_ts_s: Start epoch of request window. :param end_ts_s: End epoch of request window. :param station_ids: A list of station ids. :param metadata_to_include: A list of metadata fields to include (see: redvox.cloud.metadata.AvailableMetadata) :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d) :return: A metadata result containing the requested metadata or None on error. """ if end_ts_s <= start_ts_s: raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s") if len(station_ids) == 0: raise cloud_errors.CloudApiError("At least one station_id must be included") if len(metadata_to_include) == 0: raise cloud_errors.CloudApiError( "At least one metadata field must be included" ) if chunk_by_seconds <= 0: raise cloud_errors.CloudApiError("chunk_by_seconds must be > 0") time_chunks: List[Tuple[int, int]] = chunk_time_range( start_ts_s, end_ts_s, chunk_by_seconds ) metadata_resp: metadata_api.MetadataResp = metadata_api.MetadataResp([]) for start_ts, end_ts in time_chunks: metadata_req: metadata_api.MetadataReq = metadata_api.MetadataReq( self.auth_token, start_ts, end_ts, station_ids, metadata_to_include, self.redvox_config.secret_token, ) chunked_resp: Optional[ metadata_api.MetadataResp ] = metadata_api.request_metadata( self.redvox_config, metadata_req, session=self.__session, timeout=self.timeout, ) if chunked_resp: metadata_resp.metadata.extend(chunked_resp.metadata) return metadata_resp def request_metadata_m( self, start_ts_s: int, end_ts_s: int, station_ids: List[str], metadata_to_include: List[str], chunk_by_seconds: int = constants.SECONDS_PER_HOUR, ) -> Optional[metadata_api.MetadataRespM]: """ Requests RedVox packet metadata. :param start_ts_s: Start epoch of request window. :param end_ts_s: End epoch of request window. :param station_ids: A list of station ids. :param metadata_to_include: A list of metadata fields to include (see: redvox.cloud.metadata.AvailableMetadata) :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d) :return: A metadata result containing the requested metadata or None on error. """ metadata_resp: metadata_api.MetadataRespM = metadata_api.MetadataRespM([]) resp: Optional[metadata_api.MetadataRespM] for resp in self.request_metadata_m_stream( start_ts_s, end_ts_s, station_ids, metadata_to_include, chunk_by_seconds ): if resp is not None: metadata_resp.db_packets.extend(resp.db_packets) return metadata_resp def request_metadata_m_stream( self, start_ts_s: int, end_ts_s: int, station_ids: List[str], metadata_to_include: List[str], chunk_by_seconds: int = constants.SECONDS_PER_HOUR, ) -> Iterator[Optional[metadata_api.MetadataRespM]]: """ Requests RedVox packet metadata. :param start_ts_s: Start epoch of request window. :param end_ts_s: End epoch of request window. :param station_ids: A list of station ids. :param metadata_to_include: A list of metadata fields to include (see: redvox.cloud.metadata.AvailableMetadata) :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d) :return: A metadata result containing the requested metadata or None on error. """ if end_ts_s <= start_ts_s: raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s") if len(station_ids) == 0: raise cloud_errors.CloudApiError("At least one station_id must be included") if len(metadata_to_include) == 0: raise cloud_errors.CloudApiError( "At least one metadata field must be included" ) if chunk_by_seconds <= 0: raise cloud_errors.CloudApiError("chunk_by_seconds must be > 0") time_chunks: List[Tuple[int, int]] = chunk_time_range( start_ts_s, end_ts_s, chunk_by_seconds ) for start_ts, end_ts in time_chunks: metadata_req: metadata_api.MetadataReq = metadata_api.MetadataReq( self.auth_token, start_ts, end_ts, station_ids, metadata_to_include, self.redvox_config.secret_token, ) chunked_resp: Optional[ metadata_api.MetadataRespM ] = metadata_api.request_metadata_m( self.redvox_config, metadata_req, session=self.__session, timeout=self.timeout, ) yield chunked_resp def request_geo_metadata_stream( self, start_ts_s: int, end_ts_s: int, bounding_box: Optional[metadata_api.BoundingBox], bounding_circle: Optional[metadata_api.BoundingCircle], metadata_to_include: List[str], chunk_by_seconds: int = constants.SECONDS_PER_HOUR, ) -> Iterator[Optional[metadata_api.GeoMetadataResp]]: """ Requests RedVox packet metadata from geo bounds. :param start_ts_s: Start epoch of request window. :param end_ts_s: End epoch of request window. :param bounding_box: A bounding box given by the SW and NE coordinates. Mutually exclusive with bounding_circle. :param bounding_circle: A bounding circle given by a center point and radius in meter. Mutually exclusive with bounding_box. :param metadata_to_include: A list of metadata fields to include (see: redvox.cloud.metadata.AvailableMetadata) :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d) :return: A geo metadata result containing the requested metadata or None on error. """ if end_ts_s <= start_ts_s: raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s") if bounding_box is None and bounding_circle is None: raise cloud_errors.CloudApiError( "Exactly one of bounding_box or bounding_circle must be set" ) if bounding_box is not None and bounding_circle is not None: raise cloud_errors.CloudApiError( "Exactly one of bounding_box or bounding_circle must be set" ) if len(metadata_to_include) == 0: raise cloud_errors.CloudApiError( "At least one metadata field must be included" ) if chunk_by_seconds <= 0: raise cloud_errors.CloudApiError("chunk_by_seconds must be > 0") time_chunks: List[Tuple[int, int]] = chunk_time_range( start_ts_s, end_ts_s, chunk_by_seconds ) for start_ts, end_ts in time_chunks: geo_metadata_req: metadata_api.GeoMetadataReq = metadata_api.GeoMetadataReq( self.auth_token, start_ts, end_ts, bounding_box, bounding_circle, metadata_to_include, ) chunked_resp: Optional[ metadata_api.GeoMetadataResp ] = metadata_api.request_geo_metadata( self.redvox_config, geo_metadata_req, session=self.__session, timeout=self.timeout, ) yield chunked_resp def request_timing_metadata( self, start_ts_s: int, end_ts_s: int, station_ids: List[str], chunk_by_seconds: int = constants.SECONDS_PER_DAY, ) -> metadata_api.TimingMetaResponse: """ Requests timing metadata from RedVox packets. :param start_ts_s: Start epoch of the request. :param end_ts_s: End epoch of the request. :param station_ids: A list of station ids. :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d) :return: A response containing the requested metadata. """ if end_ts_s <= start_ts_s: raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s") if len(station_ids) == 0: raise cloud_errors.CloudApiError("At least one station_id must be included") if chunk_by_seconds <= 0: raise cloud_errors.CloudApiError("chunk_by_seconds must be > 0") time_chunks: List[Tuple[int, int]] = chunk_time_range( start_ts_s, end_ts_s, chunk_by_seconds ) metadata_resp: metadata_api.TimingMetaResponse = ( metadata_api.TimingMetaResponse([]) ) for start_ts, end_ts in time_chunks: timing_req: metadata_api.TimingMetaRequest = metadata_api.TimingMetaRequest( self.auth_token, start_ts, end_ts, station_ids, self.redvox_config.secret_token, ) chunked_resp: metadata_api.TimingMetaResponse = ( metadata_api.request_timing_metadata( self.redvox_config, timing_req, session=self.__session, timeout=self.timeout, ) ) if chunked_resp: metadata_resp.items.extend(chunked_resp.items) return metadata_resp def request_report_data(self, report_id: str) -> Optional[data_api.ReportDataResp]: """ Requests a signed URL for a given report ID. :param report_id: The report ID to request data for. :return: A response containing a signed URL of the report data. """ if len(report_id) == 0: raise cloud_errors.CloudApiError("report_id must be included") report_data_req: data_api.ReportDataReq = data_api.ReportDataReq( self.auth_token, report_id, self.redvox_config.secret_token ) return data_api.request_report_data( self.redvox_config, report_data_req, session=self.__session, timeout=self.timeout, ) def request_station_statuses( self, start_ts_s: int, end_ts_s: int, station_ids: List[str], ) -> Optional[metadata_api.StationStatusResp]: """ Requests station timing information from the cloud services. :param start_ts_s: The start of the request data window. :param end_ts_s: The end of the request data window. :param station_ids: A list of station IDs. :return: A StationStatsResp. """ if end_ts_s <= start_ts_s: raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s") station_status_req: metadata_api.StationStatusReq = ( metadata_api.StationStatusReq( self.redvox_config.secret_token, self.auth_token, start_ts_s, end_ts_s, station_ids, ) ) return metadata_api.request_station_statuses( self.redvox_config, station_status_req, session=self.__session, timeout=self.timeout, ) def request_data_range( self, start_ts_s: int, end_ts_s: int, station_ids: List[str], req_type: data_api.DataRangeReqType = data_api.DataRangeReqType.API_900_1000, correct_query_timing: bool = True, out_queue: Optional[Queue] = None, ) -> data_api.DataRangeResp: """ Request signed URLs for RedVox packets. :param start_ts_s: The start epoch of the window. :param end_ts_s: The end epoch of the window. :param station_ids: A list of station ids. :param req_type: The type of data to request. :param correct_query_timing: If set to true, timing correction will be applied to each station before the data is correct_timing queried. :return: A response containing a list of signed URLs for the RedVox packets. """ def _make_req( _start_ts_s: int, _end_ts_s: int, _station_ids: List[str] ) -> data_api.DataRangeResp: """ Makes the actual data request after timing correction was or was not applied. :param _start_ts_s: The start epoch of the window. :param _end_ts_s: The end epoch of the window. :param _station_ids: A list of station IDs. :return: A response containing a list of signed URLs for the RedVox packets. """ data_range_req: data_api.DataRangeReq = data_api.DataRangeReq( self.auth_token, _start_ts_s, _end_ts_s, _station_ids, self.redvox_config.secret_token, ) req_resp: data_api.DataRangeResp = data_api.request_range_data( self.redvox_config, data_range_req, session=self.__session, timeout=self.timeout, req_type=req_type, ) return req_resp if end_ts_s <= start_ts_s: raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s") if len(station_ids) == 0: raise cloud_errors.CloudApiError("At least one station_id must be provided") # If timing correction was requested, we want to find the corrected start and end offsets in order to # request the correct range of data for each station's particular clock offset if correct_query_timing: corrected_queries: Optional[ List["CorrectedQuery"] ] = do_correct_query_timing(self, start_ts_s, end_ts_s, station_ids) # If we get nothing back, we can't apply a correction if corrected_queries is None or len(corrected_queries) == 0: print("No timing corrections returned, running original query.") return _make_req(start_ts_s, end_ts_s, station_ids) # Make a request for each corrected query, aggregating the results of each request resp: data_api.DataRangeResp = data_api.DataRangeResp([]) corrected_query: "CorrectedQuery" for corrected_query in corrected_queries: correction_msg: str = ( f"Running timing corrected query for {corrected_query.station_id} " f"start offset={corrected_query.start_offset()} " f"end offset={corrected_query.end_offset()}" ) if out_queue is None: print(correction_msg) else: out_queue.put(correction_msg, block=False) resp.append( _make_req( round(corrected_query.corrected_start_ts), round(corrected_query.corrected_end_ts), [corrected_query.station_id], ) ) return resp else: # No timing correction requested, go ahead just make the original uncorrected request return _make_req(start_ts_s, end_ts_s, station_ids) def request_station_stats( self, start_ts_s: int, end_ts_s: int, station_ids: List[str], ) -> Optional[station_stats_api.StationStatsResp]: """ Request signed URLs for RedVox packets. :param start_ts_s: The start epoch of the window. :param end_ts_s: The end epoch of the window. :param station_ids: A list of station ids. :return: A response containing a list of signed URLs for the RedVox packets. """ if end_ts_s <= start_ts_s: raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s") if len(station_ids) == 0: raise cloud_errors.CloudApiError("At least one station_id must be provided") station_stats_req: station_stats_api.StationStatReq = ( station_stats_api.StationStatReq( self.auth_token, start_ts_s, end_ts_s, station_ids, self.redvox_config.secret_token, ) ) return station_stats_api.request_station_stats( self.redvox_config, station_stats_req, self.__session, self.timeout ) def request_session_model( self, session_key: str ) -> session_model_api.SessionModelResp: """ Requests a session model. :param session_key: The session key. :return: An instance of a SessionModelResp. """ req: session_model_api.SessionModelReq = session_model_api.SessionModelReq( self.auth_token, session_key ) return session_model_api.request_session( self.redvox_config, req, self.__session, self.timeout ) def request_session_models( self, id_uuids: Optional[List[str]] = None, owner: Optional[str] = None, start_ts: Optional[int] = None, end_ts: Optional[int] = None, include_public: bool = False, ) -> session_model_api.SessionModelsResp: """ Requests a range of session models. :param id_uuids: An optional list of IDs or ID:UUIDs. :param owner: An optional owner. :param start_ts: An optional start timestamp (microseconds). :param end_ts: An optional end timestamp (microseconds). :param include_public: Additionally include public sessions that may not be the same as the owner. :return: An instance of SessionModelsResp. """ req: session_model_api.SessionModelsReq = session_model_api.SessionModelsReq( self.auth_token, id_uuids, owner, start_ts, end_ts, include_public ) return session_model_api.request_sessions( self.redvox_config, req, self.__session, self.timeout ) def request_dynamic_session_model( self, session_key: str, start_ts: int, end_ts: int, ) -> session_model_api.DynamicSessionModelResp: """ Requests a dynamic session given the fields that make up its key. :param session_key: The parent session model key. :param start_ts: The start of the dynamic session (microseconds). :param end_ts: The end of the dynamic session (microseconds). :return: An instance of DynamicSessionModelResp. """ req: session_model_api.DynamicSessionModelReq = ( session_model_api.DynamicSessionModelReq( self.auth_token, session_key, start_ts, end_ts ) ) return session_model_api.request_dynamic_session( self.redvox_config, req, self.__session, self.timeout )Methods- def authenticate_user(self, username: str, password: str) ‑> AuthResp
- 
Attempts to authenticate the given RedVox user. :param username: The RedVox username. :param password: The RedVox password. :return: An authenticate response. Expand source codedef authenticate_user(self, username: str, password: str) -> auth_api.AuthResp: """ Attempts to authenticate the given RedVox user. :param username: The RedVox username. :param password: The RedVox password. :return: An authenticate response. """ if len(username) == 0: raise cloud_errors.CloudApiError("Username must be provided") if len(password) == 0: raise cloud_errors.CloudApiError("Password must be provided") auth_req: auth_api.AuthReq = auth_api.AuthReq(username, password) return auth_api.authenticate_user( self.redvox_config, auth_req, session=self.__session, timeout=self.timeout )
- def close(self)
- 
Terminates this client process by cancelling the refresh token timer. Expand source codedef close(self): """ Terminates this client process by cancelling the refresh token timer. """ # pylint: disable=W0702 # noinspection PyBroadException try: if self.__refresh_timer is not None: self.__refresh_timer.cancel() except: pass # pylint: disable=W0702 # noinspection PyBroadException try: if self.__session is not None: self.__session.close() except: pass
- def health_check(self) ‑> bool
- 
An API call that returns True if the API Cloud server is up and running or False otherwise. :return: True if the API Cloud server is up and running or False otherwise. Expand source codedef health_check(self) -> bool: """ An API call that returns True if the API Cloud server is up and running or False otherwise. :return: True if the API Cloud server is up and running or False otherwise. """ return api.health_check( self.redvox_config, session=self.__session, timeout=self.timeout )
- def refresh_auth_token(self, auth_token: str) ‑> Optional[RefreshTokenResp]
- 
Retrieves a new authentication token from a given valid authentication token. :param auth_token: The authentication token to verify. :return: A new authentication token or None if the provide auth token is not valid. Expand source codedef refresh_auth_token( self, auth_token: str ) -> Optional[auth_api.RefreshTokenResp]: """ Retrieves a new authentication token from a given valid authentication token. :param auth_token: The authentication token to verify. :return: A new authentication token or None if the provide auth token is not valid. """ if len(auth_token) == 0: raise cloud_errors.CloudApiError("auth_token must be provided") refresh_token_req: auth_api.RefreshTokenReq = auth_api.RefreshTokenReq( auth_token ) return auth_api.refresh_token( self.redvox_config, refresh_token_req, session=self.__session, timeout=self.timeout, )
- def refresh_own_auth_token(self) ‑> Optional[RefreshTokenResp]
- 
Retrieves a new authentication token using the token held by this client as a reference. :return: A new authentication token or None if the provide auth token is not valid. Expand source codedef refresh_own_auth_token(self) -> Optional[auth_api.RefreshTokenResp]: """ Retrieves a new authentication token using the token held by this client as a reference. :return: A new authentication token or None if the provide auth token is not valid. """ return self.refresh_auth_token(self.auth_token)
- def request_data_range(self, start_ts_s: int, end_ts_s: int, station_ids: List[str], req_type: DataRangeReqType = DataRangeReqType.API_900_1000, correct_query_timing: bool = True, out_queue: Optional[- >] = None) ‑> DataRangeResp 
- 
Request signed URLs for RedVox packets. :param start_ts_s: The start epoch of the window. :param end_ts_s: The end epoch of the window. :param station_ids: A list of station ids. :param req_type: The type of data to request. :param correct_query_timing: If set to true, timing correction will be applied to each station before the data is correct_timing queried. :return: A response containing a list of signed URLs for the RedVox packets. Expand source codedef request_data_range( self, start_ts_s: int, end_ts_s: int, station_ids: List[str], req_type: data_api.DataRangeReqType = data_api.DataRangeReqType.API_900_1000, correct_query_timing: bool = True, out_queue: Optional[Queue] = None, ) -> data_api.DataRangeResp: """ Request signed URLs for RedVox packets. :param start_ts_s: The start epoch of the window. :param end_ts_s: The end epoch of the window. :param station_ids: A list of station ids. :param req_type: The type of data to request. :param correct_query_timing: If set to true, timing correction will be applied to each station before the data is correct_timing queried. :return: A response containing a list of signed URLs for the RedVox packets. """ def _make_req( _start_ts_s: int, _end_ts_s: int, _station_ids: List[str] ) -> data_api.DataRangeResp: """ Makes the actual data request after timing correction was or was not applied. :param _start_ts_s: The start epoch of the window. :param _end_ts_s: The end epoch of the window. :param _station_ids: A list of station IDs. :return: A response containing a list of signed URLs for the RedVox packets. """ data_range_req: data_api.DataRangeReq = data_api.DataRangeReq( self.auth_token, _start_ts_s, _end_ts_s, _station_ids, self.redvox_config.secret_token, ) req_resp: data_api.DataRangeResp = data_api.request_range_data( self.redvox_config, data_range_req, session=self.__session, timeout=self.timeout, req_type=req_type, ) return req_resp if end_ts_s <= start_ts_s: raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s") if len(station_ids) == 0: raise cloud_errors.CloudApiError("At least one station_id must be provided") # If timing correction was requested, we want to find the corrected start and end offsets in order to # request the correct range of data for each station's particular clock offset if correct_query_timing: corrected_queries: Optional[ List["CorrectedQuery"] ] = do_correct_query_timing(self, start_ts_s, end_ts_s, station_ids) # If we get nothing back, we can't apply a correction if corrected_queries is None or len(corrected_queries) == 0: print("No timing corrections returned, running original query.") return _make_req(start_ts_s, end_ts_s, station_ids) # Make a request for each corrected query, aggregating the results of each request resp: data_api.DataRangeResp = data_api.DataRangeResp([]) corrected_query: "CorrectedQuery" for corrected_query in corrected_queries: correction_msg: str = ( f"Running timing corrected query for {corrected_query.station_id} " f"start offset={corrected_query.start_offset()} " f"end offset={corrected_query.end_offset()}" ) if out_queue is None: print(correction_msg) else: out_queue.put(correction_msg, block=False) resp.append( _make_req( round(corrected_query.corrected_start_ts), round(corrected_query.corrected_end_ts), [corrected_query.station_id], ) ) return resp else: # No timing correction requested, go ahead just make the original uncorrected request return _make_req(start_ts_s, end_ts_s, station_ids)
- def request_dynamic_session_model(self, session_key: str, start_ts: int, end_ts: int) ‑> DynamicSessionModelResp
- 
Requests a dynamic session given the fields that make up its key. :param session_key: The parent session model key. :param start_ts: The start of the dynamic session (microseconds). :param end_ts: The end of the dynamic session (microseconds). :return: An instance of DynamicSessionModelResp. Expand source codedef request_dynamic_session_model( self, session_key: str, start_ts: int, end_ts: int, ) -> session_model_api.DynamicSessionModelResp: """ Requests a dynamic session given the fields that make up its key. :param session_key: The parent session model key. :param start_ts: The start of the dynamic session (microseconds). :param end_ts: The end of the dynamic session (microseconds). :return: An instance of DynamicSessionModelResp. """ req: session_model_api.DynamicSessionModelReq = ( session_model_api.DynamicSessionModelReq( self.auth_token, session_key, start_ts, end_ts ) ) return session_model_api.request_dynamic_session( self.redvox_config, req, self.__session, self.timeout )
- def request_geo_metadata_stream(self, start_ts_s: int, end_ts_s: int, bounding_box: Optional[BoundingBox], bounding_circle: Optional[BoundingCircle], metadata_to_include: List[str], chunk_by_seconds: int = 3600) ‑> Iterator[Optional[GeoMetadataResp]]
- 
Requests RedVox packet metadata from geo bounds. :param start_ts_s: Start epoch of request window. :param end_ts_s: End epoch of request window. :param bounding_box: A bounding box given by the SW and NE coordinates. Mutually exclusive with bounding_circle. :param bounding_circle: A bounding circle given by a center point and radius in meter. Mutually exclusive with bounding_box. :param metadata_to_include: A list of metadata fields to include (see: redvox.cloud.metadata.AvailableMetadata) :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d) :return: A geo metadata result containing the requested metadata or None on error. Expand source codedef request_geo_metadata_stream( self, start_ts_s: int, end_ts_s: int, bounding_box: Optional[metadata_api.BoundingBox], bounding_circle: Optional[metadata_api.BoundingCircle], metadata_to_include: List[str], chunk_by_seconds: int = constants.SECONDS_PER_HOUR, ) -> Iterator[Optional[metadata_api.GeoMetadataResp]]: """ Requests RedVox packet metadata from geo bounds. :param start_ts_s: Start epoch of request window. :param end_ts_s: End epoch of request window. :param bounding_box: A bounding box given by the SW and NE coordinates. Mutually exclusive with bounding_circle. :param bounding_circle: A bounding circle given by a center point and radius in meter. Mutually exclusive with bounding_box. :param metadata_to_include: A list of metadata fields to include (see: redvox.cloud.metadata.AvailableMetadata) :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d) :return: A geo metadata result containing the requested metadata or None on error. """ if end_ts_s <= start_ts_s: raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s") if bounding_box is None and bounding_circle is None: raise cloud_errors.CloudApiError( "Exactly one of bounding_box or bounding_circle must be set" ) if bounding_box is not None and bounding_circle is not None: raise cloud_errors.CloudApiError( "Exactly one of bounding_box or bounding_circle must be set" ) if len(metadata_to_include) == 0: raise cloud_errors.CloudApiError( "At least one metadata field must be included" ) if chunk_by_seconds <= 0: raise cloud_errors.CloudApiError("chunk_by_seconds must be > 0") time_chunks: List[Tuple[int, int]] = chunk_time_range( start_ts_s, end_ts_s, chunk_by_seconds ) for start_ts, end_ts in time_chunks: geo_metadata_req: metadata_api.GeoMetadataReq = metadata_api.GeoMetadataReq( self.auth_token, start_ts, end_ts, bounding_box, bounding_circle, metadata_to_include, ) chunked_resp: Optional[ metadata_api.GeoMetadataResp ] = metadata_api.request_geo_metadata( self.redvox_config, geo_metadata_req, session=self.__session, timeout=self.timeout, ) yield chunked_resp
- def request_metadata(self, start_ts_s: int, end_ts_s: int, station_ids: List[str], metadata_to_include: List[str], chunk_by_seconds: int = 86400) ‑> Optional[MetadataResp]
- 
Requests RedVox packet metadata. :param start_ts_s: Start epoch of request window. :param end_ts_s: End epoch of request window. :param station_ids: A list of station ids. :param metadata_to_include: A list of metadata fields to include (see: redvox.cloud.metadata.AvailableMetadata) :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d) :return: A metadata result containing the requested metadata or None on error. Expand source codedef request_metadata( self, start_ts_s: int, end_ts_s: int, station_ids: List[str], metadata_to_include: List[str], chunk_by_seconds: int = constants.SECONDS_PER_DAY, ) -> Optional[metadata_api.MetadataResp]: """ Requests RedVox packet metadata. :param start_ts_s: Start epoch of request window. :param end_ts_s: End epoch of request window. :param station_ids: A list of station ids. :param metadata_to_include: A list of metadata fields to include (see: redvox.cloud.metadata.AvailableMetadata) :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d) :return: A metadata result containing the requested metadata or None on error. """ if end_ts_s <= start_ts_s: raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s") if len(station_ids) == 0: raise cloud_errors.CloudApiError("At least one station_id must be included") if len(metadata_to_include) == 0: raise cloud_errors.CloudApiError( "At least one metadata field must be included" ) if chunk_by_seconds <= 0: raise cloud_errors.CloudApiError("chunk_by_seconds must be > 0") time_chunks: List[Tuple[int, int]] = chunk_time_range( start_ts_s, end_ts_s, chunk_by_seconds ) metadata_resp: metadata_api.MetadataResp = metadata_api.MetadataResp([]) for start_ts, end_ts in time_chunks: metadata_req: metadata_api.MetadataReq = metadata_api.MetadataReq( self.auth_token, start_ts, end_ts, station_ids, metadata_to_include, self.redvox_config.secret_token, ) chunked_resp: Optional[ metadata_api.MetadataResp ] = metadata_api.request_metadata( self.redvox_config, metadata_req, session=self.__session, timeout=self.timeout, ) if chunked_resp: metadata_resp.metadata.extend(chunked_resp.metadata) return metadata_resp
- def request_metadata_m(self, start_ts_s: int, end_ts_s: int, station_ids: List[str], metadata_to_include: List[str], chunk_by_seconds: int = 3600) ‑> Optional[MetadataRespM]
- 
Requests RedVox packet metadata. :param start_ts_s: Start epoch of request window. :param end_ts_s: End epoch of request window. :param station_ids: A list of station ids. :param metadata_to_include: A list of metadata fields to include (see: redvox.cloud.metadata.AvailableMetadata) :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d) :return: A metadata result containing the requested metadata or None on error. Expand source codedef request_metadata_m( self, start_ts_s: int, end_ts_s: int, station_ids: List[str], metadata_to_include: List[str], chunk_by_seconds: int = constants.SECONDS_PER_HOUR, ) -> Optional[metadata_api.MetadataRespM]: """ Requests RedVox packet metadata. :param start_ts_s: Start epoch of request window. :param end_ts_s: End epoch of request window. :param station_ids: A list of station ids. :param metadata_to_include: A list of metadata fields to include (see: redvox.cloud.metadata.AvailableMetadata) :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d) :return: A metadata result containing the requested metadata or None on error. """ metadata_resp: metadata_api.MetadataRespM = metadata_api.MetadataRespM([]) resp: Optional[metadata_api.MetadataRespM] for resp in self.request_metadata_m_stream( start_ts_s, end_ts_s, station_ids, metadata_to_include, chunk_by_seconds ): if resp is not None: metadata_resp.db_packets.extend(resp.db_packets) return metadata_resp
- def request_metadata_m_stream(self, start_ts_s: int, end_ts_s: int, station_ids: List[str], metadata_to_include: List[str], chunk_by_seconds: int = 3600) ‑> Iterator[Optional[MetadataRespM]]
- 
Requests RedVox packet metadata. :param start_ts_s: Start epoch of request window. :param end_ts_s: End epoch of request window. :param station_ids: A list of station ids. :param metadata_to_include: A list of metadata fields to include (see: redvox.cloud.metadata.AvailableMetadata) :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d) :return: A metadata result containing the requested metadata or None on error. Expand source codedef request_metadata_m_stream( self, start_ts_s: int, end_ts_s: int, station_ids: List[str], metadata_to_include: List[str], chunk_by_seconds: int = constants.SECONDS_PER_HOUR, ) -> Iterator[Optional[metadata_api.MetadataRespM]]: """ Requests RedVox packet metadata. :param start_ts_s: Start epoch of request window. :param end_ts_s: End epoch of request window. :param station_ids: A list of station ids. :param metadata_to_include: A list of metadata fields to include (see: redvox.cloud.metadata.AvailableMetadata) :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d) :return: A metadata result containing the requested metadata or None on error. """ if end_ts_s <= start_ts_s: raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s") if len(station_ids) == 0: raise cloud_errors.CloudApiError("At least one station_id must be included") if len(metadata_to_include) == 0: raise cloud_errors.CloudApiError( "At least one metadata field must be included" ) if chunk_by_seconds <= 0: raise cloud_errors.CloudApiError("chunk_by_seconds must be > 0") time_chunks: List[Tuple[int, int]] = chunk_time_range( start_ts_s, end_ts_s, chunk_by_seconds ) for start_ts, end_ts in time_chunks: metadata_req: metadata_api.MetadataReq = metadata_api.MetadataReq( self.auth_token, start_ts, end_ts, station_ids, metadata_to_include, self.redvox_config.secret_token, ) chunked_resp: Optional[ metadata_api.MetadataRespM ] = metadata_api.request_metadata_m( self.redvox_config, metadata_req, session=self.__session, timeout=self.timeout, ) yield chunked_resp
- def request_report_data(self, report_id: str) ‑> Optional[ReportDataResp]
- 
Requests a signed URL for a given report ID. :param report_id: The report ID to request data for. :return: A response containing a signed URL of the report data. Expand source codedef request_report_data(self, report_id: str) -> Optional[data_api.ReportDataResp]: """ Requests a signed URL for a given report ID. :param report_id: The report ID to request data for. :return: A response containing a signed URL of the report data. """ if len(report_id) == 0: raise cloud_errors.CloudApiError("report_id must be included") report_data_req: data_api.ReportDataReq = data_api.ReportDataReq( self.auth_token, report_id, self.redvox_config.secret_token ) return data_api.request_report_data( self.redvox_config, report_data_req, session=self.__session, timeout=self.timeout, )
- def request_session_model(self, session_key: str) ‑> SessionModelResp
- 
Requests a session model. :param session_key: The session key. :return: An instance of a SessionModelResp. Expand source codedef request_session_model( self, session_key: str ) -> session_model_api.SessionModelResp: """ Requests a session model. :param session_key: The session key. :return: An instance of a SessionModelResp. """ req: session_model_api.SessionModelReq = session_model_api.SessionModelReq( self.auth_token, session_key ) return session_model_api.request_session( self.redvox_config, req, self.__session, self.timeout )
- def request_session_models(self, id_uuids: Optional[List[str]] = None, owner: Optional[str] = None, start_ts: Optional[int] = None, end_ts: Optional[int] = None, include_public: bool = False) ‑> SessionModelsResp
- 
Requests a range of session models. :param id_uuids: An optional list of IDs or ID:UUIDs. :param owner: An optional owner. :param start_ts: An optional start timestamp (microseconds). :param end_ts: An optional end timestamp (microseconds). :param include_public: Additionally include public sessions that may not be the same as the owner. :return: An instance of SessionModelsResp. Expand source codedef request_session_models( self, id_uuids: Optional[List[str]] = None, owner: Optional[str] = None, start_ts: Optional[int] = None, end_ts: Optional[int] = None, include_public: bool = False, ) -> session_model_api.SessionModelsResp: """ Requests a range of session models. :param id_uuids: An optional list of IDs or ID:UUIDs. :param owner: An optional owner. :param start_ts: An optional start timestamp (microseconds). :param end_ts: An optional end timestamp (microseconds). :param include_public: Additionally include public sessions that may not be the same as the owner. :return: An instance of SessionModelsResp. """ req: session_model_api.SessionModelsReq = session_model_api.SessionModelsReq( self.auth_token, id_uuids, owner, start_ts, end_ts, include_public ) return session_model_api.request_sessions( self.redvox_config, req, self.__session, self.timeout )
- def request_station_stats(self, start_ts_s: int, end_ts_s: int, station_ids: List[str]) ‑> Optional[StationStatsResp]
- 
Request signed URLs for RedVox packets. :param start_ts_s: The start epoch of the window. :param end_ts_s: The end epoch of the window. :param station_ids: A list of station ids. :return: A response containing a list of signed URLs for the RedVox packets. Expand source codedef request_station_stats( self, start_ts_s: int, end_ts_s: int, station_ids: List[str], ) -> Optional[station_stats_api.StationStatsResp]: """ Request signed URLs for RedVox packets. :param start_ts_s: The start epoch of the window. :param end_ts_s: The end epoch of the window. :param station_ids: A list of station ids. :return: A response containing a list of signed URLs for the RedVox packets. """ if end_ts_s <= start_ts_s: raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s") if len(station_ids) == 0: raise cloud_errors.CloudApiError("At least one station_id must be provided") station_stats_req: station_stats_api.StationStatReq = ( station_stats_api.StationStatReq( self.auth_token, start_ts_s, end_ts_s, station_ids, self.redvox_config.secret_token, ) ) return station_stats_api.request_station_stats( self.redvox_config, station_stats_req, self.__session, self.timeout )
- def request_station_statuses(self, start_ts_s: int, end_ts_s: int, station_ids: List[str]) ‑> Optional[StationStatusResp]
- 
Requests station timing information from the cloud services. :param start_ts_s: The start of the request data window. :param end_ts_s: The end of the request data window. :param station_ids: A list of station IDs. :return: A StationStatsResp. Expand source codedef request_station_statuses( self, start_ts_s: int, end_ts_s: int, station_ids: List[str], ) -> Optional[metadata_api.StationStatusResp]: """ Requests station timing information from the cloud services. :param start_ts_s: The start of the request data window. :param end_ts_s: The end of the request data window. :param station_ids: A list of station IDs. :return: A StationStatsResp. """ if end_ts_s <= start_ts_s: raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s") station_status_req: metadata_api.StationStatusReq = ( metadata_api.StationStatusReq( self.redvox_config.secret_token, self.auth_token, start_ts_s, end_ts_s, station_ids, ) ) return metadata_api.request_station_statuses( self.redvox_config, station_status_req, session=self.__session, timeout=self.timeout, )
- def request_timing_metadata(self, start_ts_s: int, end_ts_s: int, station_ids: List[str], chunk_by_seconds: int = 86400) ‑> TimingMetaResponse
- 
Requests timing metadata from RedVox packets. :param start_ts_s: Start epoch of the request. :param end_ts_s: End epoch of the request. :param station_ids: A list of station ids. :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d) :return: A response containing the requested metadata. Expand source codedef request_timing_metadata( self, start_ts_s: int, end_ts_s: int, station_ids: List[str], chunk_by_seconds: int = constants.SECONDS_PER_DAY, ) -> metadata_api.TimingMetaResponse: """ Requests timing metadata from RedVox packets. :param start_ts_s: Start epoch of the request. :param end_ts_s: End epoch of the request. :param station_ids: A list of station ids. :param chunk_by_seconds: Split up longer requests into chunks of chunk_by_seconds size (default 86400s/1d) :return: A response containing the requested metadata. """ if end_ts_s <= start_ts_s: raise cloud_errors.CloudApiError("start_ts_s must be < end_ts_s") if len(station_ids) == 0: raise cloud_errors.CloudApiError("At least one station_id must be included") if chunk_by_seconds <= 0: raise cloud_errors.CloudApiError("chunk_by_seconds must be > 0") time_chunks: List[Tuple[int, int]] = chunk_time_range( start_ts_s, end_ts_s, chunk_by_seconds ) metadata_resp: metadata_api.TimingMetaResponse = ( metadata_api.TimingMetaResponse([]) ) for start_ts, end_ts in time_chunks: timing_req: metadata_api.TimingMetaRequest = metadata_api.TimingMetaRequest( self.auth_token, start_ts, end_ts, station_ids, self.redvox_config.secret_token, ) chunked_resp: metadata_api.TimingMetaResponse = ( metadata_api.request_timing_metadata( self.redvox_config, timing_req, session=self.__session, timeout=self.timeout, ) ) if chunked_resp: metadata_resp.items.extend(chunked_resp.items) return metadata_resp
- def validate_auth_token(self, auth_token: str) ‑> Optional[ValidateTokenResp]
- 
Validates the provided authentication token with the cloud API. :param auth_token: Authentication token to validate. :return: An authentication response with token details or None if token in invalid Expand source codedef validate_auth_token( self, auth_token: str ) -> Optional[auth_api.ValidateTokenResp]: """ Validates the provided authentication token with the cloud API. :param auth_token: Authentication token to validate. :return: An authentication response with token details or None if token in invalid """ if len(auth_token) == 0: raise cloud_errors.CloudApiError("auth_token must be provided") token_req: auth_api.ValidateTokenReq = auth_api.ValidateTokenReq(auth_token) return auth_api.validate_token( self.redvox_config, token_req, session=self.__session, timeout=self.timeout )
- def validate_own_auth_token(self) ‑> Optional[ValidateTokenResp]
- 
Validates the current token held by this client. :return: An authentication response with token details or None if token in invalid Expand source codedef validate_own_auth_token(self) -> Optional[auth_api.ValidateTokenResp]: """ Validates the current token held by this client. :return: An authentication response with token details or None if token in invalid """ return self.validate_auth_token(self.auth_token)