|
| 1 | +from typing import Optional |
| 2 | + |
| 3 | +from athenian.api.db import Database |
| 4 | +from tests.testutils.auth import force_request_auth |
| 5 | +from tests.testutils.db import DBCleaner, models_insert |
| 6 | +from tests.testutils.factory import metadata as md_factory |
| 7 | +from tests.testutils.factory.state import ( |
| 8 | + MappedJIRAIdentityFactory, |
| 9 | + TeamFactory, |
| 10 | + UserAccountFactory, |
| 11 | +) |
| 12 | +from tests.testutils.requester import Requester |
| 13 | + |
| 14 | + |
| 15 | +class BaseListTeamMembersTest(Requester): |
| 16 | + async def _request( |
| 17 | + self, |
| 18 | + team_id: int, |
| 19 | + assert_status: int = 200, |
| 20 | + recursive: Optional[bool] = None, |
| 21 | + account: Optional[int] = None, |
| 22 | + headers: Optional[dict] = None, |
| 23 | + user_id: Optional[str] = None, |
| 24 | + ) -> list[dict] | dict: |
| 25 | + path = f"/private/align/members/{team_id}" |
| 26 | + params = {} |
| 27 | + if recursive is not None: |
| 28 | + params["recursive"] = str(recursive).lower() |
| 29 | + if account is not None: |
| 30 | + params["account"] = str(account) |
| 31 | + headers = self.headers if headers is None else (self.headers | headers) |
| 32 | + |
| 33 | + with force_request_auth(user_id, headers) as headers: |
| 34 | + response = await self.client.request( |
| 35 | + method="GET", path=path, headers=headers, params=params, |
| 36 | + ) |
| 37 | + |
| 38 | + assert response.status == assert_status |
| 39 | + return await response.json() |
| 40 | + |
| 41 | + |
| 42 | +class TestListTeamMembersErrors(BaseListTeamMembersTest): |
| 43 | + async def test_invalid_auth_token(self, sdb: Database) -> None: |
| 44 | + await models_insert(sdb, TeamFactory(id=1, members=[1, 2])) |
| 45 | + headers = {"Authorization": "Bearer invalid"} |
| 46 | + res = await self._request(1, 401, headers=headers) |
| 47 | + assert isinstance(res, dict) |
| 48 | + assert res["type"] == "/errors/Unauthorized" |
| 49 | + |
| 50 | + async def test_team_not_found(self) -> None: |
| 51 | + res = await self._request(999, 404) |
| 52 | + assert isinstance(res, dict) |
| 53 | + assert res["detail"] == "Team 999 not found or access denied" |
| 54 | + |
| 55 | + async def test_team_not_found_recursive(self) -> None: |
| 56 | + res = await self._request(999, 404, recursive=True) |
| 57 | + assert isinstance(res, dict) |
| 58 | + assert res["detail"] == "Team 999 not found or access denied" |
| 59 | + |
| 60 | + async def test_team_account_mismatch(self, sdb: Database) -> None: |
| 61 | + await models_insert(sdb, TeamFactory(id=10, owner_id=3)) |
| 62 | + res = await self._request(10, 404) |
| 63 | + assert isinstance(res, dict) |
| 64 | + assert res["detail"] == "Team 10 not found or access denied" |
| 65 | + |
| 66 | + async def test_team_correct_account_param_mismatch(self, sdb: Database) -> None: |
| 67 | + await models_insert(sdb, TeamFactory(id=10)) |
| 68 | + res = await self._request(10, 404, account=3) |
| 69 | + assert isinstance(res, dict) |
| 70 | + assert res["detail"] == "Team 10 not found or access denied" |
| 71 | + |
| 72 | + async def test_team_account_mismatch_non_default_user(self, sdb: Database) -> None: |
| 73 | + await models_insert( |
| 74 | + sdb, UserAccountFactory(user_id="my-user", account_id=2), TeamFactory(id=10), |
| 75 | + ) |
| 76 | + await self._request(10, 404, user_id="my-user") |
| 77 | + |
| 78 | + async def test_root_team_multiple_roots(self, sdb: Database) -> None: |
| 79 | + await models_insert(sdb, TeamFactory(), TeamFactory()) |
| 80 | + res = await self._request(0, 500, account=1) |
| 81 | + assert isinstance(res, dict) |
| 82 | + assert res["title"] == "Multiple root teams" |
| 83 | + assert res["detail"] == "Account 1 has multiple root teams" |
| 84 | + |
| 85 | + async def test_root_team_no_team_exists(self, sdb: Database) -> None: |
| 86 | + res = await self._request(0, 404, account=1) |
| 87 | + assert isinstance(res, dict) |
| 88 | + assert res["type"] == "/errors/teams/TeamNotFound" |
| 89 | + assert res["detail"] == "Root team not found or access denied" |
| 90 | + |
| 91 | + async def test_root_team_missing_account(self, sdb: Database) -> None: |
| 92 | + res = await self._request(0, 400, account=None) |
| 93 | + assert isinstance(res, dict) |
| 94 | + assert res["type"] == "/errors/BadRequest" |
| 95 | + |
| 96 | + |
| 97 | +class TestListTeamMembers(BaseListTeamMembersTest): |
| 98 | + async def test_explicit_non_root_team(self, sdb: Database) -> None: |
| 99 | + # members id are from the 6 MB metadata db fixture |
| 100 | + await models_insert( |
| 101 | + sdb, |
| 102 | + TeamFactory(id=1, members=[]), |
| 103 | + TeamFactory(id=2, members=[40078, 39652900], parent_id=1), |
| 104 | + ) |
| 105 | + |
| 106 | + res = await self._request(2) |
| 107 | + assert [m["login"] for m in res] == ["github.com/leantrace", "github.com/reujab"] |
| 108 | + assert [m["name"] for m in res] == ["Alexander Schamne", "Christopher Knight"] |
| 109 | + |
| 110 | + assert "email" in res[0] |
| 111 | + assert "picture" in res[0] |
| 112 | + |
| 113 | + async def test_explicit_root_team(self, sdb: Database) -> None: |
| 114 | + await models_insert(sdb, TeamFactory(id=1, members=[40078], parent_id=None)) |
| 115 | + res = await self._request(1) |
| 116 | + assert [m["login"] for m in res] == ["github.com/reujab"] |
| 117 | + |
| 118 | + async def test_implicit_root_team_empty(self, sdb: Database) -> None: |
| 119 | + await models_insert( |
| 120 | + sdb, |
| 121 | + TeamFactory(id=10, members=[]), |
| 122 | + TeamFactory(id=20, parent_id=10, members=[3]), |
| 123 | + TeamFactory(id=30, parent_id=10, members=[3, 4]), |
| 124 | + ) |
| 125 | + res = await self._request(0, account=1) |
| 126 | + assert res == [] |
| 127 | + |
| 128 | + async def test_implicit_root_team(self, sdb: Database) -> None: |
| 129 | + await models_insert( |
| 130 | + sdb, |
| 131 | + TeamFactory(id=10, members=[40078]), |
| 132 | + TeamFactory(id=11, parent_id=10, members=[39652900]), |
| 133 | + ) |
| 134 | + res = await self._request(0, account=1) |
| 135 | + assert [m["login"] for m in res] == ["github.com/reujab"] |
| 136 | + |
| 137 | + async def test_recursive(self, sdb: Database) -> None: |
| 138 | + await models_insert( |
| 139 | + sdb, |
| 140 | + TeamFactory(id=1, members=[39652900]), |
| 141 | + TeamFactory(id=2, parent_id=1, members=[40078]), |
| 142 | + ) |
| 143 | + |
| 144 | + res = await self._request(1, recursive=True) |
| 145 | + assert [m["login"] for m in res] == ["github.com/leantrace", "github.com/reujab"] |
| 146 | + |
| 147 | + res = await self._request(1, recursive=False) |
| 148 | + assert [m["login"] for m in res] == ["github.com/leantrace"] |
| 149 | + |
| 150 | + res = await self._request(1) |
| 151 | + assert [m["login"] for m in res] == ["github.com/leantrace"] |
| 152 | + |
| 153 | + async def test_result_ordering(self, sdb: Database, mdb_rw: Database) -> None: |
| 154 | + await models_insert( |
| 155 | + sdb, |
| 156 | + TeamFactory(id=1, members=[100, 101, 102, 103]), |
| 157 | + ) |
| 158 | + |
| 159 | + async with DBCleaner(mdb_rw) as mdb_cleaner: |
| 160 | + models = [ |
| 161 | + md_factory.UserFactory(node_id=100, html_url="https://c", name="Foo Bar"), |
| 162 | + md_factory.UserFactory(node_id=101, html_url="https://a", name=None), |
| 163 | + md_factory.UserFactory(node_id=102, html_url="https://b", name="zz-Top"), |
| 164 | + md_factory.UserFactory(node_id=103, html_url="https://d", name="aa"), |
| 165 | + ] |
| 166 | + mdb_cleaner.add_models(*models) |
| 167 | + await models_insert(mdb_rw, *models) |
| 168 | + |
| 169 | + res = await self._request(1, recursive=True) |
| 170 | + |
| 171 | + assert [m["login"] for m in res] == ["d", "c", "b", "a"] |
| 172 | + assert [m.get("name") for m in res] == ["aa", "Foo Bar", "zz-Top", None] |
| 173 | + |
| 174 | + async def test_jira_user_field(self, sdb: Database, mdb_rw: Database) -> None: |
| 175 | + await models_insert( |
| 176 | + sdb, |
| 177 | + TeamFactory(id=1, members=[100, 101]), |
| 178 | + MappedJIRAIdentityFactory(github_user_id=100, jira_user_id="200"), |
| 179 | + ) |
| 180 | + |
| 181 | + async with DBCleaner(mdb_rw) as mdb_cleaner: |
| 182 | + models = [ |
| 183 | + md_factory.UserFactory(node_id=100), |
| 184 | + md_factory.UserFactory(node_id=101), |
| 185 | + md_factory.JIRAUserFactory(id="200", display_name="My JIRA name"), |
| 186 | + ] |
| 187 | + mdb_cleaner.add_models(*models) |
| 188 | + await models_insert(mdb_rw, *models) |
| 189 | + res = await self._request(1) |
| 190 | + |
| 191 | + assert [m.get("jira_user") for m in res] == ["My JIRA name", None] |
0 commit comments