-
-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathgithubApi.ts
148 lines (133 loc) Β· 3.86 KB
/
githubApi.ts
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
import { Octokit } from '@octokit/rest';
import { LogLevel, logger } from '../logger';
import { ConfigurationError } from './errors';
/**
* Abstraction for GitHub remotes
*/
export class GitHubRemote {
/** GitHub owner */
public readonly owner: string;
/** GitHub repository name */
public readonly repo: string;
/** GitHub personal authentication token */
protected apiToken?: string;
/** GitHub hostname */
protected readonly GITHUB_HOSTNAME: string = 'github.com';
/** Protocol prefix */
protected readonly PROTOCOL_PREFIX: string = 'https://';
/** Url in the form of /OWNER/REPO/ */
protected readonly url: string;
public constructor(
owner: string,
repo: string,
apiToken?: string
) {
this.owner = owner;
this.repo = repo;
this.apiToken = apiToken;
this.url = `/${this.owner}/${this.repo}/`;
}
/**
* Returns an HTTP-based git remote
*
* It is guaranteed not to contain any sensitive information (e.g. API tokens)
*/
public getRemoteString(): string {
return this.PROTOCOL_PREFIX + this.GITHUB_HOSTNAME + this.url;
}
/**
* Returns an HTTP-based git remote with embedded HTTP basic auth
*
* Using dummy username as it does not matter for cloning
*
* It MAY contain sensitive information (e.g. API tokens)
*/
public getRemoteStringWithAuth(): string {
const authData =
this.apiToken
? `placeholderusername:${this.apiToken}@`
: '';
return this.PROTOCOL_PREFIX + authData + this.GITHUB_HOSTNAME + this.url;
}
}
/**
* Gets GitHub API token from environment
*
* @returns GitHub authentication token if found
*/
export function getGitHubApiToken(): string {
const githubApiToken =
process.env.GITHUB_TOKEN || process.env.GITHUB_API_TOKEN;
if (!githubApiToken) {
throw new ConfigurationError(
'GitHub target: GITHUB_TOKEN not found in the environment'
);
}
return githubApiToken;
}
const _GitHubClientCache: Record<string, Octokit> = {};
/**
* Gets an authenticated GitHub client object
*
* The authentication token is taken from the environment, if not provided.
*
* @param token GitHub authentication token
* @returns GitHub client
*/
export function getGitHubClient(token = ''): Octokit {
const githubApiToken = token || getGitHubApiToken();
if (!_GitHubClientCache[githubApiToken]) {
const attrs: any = {
auth: `token ${githubApiToken}`,
};
// Silence debug logs, as they do not provide any useful information
// about the requests, yet they are very noisy and make it difficult
// to track what's going on.
if (logger.level >= LogLevel.Debug) {
attrs.log = {
info: (message: string) => logger.debug(message),
};
}
// eslint-disable-next-line @typescript-eslint/no-var-requires
const { retry } = require('@octokit/plugin-retry');
const octokitWithRetries = Octokit.plugin(retry);
_GitHubClientCache[githubApiToken] = new octokitWithRetries(attrs);
}
return _GitHubClientCache[githubApiToken];
}
/**
* Loads a file from the context's repository
*
* @param github GitHub client
* @param owner Repository owner
* @param repo Repository name
* @param path The path of the file in the repository
* @param ref The string name of commit / branch / tag
* @returns The decoded file contents
*/
export async function getFile(
github: Octokit,
owner: string,
repo: string,
path: string,
ref: string
): Promise<string | undefined> {
try {
const response = await github.repos.getContent({
owner,
path,
ref,
repo,
});
// Response theoretically could be a list of files
if (response.data instanceof Array || !('content' in response.data)) {
return undefined;
}
return Buffer.from(response.data.content, 'base64').toString();
} catch (e: any) {
if (e.status === 404) {
return undefined;
}
throw e;
}
}