From 48abf612afe81273d24a32c4e6ed6747b9ca348b Mon Sep 17 00:00:00 2001 From: Tyler Nullmeier Date: Fri, 25 Oct 2024 15:56:45 -0500 Subject: [PATCH] Get rex release version from github repository --- backend/app/app/core/config.py | 5 +++ backend/app/app/github/api.py | 37 ++++++++++++++++ backend/app/app/service/abl.py | 24 ++++++----- .../unit/data/get_rex_release_version.yaml | 18 ++++++++ backend/app/tests/unit/init_test_data.py | 11 +++++ backend/app/tests/unit/test_abl.py | 42 +++++++------------ 6 files changed, 99 insertions(+), 38 deletions(-) create mode 100644 backend/app/tests/unit/data/get_rex_release_version.yaml diff --git a/backend/app/app/core/config.py b/backend/app/app/core/config.py index 169452182..a2b98b774 100644 --- a/backend/app/app/core/config.py +++ b/backend/app/app/core/config.py @@ -36,6 +36,11 @@ REX_WEB_RELEASE_URL = os.getenv( "REX_WEB_RELEASE_URL", "https://openstax.org/rex/release.json" ) +REX_WEB_ARCHIVE_CONFIG = os.getenv( + "REX_WEB_ARCHIVE_CONFIG", + # owner:repo:path + "openstax:rex-web:src/config.archive-url.json", +) # GITHUB OAUTH CLIENT_ID = os.getenv("GITHUB_OAUTH_ID") diff --git a/backend/app/app/github/api.py b/backend/app/app/github/api.py index 05a7c51f1..b0724f6a9 100644 --- a/backend/app/app/github/api.py +++ b/backend/app/app/github/api.py @@ -2,6 +2,7 @@ import json from datetime import datetime from typing import Any, Dict, List, Tuple +from urllib.parse import urlencode from lxml import etree @@ -120,6 +121,42 @@ async def get_collections( } +def normpath(*parts: str): + return tuple(p.strip("/") for p in parts) + + +def build_url(*parts: str, **kwargs: str | None): + path = "/".join(("https://api.github.com", *normpath(*parts))) + kwargs = {k: v for k, v in kwargs.items() if v} + if kwargs: + path = "?".join((path, urlencode(kwargs))) + return path + + +async def get_file_response( + client: AuthenticatedClient, + owner: str, + repo: str, + path: str, + ref: str | None = None, +): + url = build_url("repos", owner, repo, "contents", path, ref=ref) + response = await client.get(url) + response.raise_for_status() + return response + + +async def get_file_content( + client: AuthenticatedClient, + owner: str, + repo: str, + path: str, + ref: str | None = None, +): + payload = (await get_file_response(client, owner, repo, path, ref)).json() + return base64.b64decode(payload["content"]) + + async def push_to_github( client: AuthenticatedClient, path: str, diff --git a/backend/app/app/service/abl.py b/backend/app/app/service/abl.py index 8e470ec8e..f09b16ed2 100644 --- a/backend/app/app/service/abl.py +++ b/backend/app/app/service/abl.py @@ -1,4 +1,4 @@ -import re +import json from typing import Any, Dict, List, Optional from httpx import AsyncClient, HTTPStatusError @@ -16,6 +16,7 @@ Consumer, Repository, ) +from app.github.api import get_file_content async def get_rex_release_json(client: AsyncClient): @@ -38,15 +39,18 @@ async def get_rex_books(client: AsyncClient): async def get_rex_release_version(client: AsyncClient): - rex_release = await get_rex_release_json(client) - archive_url = rex_release.get("archiveUrl", "").strip() - if archive_url == "": - raise CustomBaseError("Could not find valid REX archive URL") - # Search for: %Y%m%d.%H%M%S - version_matches = re.findall(r"\d{8}\.\d{6}", archive_url) - if len(version_matches) != 1: - raise CustomBaseError("Could not determine REX release version") - return version_matches[0] + owner, repo, path = config.REX_WEB_ARCHIVE_CONFIG.split(":", 2) + try: + raw_contents = await get_file_content(client, owner, repo, path) + except HTTPStatusError as he: + raise CustomBaseError( + f"Failed to fetch rex release version: {he.response.status_code}" + ) from he + contents = json.loads(raw_contents) + version = contents.get("REACT_APP_ARCHIVE", "").strip() + if not version: + raise CustomBaseError("Could not find valid REX version") + return version def get_rex_book_versions(rex_books: Dict[str, Any], book_uuids: List[str]): diff --git a/backend/app/tests/unit/data/get_rex_release_version.yaml b/backend/app/tests/unit/data/get_rex_release_version.yaml new file mode 100644 index 000000000..f8fe04de6 --- /dev/null +++ b/backend/app/tests/unit/data/get_rex_release_version.yaml @@ -0,0 +1,18 @@ +interactions: +- request: + body: '' + headers: + host: + - api.github.com + method: GET + uri: https://api.github.com/repos/openstax/rex-web/contents/src/config.archive-url.json + response: + content: '{"name":"config.archive-url.json","path":"src/config.archive-url.json","sha":"ddd2befbcff5578dda01dda94cd6c2474999d876","size":95,"url":"https://api.github.com/repos/openstax/rex-web/contents/src/config.archive-url.json?ref=main","html_url":"https://github.com/openstax/rex-web/blob/main/src/config.archive-url.json","git_url":"https://api.github.com/repos/openstax/rex-web/git/blobs/ddd2befbcff5578dda01dda94cd6c2474999d876","download_url":"https://raw.githubusercontent.com/openstax/rex-web/main/src/config.archive-url.json","type":"file","content":"ewogICJSRUFDVF9BUFBfQVJDSElWRSI6ICIyMDI0MDkxMC4xNjEyMjciLAog\nICJSRUFDVF9BUFBfQVJDSElWRV9VUkxfQkFTRSI6ICIvYXBwcy9hcmNoaXZl\nLyIKfQo=\n","encoding":"base64","_links":{"self":"https://api.github.com/repos/openstax/rex-web/contents/src/config.archive-url.json?ref=main","git":"https://api.github.com/repos/openstax/rex-web/git/blobs/ddd2befbcff5578dda01dda94cd6c2474999d876","html":"https://github.com/openstax/rex-web/blob/main/src/config.archive-url.json"}}' + headers: + Content-Type: + - application/json; charset=utf-8 + Server: + - github.com + http_version: HTTP/1.1 + status_code: 200 +version: 1 diff --git a/backend/app/tests/unit/init_test_data.py b/backend/app/tests/unit/init_test_data.py index 66ec6cfe5..51fb8e86a 100644 --- a/backend/app/tests/unit/init_test_data.py +++ b/backend/app/tests/unit/init_test_data.py @@ -15,6 +15,7 @@ get_user_repositories, get_user_teams, ) +from app.github.api import get_file_content def apply_key_whitelist(d, whitelist): @@ -103,6 +104,15 @@ async def mock_get_collections(client, repo_name, repo_owner, commit_sha): return await get_collections(client, repo_name, repo_owner, commit_sha) +@my_vcr.use_cassette( + "get_rex_release_version.yaml", serializer="base_sanitizer" +) +async def mock_get_rex_release_version(client): + return await get_file_content( + client, "openstax", "rex-web", "src/config.archive-url.json" + ) + + async def async_main(access_token: str): async with AsyncClient() as client: client.headers = {"authorization": f"Bearer {access_token}"} @@ -114,6 +124,7 @@ async def async_main(access_token: str): ) await mock_get_book_repository(client, "tiny-book", "openstax", "main") await mock_get_collections(client, "tiny-book", "openstax", "main") + await mock_get_rex_release_version(client) def main(access_token): diff --git a/backend/app/tests/unit/test_abl.py b/backend/app/tests/unit/test_abl.py index 3c2af07fe..b1c8188a1 100644 --- a/backend/app/tests/unit/test_abl.py +++ b/backend/app/tests/unit/test_abl.py @@ -1,3 +1,6 @@ +import base64 +import json + import pytest from app.core import config @@ -216,11 +219,16 @@ def test_get_rex_book_versions(rex_books, book_uuids, expected): @pytest.mark.asyncio async def test_get_rex_release_version(mock_http_client): + url = "https://api.github.com/repos/openstax/rex-web/contents/src/config.archive-url.json" + fake_api_response = { + "content": base64.b64encode( + json.dumps({"REACT_APP_ARCHIVE": "20240101.000001"}).encode() + ).decode() + } + # GIVEN: A valid response mock_client: MockAsyncClient = mock_http_client( - get={ - config.REX_WEB_RELEASE_URL: {"archiveUrl": "a/b/20240101.000001/c"} - } + get={url: fake_api_response} ) # WHEN: A request is made version = await get_rex_release_version(mock_client) @@ -231,39 +239,17 @@ async def test_get_rex_release_version(mock_http_client): # THEN: The expected version is matched assert version == "20240101.000001" - # GIVEN: An invalid response with zero matches - mock_client: MockAsyncClient = mock_http_client( - get={config.REX_WEB_RELEASE_URL: {"archiveUrl": "a/b/c"}} - ) - # WHEN: A request is made - # THEN: An error is raised - with pytest.raises(CustomBaseError) as cbe: - await get_rex_release_version(mock_client) - assert len(mock_client.responses) == 1 - assert cbe.match("Could not determine REX release version") - # GIVEN: An invalid response with more than one match + # GIVEN: An invalid response mock_client: MockAsyncClient = mock_http_client( get={ - config.REX_WEB_RELEASE_URL: { - "archiveUrl": "a/b/c/20240101.000001/d/12345678.123456" - } + url: {"content": base64.b64encode(json.dumps({}).encode()).decode()} } ) # WHEN: A request is made # THEN: An error is raised with pytest.raises(CustomBaseError) as cbe: await get_rex_release_version(mock_client) - assert len(mock_client.responses) == 1 - assert cbe.match("Could not determine REX release version") - # GIVEN: An invalid response - mock_client: MockAsyncClient = mock_http_client( - get={config.REX_WEB_RELEASE_URL: {}} - ) - # WHEN: A request is made - # THEN: An error is raised - with pytest.raises(CustomBaseError) as cbe: - await get_rex_release_version(mock_client) - assert cbe.match("Could not find valid REX archive URL") + assert cbe.match("Could not find valid REX version") # GIVEN: A no response mock_client: MockAsyncClient = mock_http_client() # WHEN: A request is made