forked from anuraghazra/github-readme-stats
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: add PAT monitoring functions (anuraghazra#2178)
* feat: add PAT monitoring functions This commit adds two monitoring functions that can be used to check whether the PATs are functioning correctly: - status/up: Returns whether the PATs are rate limited. - status/pat-info: Returns information about the PATs. * feat: add shields.io dynamic badge json response This commit adds the ability to set the return format of the `/api/status/up` cloud function. When this format is set to `shields` a dynamic shields.io badge json is returned. * feat: add 'json' type to up monitor * feat: cleanup status functions * ci: decrease pat-info rate limiting time * feat: decrease monitoring functions rate limits * refactor: pat code * feat: add PAT monitoring functions This commit adds two monitoring functions that can be used to check whether the PATs are functioning correctly: - status/up: Returns whether the PATs are rate limited. - status/pat-info: Returns information about the PATs. * feat: add shields.io dynamic badge json response This commit adds the ability to set the return format of the `/api/status/up` cloud function. When this format is set to `shields` a dynamic shields.io badge json is returned. * feat: add 'json' type to up monitor * feat: cleanup status functions * ci: decrease pat-info rate limiting time * feat: decrease monitoring functions rate limits * refactor: pat code * test: fix pat-info tests * Update api/status/pat-info.js Co-authored-by: Anurag Hazra <hazru.anurag@gmail.com> * test: fix broken tests * chore: fix suspended account * chore: simplify and refactor * chore: fix test * chore: add resetIn field --------- Co-authored-by: Anurag <hazru.anurag@gmail.com>
- Loading branch information
Showing
7 changed files
with
685 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,131 @@ | ||
/** | ||
* @file Contains a simple cloud function that can be used to check which PATs are no | ||
* longer working. It returns a list of valid PATs, expired PATs and PATs with errors. | ||
* | ||
* @description This function is currently rate limited to 1 request per 10 minutes. | ||
*/ | ||
|
||
import { logger, request, dateDiff } from "../../src/common/utils.js"; | ||
export const RATE_LIMIT_SECONDS = 60 * 10; // 1 request per 10 minutes | ||
|
||
/** | ||
* Simple uptime check fetcher for the PATs. | ||
* | ||
* @param {import('axios').AxiosRequestHeaders} variables | ||
* @param {string} token | ||
*/ | ||
const uptimeFetcher = (variables, token) => { | ||
return request( | ||
{ | ||
query: ` | ||
query { | ||
rateLimit { | ||
remaining | ||
resetAt | ||
}, | ||
}`, | ||
variables, | ||
}, | ||
{ | ||
Authorization: `bearer ${token}`, | ||
}, | ||
); | ||
}; | ||
|
||
const getAllPATs = () => { | ||
return Object.keys(process.env).filter((key) => /PAT_\d*$/.exec(key)); | ||
}; | ||
|
||
/** | ||
* Check whether any of the PATs is expired. | ||
*/ | ||
const getPATInfo = async (fetcher, variables) => { | ||
const details = {}; | ||
const PATs = getAllPATs(); | ||
|
||
for (const pat of PATs) { | ||
try { | ||
const response = await fetcher(variables, process.env[pat]); | ||
const errors = response.data.errors; | ||
const hasErrors = Boolean(errors); | ||
const errorType = errors?.[0]?.type; | ||
const isRateLimited = | ||
(hasErrors && errorType === "RATE_LIMITED") || | ||
response.data.data?.rateLimit?.remaining === 0; | ||
|
||
// Store PATs with errors. | ||
if (hasErrors && errorType !== "RATE_LIMITED") { | ||
details[pat] = { | ||
status: "error", | ||
error: { | ||
type: errors[0].type, | ||
message: errors[0].message, | ||
}, | ||
}; | ||
continue; | ||
} else if (isRateLimited) { | ||
const date1 = new Date(); | ||
const date2 = new Date(response.data?.data?.rateLimit?.resetAt); | ||
details[pat] = { | ||
status: "exhausted", | ||
remaining: 0, | ||
resetIn: dateDiff(date2, date1) + " minutes", | ||
}; | ||
} else { | ||
details[pat] = { | ||
status: "valid", | ||
remaining: response.data.data.rateLimit.remaining, | ||
}; | ||
} | ||
} catch (err) { | ||
// Store the PAT if it is expired. | ||
const errorMessage = err.response?.data?.message?.toLowerCase(); | ||
if (errorMessage === "bad credentials") { | ||
details[pat] = { | ||
status: "expired", | ||
}; | ||
} else if (errorMessage === "sorry. your account was suspended.") { | ||
details[pat] = { | ||
status: "suspended", | ||
}; | ||
} else { | ||
throw err; | ||
} | ||
} | ||
} | ||
|
||
const filterPATsByStatus = (status) => { | ||
return Object.keys(details).filter((pat) => details[pat].status === status); | ||
}; | ||
|
||
return { | ||
validPATs: filterPATsByStatus("valid"), | ||
expiredPATs: filterPATsByStatus("expired"), | ||
exhaustedPATS: filterPATsByStatus("exhausted"), | ||
errorPATs: filterPATsByStatus("error"), | ||
details, | ||
}; | ||
}; | ||
|
||
/** | ||
* Cloud function that returns information about the used PATs. | ||
*/ | ||
export default async (_, res) => { | ||
res.setHeader("Content-Type", "application/json"); | ||
try { | ||
// Add header to prevent abuse. | ||
const PATsInfo = await getPATInfo(uptimeFetcher, {}); | ||
if (PATsInfo) { | ||
res.setHeader( | ||
"Cache-Control", | ||
`max-age=0, s-maxage=${RATE_LIMIT_SECONDS}`, | ||
); | ||
} | ||
res.send(JSON.stringify(PATsInfo, null, 2)); | ||
} catch (err) { | ||
// Throw error if something went wrong. | ||
logger.error(err); | ||
res.setHeader("Cache-Control", "no-store"); | ||
res.send("Something went wrong: " + err.message); | ||
} | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,103 @@ | ||
/** | ||
* @file Contains a simple cloud function that can be used to check if the PATs are still | ||
* functional. | ||
* | ||
* @description This function is currently rate limited to 1 request per 10 minutes. | ||
*/ | ||
|
||
import retryer from "../../src/common/retryer.js"; | ||
import { logger, request } from "../../src/common/utils.js"; | ||
|
||
export const RATE_LIMIT_SECONDS = 60 * 10; // 1 request per 10 minutes | ||
|
||
/** | ||
* Simple uptime check fetcher for the PATs. | ||
* | ||
* @param {import('axios').AxiosRequestHeaders} variables | ||
* @param {string} token | ||
*/ | ||
const uptimeFetcher = (variables, token) => { | ||
return request( | ||
{ | ||
query: ` | ||
query { | ||
rateLimit { | ||
remaining | ||
} | ||
} | ||
`, | ||
variables, | ||
}, | ||
{ | ||
Authorization: `bearer ${token}`, | ||
}, | ||
); | ||
}; | ||
|
||
/** | ||
* Creates Json response that can be used for shields.io dynamic card generation. | ||
* | ||
* @param {*} up Whether the PATs are up or not. | ||
* @returns Dynamic shields.io JSON response object. | ||
* | ||
* @see https://shields.io/endpoint. | ||
*/ | ||
const shieldsUptimeBadge = (up) => { | ||
const schemaVersion = 1; | ||
const isError = true; | ||
const label = "Public Instance"; | ||
const message = up ? "up" : "down"; | ||
const color = up ? "brightgreen" : "red"; | ||
return { | ||
schemaVersion, | ||
label, | ||
message, | ||
color, | ||
isError, | ||
}; | ||
}; | ||
|
||
/** | ||
* Cloud function that returns whether the PATs are still functional. | ||
*/ | ||
export default async (req, res) => { | ||
let { type } = req.query; | ||
type = type ? type.toLowerCase() : "boolean"; | ||
|
||
res.setHeader("Content-Type", "application/json"); | ||
|
||
try { | ||
let PATsValid = true; | ||
try { | ||
await retryer(uptimeFetcher, {}); | ||
} catch (err) { | ||
PATsValid = false; | ||
} | ||
|
||
if (PATsValid) { | ||
res.setHeader( | ||
"Cache-Control", | ||
`max-age=0, s-maxage=${RATE_LIMIT_SECONDS}`, | ||
); | ||
} else { | ||
res.setHeader("Cache-Control", "no-store"); | ||
} | ||
|
||
switch (type) { | ||
case "shields": | ||
res.send(shieldsUptimeBadge(PATsValid)); | ||
break; | ||
case "json": | ||
res.send({ up: PATsValid }); | ||
break; | ||
default: | ||
res.send(PATsValid); | ||
break; | ||
} | ||
} catch (err) { | ||
// Return fail boolean if something went wrong. | ||
logger.error(err); | ||
res.setHeader("Cache-Control", "no-store"); | ||
res.send("Something went wrong: " + err.message); | ||
} | ||
}; |
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.