* First run of script * Get the app running --- ish * Get NextJS working * Remove `node:` * Get more tests passing in unit directory * Update FailBot test to use nock * Update test.yml * Update Dockerfile * tests/content fixes * Update page.js * Update build-changelog.js * updating tests/routing * Update orphan-tests.js * updating tests/rendering * Update .eslintrc.js * Update .eslintrc.js * Install jest/globals * "linting" tests * staging update to server.mjs * Change '.github/allowed-actions.js' to a ESM export * Lint * Fixes for the main package.json * Move Jest to be last in the npm test command so we can pass args * Just use 'npm run lint' in the npm test command * update algolia label script * update openapi script * update require on openapi * Update enterprise-algolia-label.js * forgot JSON.parse * Update lunr-search-index.js * Always explicitly include process.cwd() for JSON file reads pathed from project root * update graphql/update-files.js script * Update other npm scripts using jest to pass ESM NODE_OPTIONS * Update check-for-enterprise-issues-by-label.js for ESM * Update create-enterprise-issue.js for ESM * Import jest global for browser tests * Convert 'script/deploy' to ESM Co-authored-by: Grace Park <gracepark@github.com> Co-authored-by: James M. Greene <jamesmgreene@github.com>
93 lines
2.4 KiB
JavaScript
93 lines
2.4 KiB
JavaScript
import fetch from 'node-fetch'
|
|
|
|
export default class FailBot {
|
|
constructor ({ app, haystackURL, headers }) {
|
|
this.app = app
|
|
this.haystackURL = haystackURL
|
|
this.headers = headers
|
|
}
|
|
|
|
/**
|
|
* Report an error to Sentry
|
|
* @param {Error} error
|
|
* @param {any} metadata
|
|
* @param {any} [headers]
|
|
*/
|
|
static async report (error, metadata, headers = {}) {
|
|
// If there's no HAYSTACK_URL set, bail early
|
|
if (!process.env.HAYSTACK_URL) return
|
|
|
|
const failbot = new FailBot({
|
|
app: 'docs',
|
|
haystackURL: process.env.HAYSTACK_URL,
|
|
headers
|
|
})
|
|
|
|
return failbot.sendException(error, metadata)
|
|
}
|
|
|
|
/**
|
|
* Create a rollup of this error by generating a base64 representation
|
|
* @param {Error} error
|
|
*/
|
|
createRollup (error) {
|
|
const stackLine = error.stack && error.stack.split('\n')[1]
|
|
const str = `${error.name}:${stackLine}`.replace(/=/g, '')
|
|
return Buffer.from(str).toString('base64')
|
|
}
|
|
|
|
/**
|
|
* Format the error to a plain JSON object with additional data
|
|
* @param {Error} error
|
|
* @param {any} metadata
|
|
*/
|
|
formatJSON (error, metadata) {
|
|
return Object.assign({}, metadata, {
|
|
/* eslint-disable camelcase */
|
|
created_at: new Date().toISOString(),
|
|
rollup: this.createRollup(error),
|
|
class: error.name,
|
|
message: error.message,
|
|
backtrace: error.stack || '',
|
|
js_environment: `Node.js ${process.version}`
|
|
/* eslint-enable camelcase */
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Populate default context from settings. Since settings commonly comes from
|
|
* ENV, this allows setting defaults for the context via the environment.
|
|
*/
|
|
getFailbotContext () {
|
|
const failbotKeys = {}
|
|
|
|
for (const key in process.env) {
|
|
if (key.startsWith('FAILBOT_CONTEXT_')) {
|
|
const formattedKey = key.replace(/^FAILBOT_CONTEXT_/, '').toLowerCase()
|
|
failbotKeys[formattedKey] = process.env[key]
|
|
}
|
|
}
|
|
|
|
return failbotKeys
|
|
}
|
|
|
|
/**
|
|
* Send the error to Sentry
|
|
* @param {Error} error
|
|
* @param {any} metadata
|
|
*/
|
|
async sendException (error, metadata = {}) {
|
|
const data = Object.assign({ app: this.app }, this.getFailbotContext(), metadata)
|
|
const body = this.formatJSON(error, Object.assign({ app: this.app }, data))
|
|
|
|
return fetch(this.haystackURL, {
|
|
method: 'POST',
|
|
body: JSON.stringify(body),
|
|
headers: {
|
|
...this.headers,
|
|
'Content-Type': 'application/json'
|
|
}
|
|
})
|
|
}
|
|
}
|