forked from NiklasRosenstein/python-github-bot-api
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathapp.py
More file actions
213 lines (165 loc) · 6.57 KB
/
app.py
File metadata and controls
213 lines (165 loc) · 6.57 KB
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
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
"""
Registry for GitHub event handlers.
"""
import dataclasses
import logging
import sys
import threading
import typing as t
import deprecated
import requests
import urllib3 # type: ignore[import]
from nr.functional import coalesce
from . import __version__
from .token import InstallationTokenSupplier, JwtSupplier, TokenInfo
T = t.TypeVar('T')
logger = logging.getLogger(__name__)
user_agent = f'python/{sys.version.split()[0]} github-bot-api/{__version__}'
if t.TYPE_CHECKING:
import github
@dataclasses.dataclass
class GithubClientSettings:
"""
Settings for constructing a #github.Github client object.
"""
base_url: t.Optional[str] = None
user_agent: t.Optional[str] = None
timeout: t.Optional[float] = None
per_page: t.Optional[int] = None
verify: t.Optional[bool] = None
retry: t.Optional[urllib3.Retry] = None
def update(self, other: 'GithubClientSettings') -> 'GithubClientSettings':
result = GithubClientSettings()
for field in dataclasses.fields(self):
value = getattr(other, field.name)
if value is None:
value = getattr(self, field.name)
setattr(result, field.name, value)
return result
def make_client(self, login_or_token: t.Optional[str] = None, jwt: t.Optional[str] = None) -> 'github.Github':
import github, github.MainClass
return github.Github(
login_or_token=login_or_token,
jwt=jwt,
base_url=self.base_url or github.MainClass.DEFAULT_BASE_URL, # type: ignore[attr-defined]
user_agent=self.user_agent or "PyGithub/Python",
timeout=coalesce(self.timeout, github.MainClass.DEFAULT_TIMEOUT), # type: ignore[attr-defined]
per_page=coalesce(self.per_page, github.MainClass.DEFAULT_PER_PAGE), # type: ignore[attr-defined]
verify=coalesce(self.verify, True),
retry=self.retry)
@dataclasses.dataclass
class GithubApp:
"""
Represents a GitHub application and all the required details.
# Example
```py
import os
from github_bot_api import GithubApp
with open(os.environ['PRIVATE_KEY_FILE']) as fp:
private_key = fp.read()
app = GithubApp(
user_agent='my-bot/0.0.0',
app_id=int(os.environ['APP_ID']),
private_key=private_key)
print(app.app_client().get_app().owner)
```
"""
PUBLIC_GITHUB_V3_API_URL = 'https://api.github.com'
#: User agent of the application. This will be respected in #get_user_agent().
user_agent: str
#: GitHub Application ID.
app_id: int
#: RSA private key to sign the JWT with.
private_key: str
#: GitHub API base URL. Defaults to the public GitHub API.
v3_api_url: str = PUBLIC_GITHUB_V3_API_URL
def __post_init__(self):
self._jwt_supplier = JwtSupplier(self.app_id, self.private_key)
self._lock = threading.Lock()
self._installation_tokens: t.Dict[int, InstallationTokenSupplier] = {}
def _get_base_github_client_settings(self) -> GithubClientSettings:
return GithubClientSettings(self.v3_api_url, self.get_user_agent())
def get_user_agent(self, installation_id: t.Optional[int] = None) -> str:
"""
Create a user agent string for the PyGithub client, including the installation if specified.
"""
user_agent = f'{self.user_agent} PyGithub/python (app_id={self.app_id}'
if installation_id:
user_agent += f', installation_id={installation_id})'
return user_agent
@property
def jwt(self) -> TokenInfo:
"""
Returns the JWT for your GitHub application. The JWT is the token to use with GitHub application APIs.
"""
return self._jwt_supplier()
@property
def jwt_supplier(self) -> JwtSupplier:
"""
Returns a new #JwtSupplier that is used for generating JWT tokens for your GitHub application.
"""
return JwtSupplier(self.app_id, self.private_key)
@property # type: ignore[misc]
@deprecated.deprecated(reason='use GithubApp.app_client() instead', version='0.4.0')
def client(self) -> 'github.Github':
"""
Use #app_client() instead.
"""
return self.app_client()
def app_client(self, settings: t.Union[GithubClientSettings, t.Dict[str, t.Any], None] = None) -> 'github.Github':
"""
Returns a PyGithub client for your GitHub application.
Note that the client's token will expire after 10 minutes and you will have to create a new client or update the
client's token with the value returned by #jwt. It is recommended that you create a new client for each atomic
operation you perform.
This requires you to install `PyGithub`.
"""
if isinstance(settings, dict):
settings = GithubClientSettings(**settings)
elif settings is None:
settings = GithubClientSettings()
settings = self._get_base_github_client_settings().update(settings)
return settings.make_client(jwt=self.jwt.value)
def __requestor(self, auth_header: str, installation_id: int) -> t.Dict[str, str]:
return requests.post(
self.v3_api_url.rstrip('/') + f'/app/installations/{installation_id}/access_tokens',
headers={'Authorization': auth_header, 'User-Agent': user_agent},
).json()
def get_installation_token_supplier(self, installation_id: int) -> InstallationTokenSupplier:
"""
Create an #InstallationTokenSupplier for your GitHub application to act within the scope of the given
*installation_id*.
"""
with self._lock:
return self._installation_tokens.setdefault(
installation_id,
InstallationTokenSupplier(
self._jwt_supplier,
installation_id,
self.__requestor,
)
)
def installation_token(self, installation_id: int) -> TokenInfo:
"""
A short-hand to retrieve a new installation token for the given *installation_id*.
"""
return self.get_installation_token_supplier(installation_id)()
def installation_client(
self,
installation_id: int,
settings: t.Union[GithubClientSettings, t.Dict[str, t.Any], None] = None,
) -> 'github.Github':
"""
Returns a PyGithub client for your GitHub application to act in the scope of the given *installation_id*.
Note that the client's token will expire after 10 minutes and you will have to create a new client or update the
client's token with the value returned by #jwt. It is recommended that you create a new client for each atomic
operation you perform.
This requires you to install `PyGithub`.
"""
if isinstance(settings, dict):
settings = GithubClientSettings(**settings)
elif settings is None:
settings = GithubClientSettings()
token = self.installation_token(installation_id).value
settings = self._get_base_github_client_settings().update(settings)
return settings.make_client(login_or_token=token)