| 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
 | """An API wrapper around the Site API."""
import asyncio
from urllib.parse import quote as quote_url
import aiohttp
from pydis_core.utils.logging import get_logger
log = get_logger(__name__)
class ResponseCodeError(ValueError):
    """Raised in :meth:`APIClient.request` when a non-OK HTTP response is received."""
    def __init__(
        self,
        response: aiohttp.ClientResponse,
        response_json: dict | None = None,
        response_text: str | None = None
    ):
        """
        Initialize a new :obj:`ResponseCodeError` instance.
        Args:
            response (:obj:`aiohttp.ClientResponse`): The response object from the request.
            response_json: The JSON response returned from the request, if any.
            response_text: The text of the request, if any.
        """
        self.status = response.status
        self.response_json = response_json or {}
        self.response_text = response_text
        self.response = response
    def __str__(self):
        """Return a string representation of the error."""
        response = self.response_json or self.response_text
        return f"Status: {self.status} Response: {response}"
class APIClient:
    """A wrapper for the Django Site API."""
    session: aiohttp.ClientSession | None = None
    loop: asyncio.AbstractEventLoop = None
    def __init__(self, site_api_url: str, site_api_token: str, **session_kwargs):
        """
        Initialize a new :obj:`APIClient` instance.
        Args:
            site_api_url: The URL of the site API.
            site_api_token: The token to use for authentication.
            session_kwargs: Keyword arguments to pass to the :obj:`aiohttp.ClientSession` constructor.
        """
        self.site_api_url = site_api_url
        auth_headers = {
            "Authorization": f"Token {site_api_token}"
        }
        if "headers" in session_kwargs:
            session_kwargs["headers"].update(auth_headers)
        else:
            session_kwargs["headers"] = auth_headers
        # aiohttp will complain if APIClient gets instantiated outside a coroutine. Thankfully, we
        # don't and shouldn't need to do that, so we can avoid scheduling a task to create it.
        self.session = aiohttp.ClientSession(**session_kwargs)
    def _url_for(self, endpoint: str) -> str:
        return f"{self.site_api_url}/{quote_url(endpoint)}"
    async def close(self) -> None:
        """Close the aiohttp session."""
        await self.session.close()
    @staticmethod
    async def maybe_raise_for_status(response: aiohttp.ClientResponse, *, should_raise: bool) -> None:
        """
        Raise :exc:`ResponseCodeError` for non-OK response if an exception should be raised.
        Args:
            response (:obj:`aiohttp.ClientResponse`): The response to check.
            should_raise: Whether or not to raise an exception.
        Raises:
            :exc:`ResponseCodeError`:
                If the response is not OK and ``should_raise`` is True.
        """
        if should_raise and response.status >= 400:
            try:
                response_json = await response.json()
                raise ResponseCodeError(response=response, response_json=response_json)
            except aiohttp.ContentTypeError:
                response_text = await response.text()
                raise ResponseCodeError(response=response, response_text=response_text)
    async def request(self, method: str, endpoint: str, *, raise_for_status: bool = True, **kwargs) -> dict | None:
        """
        Send an HTTP request to the site API and return the JSON response.
        Args:
            method: The HTTP method to use.
            endpoint: The endpoint to send the request to.
            raise_for_status: Whether or not to raise an exception if the response is not OK.
            **kwargs: Any extra keyword arguments to pass to :func:`aiohttp.request`.
        Returns:
            The JSON response the API returns, or :obj:`None` if the response code is 204.
        Raises:
            :exc:`ResponseCodeError`:
                If the response is not OK and ``raise_for_status`` is True.
        """
        async with self.session.request(method.upper(), self._url_for(endpoint), **kwargs) as resp:
            if resp.status == 204:
                return None
            await self.maybe_raise_for_status(resp, should_raise=raise_for_status)
            return await resp.json()
    async def get(self, endpoint: str, *, raise_for_status: bool = True, **kwargs) -> dict | None:
        """Equivalent to :meth:`APIClient.request` with GET passed as the method."""
        return await self.request("GET", endpoint, raise_for_status=raise_for_status, **kwargs)
    async def patch(self, endpoint: str, *, raise_for_status: bool = True, **kwargs) -> dict | None:
        """Equivalent to :meth:`APIClient.request` with PATCH passed as the method."""
        return await self.request("PATCH", endpoint, raise_for_status=raise_for_status, **kwargs)
    async def post(self, endpoint: str, *, raise_for_status: bool = True, **kwargs) -> dict | None:
        """Equivalent to :meth:`APIClient.request` with POST passed as the method."""
        return await self.request("POST", endpoint, raise_for_status=raise_for_status, **kwargs)
    async def put(self, endpoint: str, *, raise_for_status: bool = True, **kwargs) -> dict | None:
        """Equivalent to :meth:`APIClient.request` with PUT passed as the method."""
        return await self.request("PUT", endpoint, raise_for_status=raise_for_status, **kwargs)
    async def delete(self, endpoint: str, *, raise_for_status: bool = True, **kwargs) -> dict | None:
        """Equivalent to :meth:`APIClient.request` with DELETE passed as the method."""
        return await self.request("DELETE", endpoint, raise_for_status=raise_for_status, **kwargs)
__all__ = ["APIClient", "ResponseCodeError"]
 |