diff --git a/.github/workflows/pull-request.yaml b/.github/workflows/pull-request.yaml new file mode 100644 index 00000000..22665bb9 --- /dev/null +++ b/.github/workflows/pull-request.yaml @@ -0,0 +1,48 @@ +name: Pull Request +on: + pull_request: +concurrency: + group: ${{ github.head_ref }} + cancel-in-progress: true +jobs: + test: + runs-on: ubuntu-latest + strategy: + fail-fast: false + matrix: + python-version: + - '3.8' + - '3.9' + - '3.10' + - '3.11' + - '3.12' + steps: + - uses: actions/checkout@v4 + - uses: extractions/setup-just@v1 + - uses: actions/setup-python@v5 + with: + python-version: ${{ matrix.python-version }} + - run: just deps + - run: just test + lint: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + - uses: extractions/setup-just@v1 + - uses: actions/setup-python@v5 + - run: just deps + - run: just lint + cov: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + - uses: extractions/setup-just@v1 + - uses: actions/setup-python@v5 + - run: just deps + - run: just test + - run: just cov xml + - if: always() + uses: orgoro/coverage@v3.1 + with: + coverageFile: coverage.xml + token: ${{ secrets.GITHUB_TOKEN }} diff --git a/.github/workflows/push.yaml b/.github/workflows/push.yaml deleted file mode 100644 index d6f44510..00000000 --- a/.github/workflows/push.yaml +++ /dev/null @@ -1,24 +0,0 @@ -name: Test -on: [push] -jobs: - test: - runs-on: ubuntu-latest - strategy: - fail-fast: false - matrix: - python-version: - - '3.8' - - '3.9' - - '3.10' - - '3.11' - - '3.12' - steps: - - uses: actions/checkout@v4 - - uses: extractions/setup-just@v1 - - uses: actions/setup-python@v5 - with: - python-version: ${{ matrix.python-version }} - - run: just deps - - run: just test - - run: just cov - - run: just lint diff --git a/src/posit/connect/client.py b/src/posit/connect/client.py index e0fa2474..59ac246c 100644 --- a/src/posit/connect/client.py +++ b/src/posit/connect/client.py @@ -8,7 +8,7 @@ from .auth import Auth from .config import Config -from .users import Users +from .users import LazyUsers, Users @contextmanager @@ -32,8 +32,6 @@ def create_client( class Client: - users: Users - def __init__( self, api_key: Optional[str] = None, @@ -56,7 +54,7 @@ def __init__( session.hooks["response"].append(hooks.handle_errors) # Initialize the Users instance. - self.users = Users(config=config, session=session) + self.users: Users = LazyUsers(config=config, session=session) # Store the Session object. self._session = session diff --git a/src/posit/connect/client_test.py b/src/posit/connect/client_test.py index 49ee14f4..490ab898 100644 --- a/src/posit/connect/client_test.py +++ b/src/posit/connect/client_test.py @@ -13,12 +13,16 @@ def test(self, Client: MagicMock): class TestClient: - @patch("posit.connect.client.Users") + @patch("posit.connect.client.LazyUsers") @patch("posit.connect.client.Session") @patch("posit.connect.client.Config") @patch("posit.connect.client.Auth") def test_init( - self, Auth: MagicMock, Config: MagicMock, Session: MagicMock, Users: MagicMock + self, + Auth: MagicMock, + Config: MagicMock, + Session: MagicMock, + LazyUsers: MagicMock, ): api_key = "foobar" endpoint = "http://foo.bar" @@ -27,12 +31,11 @@ def test_init( Auth.assert_called_once_with(config=config) Config.assert_called_once_with(api_key=api_key, endpoint=endpoint) Session.assert_called_once() - Users.assert_called_once_with(config=config, session=Session.return_value) + LazyUsers.assert_called_once_with(config=config, session=Session.return_value) - @patch("posit.connect.client.Users") @patch("posit.connect.client.Session") @patch("posit.connect.client.Auth") - def test_del(self, Auth: MagicMock, Session: MagicMock, Users: MagicMock): + def test_del(self, Auth: MagicMock, Session: MagicMock): api_key = "foobar" endpoint = "http://foo.bar" client = Client(api_key=api_key, endpoint=endpoint) diff --git a/src/posit/connect/resources.py b/src/posit/connect/resources.py new file mode 100644 index 00000000..9427b95a --- /dev/null +++ b/src/posit/connect/resources.py @@ -0,0 +1,81 @@ +from __future__ import annotations + + +from abc import ABC, abstractmethod +from typing import Generic, Iterator, Optional, TypeVar, List, TypedDict, Tuple + + +class Resource(TypedDict): + pass + + +T = TypeVar("T", bound=Resource) + + +class Resources(ABC, Generic[T], Iterator[T]): + def __init__(self, data: List[T] = []) -> None: + super().__init__() + self.data = data + + @abstractmethod + def find(self, *args, **kwargs) -> Resources[T]: + raise NotImplementedError() + + @abstractmethod + def find_one(self, *args, **kwargs) -> Optional[T]: + raise NotImplementedError() + + @abstractmethod + def get(self, id: str) -> T: + raise NotImplementedError() + + def __iter__(self) -> Iterator[T]: + self.index = 0 + return self + + def __next__(self) -> T: + if self.index >= len(self.data): + raise StopIteration + + v = self.data[self.index] + self.index += 1 + return v + + def to_pandas(self): + try: + from pandas import DataFrame + + return DataFrame(self) + except ImportError: + return None + + +class LazyResources(Resources[T]): + def __init__(self, data: List[T] = []) -> None: + super().__init__(data) + self.data = data + self.exhausted = False + self.index = 0 + + @abstractmethod + def fetch(self, index) -> Tuple[Optional[Iterator[T]], bool]: + raise NotImplementedError() + + def __iter__(self) -> Iterator[T]: + self.index = 0 + return self + + def __next__(self) -> T: + if self.index >= len(self.data): + if self.exhausted: + raise StopIteration + + results, self.exhausted = self.fetch(self.index) + if not results: + raise StopIteration + + self.data += results + + v = self.data[self.index] + self.index += 1 + return v diff --git a/src/posit/connect/resources_test.py b/src/posit/connect/resources_test.py new file mode 100644 index 00000000..2720690a --- /dev/null +++ b/src/posit/connect/resources_test.py @@ -0,0 +1,40 @@ +from typing import Iterator, Tuple, Optional +from unittest.mock import Mock + +from .resources import Resource, Resources, LazyResources + + +class FakeResource(Resource): + pass + + +class FakeResources(Resources[FakeResource]): + def find(self) -> Resources[FakeResource]: + return self + + def find_one(self) -> Optional[FakeResource]: + return Mock(spec=FakeResource) + + def get(self, _: str) -> FakeResource: + return Mock(spec=FakeResource) + + +class TestResources: + def test(self): + resources = FakeResources() + assert resources == resources.find() + assert resources.find_one() + assert resources.get(None) + + +class FakeLazyResources(FakeResources, LazyResources): + def fetch(self, index) -> Tuple[Optional[Iterator[FakeResource]], bool]: + return iter([FakeResource()]), len(self.data) > 0 + + +class TestFakeLazyResources: + def test(self): + resources = FakeLazyResources() + assert resources == resources.find() + assert resources.find_one() + assert resources.get(None) diff --git a/src/posit/connect/users.py b/src/posit/connect/users.py index 0bc8dc14..248debeb 100644 --- a/src/posit/connect/users.py +++ b/src/posit/connect/users.py @@ -3,17 +3,17 @@ import os from datetime import datetime +from typing import Iterator, Callable, List + from requests import Session -from typing import Iterator, List, Optional, TypedDict from .config import Config -from .endpoints import get_users -from .errors import ClientError +from .resources import LazyResources, Resource, Resources _MAX_PAGE_SIZE = 500 -class User(TypedDict, total=False): +class User(Resource, total=False): guid: str email: str username: str @@ -27,132 +27,55 @@ class User(TypedDict, total=False): locked: bool -class Users(Iterator[User]): - def __init__( - self, config: Config, session: Session, *, users: Optional[List[User]] = None - ): - self._config = config - self._session = session - - self._cached_users: List[User] = users or [] - self._exhausted: bool = users is not None - self._index: int = 0 - self._page_number: int = 0 - - def __iter__(self) -> Iterator[User]: - """ - Initialize the iterator by resetting the index to the beginning of the cached user list. - - Returns: - Iterator: The initialized iterator object. - """ - # Reset the index to the beginning of the cached user list. - self._index = 0 - # Return the iterator object. - return self - - def __next__(self): - """Retrieve the next user in the list. If necessary, fetch a new page of users beforehand. +class Users(Resources[User]): + def find(self, filter: Callable[[User], bool] = lambda _: True) -> Users: + return Users([user for user in self if filter(user)]) - Raises: - StopIteration: If the end of the user list is reached. - StopIteration: If no users are returned for the current page. + def find_one(self, filter: Callable[[User], bool] = lambda _: True) -> User | None: + return next((user for user in self if filter(user)), None) - Returns: - dict: Information about the next user. - """ - # Check if the current index is greater than or equal to the length of the cached user list. - if self._index >= len(self._cached_users): - # Check if the endpoint was exhausted on the previous iteration - if self._exhausted: - # Stop iteration if the index is not aligned with page boundaries. - raise StopIteration - # Fetch the current page of users. - results, exhausted = get_users( - self._config.endpoint, self._session, self._page_number - ) - # Mark if the endpoint is exhausted for the next iteration - self._exhausted = exhausted - # Increment the page counter for the next iteration. - self._page_number += 1 - # Append the fetched users to the cached user list. - self._cached_users += [User(**result) for result in results] - # Check if the fetched results list is empty. - if not results: - # Stop iteration if no users are returned for the current page. - raise StopIteration - # Get the current user by index. - user = self._cached_users[self._index] - # Increment the index for the next iteration. - self._index += 1 - # Return the current user. + def get(self, id: str) -> User: + user = next((user for user in self if user["guid"] == id), None) + if user is None: + raise RuntimeError(f"failed to get user with id '{id}'") return user - def find(self, params: User) -> Users: - """ - Finds users that match the provided filter conditions. - - Args: - params (User): Filter conditions. - - Returns: - Users: A list of users matching the filter conditions. - """ - found: List[User] = [] - for user in self: - # Check if the items in params are subset of user's items. - if params.items() <= user.items(): - # Append the user to the found list. - found.append(user) - return Users(self._config, self._session, users=found) - - def find_one(self, params: User) -> Optional[User]: - """ - Finds one User matching the provided parameters. - - Keyword Arguments: - params -- Dictionary of filter conditions (default: {}). - Returns: - A matching User if found, otherwise None. - - Note: - This method first checks if 'guid' is present in params. If so, it attempts a direct lookup using self.get(). - If an error with code '4' is encountered (indicating no matching user), it logs a warning and returns None. - If 'guid' is not provided, it performs a normal search using self.find() and return the first value found. - """ - # Check if 'guid' is provided in params - if "guid" in params: - try: - # Attempt direct lookup - self.get(params["guid"]) - except ClientError as e: - # Check for error code '4' (no matching user) - if e.error_code == 4: - import logging - - logging.warning(e) - # Return None if user not found - return None - raise e - - # If 'guid' not provided perform a normal search - return next(iter(self.find(params)), None) - - def get(self, guid: str) -> User: - """Gets a user by guid. - - Arguments: - guid -- the users guid. - - Returns: - A :class:`User`. - """ - endpoint = os.path.join(self._config.endpoint, "v1/users", guid) - response = self._session.get(endpoint) +class LazyUsers(Users, LazyResources[User]): + def __init__( + self, config: Config, session: Session, *, page_size=_MAX_PAGE_SIZE + ) -> None: + super().__init__() + self.config = config + self.session = session + self.page_size = page_size + + def fetch(self, index) -> tuple[Iterator[User] | None, bool]: + # Check if index is a multiple of page_size. + if (index % self.page_size) != 0: + # + raise ValueError( + f"index ({index}) must be a multiple of page size ({self.page_size})" + ) + # Construct the endpoint URL. + endpoint = os.path.join(self.config.endpoint, "v1/users") + # Define the page number using 1-based indexing. + page_number = int(index / self.page_size) + 1 + # Define query parameters for pagination. + params = {"page_number": page_number, "page_size": self.page_size} + # Send a GET request to the endpoint with the specified parameters. + response = self.session.get(endpoint, params=params) + # Convert response to dict + json: dict = dict(response.json()) + # Parse the JSON response and extract the results. + results: List[dict] = json["results"] + # Mark exhausted if the result size is less than the maximum page size. + exhausted = len(results) < self.page_size + # Create User objects from the results and return them as a list. + users: Iterator[User] = iter(User(**result) for result in results) + return (users, exhausted) + + def get(self, id: str) -> User: + endpoint = os.path.join(self.config.endpoint, "v1/users", id) + response = self.session.get(endpoint) return User(**response.json()) - - def to_pandas_data_frame(self): # noqa - import pandas as pd - - return pd.DataFrame((user for user in self)) diff --git a/src/posit/connect/users_test.py b/src/posit/connect/users_test.py index aa4f49c7..e69de29b 100644 --- a/src/posit/connect/users_test.py +++ b/src/posit/connect/users_test.py @@ -1,150 +0,0 @@ -import pytest - -from unittest.mock import MagicMock, patch - -from .users import Users, User - - -class TestUsers: - @patch("posit.connect.users.Session") - @patch("posit.connect.users.Config") - def test_init(self, Config: MagicMock, Session: MagicMock): - config = Config.return_value - session = Session.return_value - users = Users(config, session) - assert users._config == config - assert users._session == session - - @patch("posit.connect.users.Session") - @patch("posit.connect.users.Config") - def test_iter(self, Config: MagicMock, Session: MagicMock): - config = Config.return_value - session = Session.return_value - users = Users(config, session) - iter(users) - assert users._index == 0 - - @patch("posit.connect.users.Session") - @patch("posit.connect.users.Config") - def test_next_with_empty_result_set(self, Config: MagicMock, Session: MagicMock): - config = Config.return_value - session = Session.return_value - users = Users(config, session) - with patch("posit.connect.users.get_users") as get_users: - get_users.return_value = [], True - with pytest.raises(StopIteration): - next(users) - - assert users._cached_users == [] - assert users._exhausted is True - assert users._index == 0 - assert users._page_number == 1 - - @patch("posit.connect.users.Session") - @patch("posit.connect.users.Config") - def test_next_with_single_page(self, Config: MagicMock, Session: MagicMock): - config = Config.return_value - config.endpoint = "http://foo.bar" - session = Session.return_value - users = Users(config, session) - user: User = {} - with patch("posit.connect.users.get_users") as get_users: - get_users.return_value = [user], True - assert next(users) == user - get_users.assert_called_with(config.endpoint, session, 0) - - with pytest.raises(StopIteration): - next(users) - - assert users._cached_users == [user] - assert users._exhausted is True - assert users._index == 1 - assert users._page_number == 1 - - @patch("posit.connect.users.Session") - @patch("posit.connect.users.Config") - def test_next_with_multiple_pages(self, Config: MagicMock, Session: MagicMock): - config = Config.return_value - config.endpoint = "http://foo.bar" - session = Session.return_value - users = Users(config, session) - user: User = {} - with patch("posit.connect.users.get_users") as get_users: - get_users.return_value = [user], False - assert next(users) == user - get_users.assert_called_with(config.endpoint, session, 0) - - get_users.return_value = [user], True - assert next(users) == user - get_users.assert_called_with(config.endpoint, session, 1) - - assert users._cached_users == [user, user] - assert users._exhausted is True - assert users._index == 2 - assert users._page_number == 2 - - @patch("posit.connect.users.Session") - @patch("posit.connect.users.Config") - def test_find(self, Config: MagicMock, Session: MagicMock): - config = Config.return_value - config.endpoint = "http://foo.bar" - session = Session.return_value - users = Users(config, session) - user = {"username": "foobar"} - with patch("posit.connect.users.get_users") as get_users: - get_users.return_value = [user], True - found = users.find({"username": "foobar"}) - assert list(found) == [user] - - @patch("posit.connect.users.Session") - @patch("posit.connect.users.Config") - def test_find_miss(self, Config: MagicMock, Session: MagicMock): - config = Config.return_value - config.endpoint = "http://foo.bar" - session = Session.return_value - users = Users(config, session) - user = {"username": "foo"} - with patch("posit.connect.users.get_users") as get_users: - get_users.return_value = [user], True - assert list(users.find({"username": "bar"})) == [] - - @patch("posit.connect.users.Session") - @patch("posit.connect.users.Config") - def test_find_one(self, Config: MagicMock, Session: MagicMock): - config = Config.return_value - config.endpoint = "http://foo.bar" - session = Session.return_value - users = Users(config, session) - user = {"username": "foobar"} - with patch("posit.connect.users.get_users") as get_users: - get_users.return_value = [user], True - assert users.find_one({"username": "foobar"}) == user - - @patch("posit.connect.users.Session") - @patch("posit.connect.users.Config") - def test_find_one_miss(self, Config: MagicMock, Session: MagicMock): - config = Config.return_value - config.endpoint = "http://foo.bar" - session = Session.return_value - users = Users(config, session) - user = {"username": "foo"} - with patch("posit.connect.users.get_users") as get_users: - get_users.return_value = [user], True - assert users.find_one({"username": "bar"}) is None - - @patch("posit.connect.users.Session") - @patch("posit.connect.users.Config") - def test_get(self, Config: MagicMock, Session: MagicMock): - config = Config.return_value - config.endpoint = "http://foo.bar" - - user = {"guid": "foobar"} - response = MagicMock() - response.json = MagicMock() - response.json.return_value = user - session = Session.return_value - session.get = MagicMock() - session.get.return_value = response - - users = Users(config, session) - assert users.get("foobar") == user diff --git a/tinkering.py b/tinkering.py index 74be435e..a8030972 100644 --- a/tinkering.py +++ b/tinkering.py @@ -1,12 +1,10 @@ from posit.connect.client import create_client with create_client() as client: + print(client.users.get("f55ca95d-ce52-43ed-b31b-48dc4a07fe13")) + users = client.users - print( - users.find({"username": "taylor_steinberg"}).find( - {"username": "taylor_steinberg"} - ) - ) - print(users.find_one({"username": "taylor_steinberg"})) - print(users.get("f55ca95d-ce52-43ed-b31b-48dc4a07fe13")) - print(users.to_pandas_data_frame()) + users = users.find(lambda user: user["first_name"].startswith("T")) + users = users.find(lambda user: user["last_name"].startswith("S")) + user = users.find_one(lambda user: user["user_role"] == "administrator") + print(user)