-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcircleci.ts
254 lines (227 loc) · 7.62 KB
/
circleci.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
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
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
export type TrendsOrMetrics = {
total_credits_used: number
total_duration_secs: number
throughput: number
total_runs: number
success_rate: number
}
export type Project = {
project_name: string
}
export type TrendsAndMetrics = {
trends: TrendsOrMetrics
metrics: TrendsOrMetrics
}
export type OrgSummaryData = {
org_data: TrendsAndMetrics
all_projects: string[]
org_project_data: (Project & TrendsAndMetrics)[]
}
export enum ReportingWindow {
LAST_90_DAYS = "last-90-days",
LAST_60_DAYS = "last-60-days",
LAST_30_DAYS = "last-30-days",
LAST_7_DAYS = "last-7-days",
LAST_24_HOURS = "last-24-hours"
}
export type Workflow = {
workflow_name: string
}
export type Branch = {
branch: string
}
export type ProjectWorkflowsPageData = {
project_workflow_branch_data: (Workflow & Branch & TrendsAndMetrics)[]
all_workflows: string[]
org_id: string
all_branches: string[]
project_workflow_data: (Workflow & TrendsAndMetrics)[]
project_id: string
project_data: TrendsAndMetrics
}
export type Paged<T> = {
items: T[]
next_page_token?: string
}
export type Metrics = {
name: string
metrics: {
total_runs: number
successful_runs: number
mttr: number
total_credits_used: number
failed_runs: number
median_credits_used: number
success_rate: number
duration_metrics: {
min: number
mean: number
median: number
p95: number
max: number
standard_deviation: number
total_duration: number
}
total_recoveries: number
throughput: number
}
window_start: string
window_end: string
}
export type ProjectWorkflowMetrics = {
project_id: string
} & Metrics
export type ProjectWorkflowRuns = {
id: string
duration: number
status: string
created_at: string
stopped_at: string
credits_used: number
branch: string
is_approval: boolean
}
export type ProjectWorkflowJobMetrics = Metrics
export type WorkflowJob = {
dependencies: string[]
job_number: number
id: string
started_at: string
name: string
project_slug: string
status: string
type: string
stopped_at: string
}
export type JobDetails = {
created_at: string
duration: number
executor: {
resource_class: string
type: string
}
messages: string[]
queued_at: string
started_at: string
parallel_runs: {
index: number
status: string
}[]
contexts: string[]
latest_workflow: {
id: string
name: string
}
name: string
number: number
organization: {
name: string
}
parallelism: number
pipeline: {
id: string
}
project: {
external_url: string
id: string
name: string
slug: string
}
web_url: string
}
export type ProjectBySlug = {
slug: string
name: string
id: string
organization_name: string
organization_slug: string
organization_id: string
vcs_info: {
vcs_url: string
provider: string
default_branch: string
}
}
export class Client {
token: string
debug: boolean
constructor(token: string, debug: boolean) {
this.token = token
this.debug = debug
}
async getJSON<T>(url: string, params: URLSearchParams = new URLSearchParams()): Promise<T> {
if (this.debug) {
console.log(`GET ${url}?${params}`)
}
const res = await fetch(`${url}?${params}`, {
headers: {
"Circle-Token": this.token
}
})
const json = await res.json()
if (this.debug) {
console.log(JSON.stringify(json, null, 2))
}
return json
}
async getPagedJSON<T>(url: string, params: URLSearchParams = new URLSearchParams()): Promise<T[]> {
const res: T[] = []
let page: Paged<T> | undefined
do {
if (page?.next_page_token) {
params.set("page-token", page?.next_page_token)
}
page = await this.getJSON<Paged<T>>(url, params)
res.push(...page.items)
} while (page?.next_page_token)
return res
}
// https://circleci.com/docs/api/v2/index.html#operation/getOrgSummaryData
getOrgSummaryData(org: string, reportingWindow: ReportingWindow = ReportingWindow.LAST_90_DAYS): Promise<OrgSummaryData> {
return this.getJSON(`https://circleci.com/api/v2/insights/gh/${org}/summary`, new URLSearchParams({"reporting-window": reportingWindow}))
}
// https://circleci.com/docs/api/v2/index.html#operation/getProjectWorkflowsPageData
getProjectWorkflowsPageData(org: string, repo: string, reportingWindow: ReportingWindow = ReportingWindow.LAST_90_DAYS): Promise<ProjectWorkflowsPageData> {
return this.getJSON(`https://circleci.com/api/v2/insights/pages/gh/${org}/${repo}/summary`, new URLSearchParams({"reporting-window": reportingWindow}))
}
// https://circleci.com/docs/api/v2/index.html#operation/getProjectWorkflowMetrics
getProjectWorkflowMetrics(org: string, repo: string, allBranches = false, reportingWindow: ReportingWindow = ReportingWindow.LAST_90_DAYS): Promise<ProjectWorkflowMetrics[]> {
return this.getPagedJSON(`https://circleci.com/api/v2/insights/gh/${org}/${repo}/workflows`, new URLSearchParams({"reporting-window": reportingWindow, "all-branches": allBranches.toString()}))
}
// https://circleci.com/docs/api/v2/index.html#operation/getProjectWorkflowRuns
async getProjectWorkflowRuns(org: string, repo: string, workflow: string, allBranches = false, reportingWindow: ReportingWindow = ReportingWindow.LAST_90_DAYS): Promise<ProjectWorkflowRuns[]> {
const [startDate, endDate] = ((reportingWindow: ReportingWindow) => {
const now = Date.now()
switch (reportingWindow) {
case ReportingWindow.LAST_90_DAYS:
return [new Date(now - 90 * 24 * 60 * 60 * 1000).toISOString(), new Date(now).toISOString()]
case ReportingWindow.LAST_60_DAYS:
return [new Date(now - 60 * 24 * 60 * 60 * 1000).toISOString(), new Date(now).toISOString()]
case ReportingWindow.LAST_30_DAYS:
return [new Date(now - 30 * 24 * 60 * 60 * 1000).toISOString(), new Date(now).toISOString()]
case ReportingWindow.LAST_7_DAYS:
return [new Date(now - 7 * 24 * 60 * 60 * 1000).toISOString(), new Date(now).toISOString()]
case ReportingWindow.LAST_24_HOURS:
return [new Date(now - 24 * 60 * 60 * 1000).toISOString(), new Date(now).toISOString()]
}
})(reportingWindow)
const json = await this.getJSON<Paged<ProjectWorkflowRuns>>(`https://circleci.com/api/v2/insights/gh/${org}/${repo}/workflows/${workflow}`, new URLSearchParams({"all-branches": `${allBranches}`, "start-date": startDate, "end-date": endDate}))
return json.items
}
// https://circleci.com/docs/api/v2/index.html#operation/getProjectWorkflowJobMetrics
getProjectWorkflowJobMetrics(org: string, repo: string, workflow: string, allBranches = false, reportingWindow: ReportingWindow = ReportingWindow.LAST_90_DAYS): Promise<ProjectWorkflowJobMetrics[]> {
return this.getPagedJSON(`https://circleci.com/api/v2/insights/gh/${org}/${repo}/workflows/${workflow}/jobs`, new URLSearchParams({"reporting-window": reportingWindow, "all-branches": allBranches.toString()}))
}
// https://circleci.com/docs/api/v2/index.html#operation/listWorkflowJobs
listWorkflowJobs(id: string): Promise<WorkflowJob[]> {
return this.getPagedJSON(`https://circleci.com/api/v2/workflow/${id}/job`)
}
// https://circleci.com/docs/api/v2/index.html#operation/getJobDetails
getJobDetails(org: string, repo: string, number: string): Promise<JobDetails> {
return this.getJSON(`https://circleci.com/api/v2/project/gh/${org}/${repo}/job/${number}`)
}
// https://circleci.com/docs/api/v2/index.html#operation/getProjectBySlug
getProjectBySlug(org: string, repo: string): Promise<ProjectBySlug> {
return this.getJSON(`https://circleci.com/api/v2/project/gh/${org}/${repo}`)
}
}