badgen.net/libs/create-badgen-handler-next.ts

110 wiersze
3.1 KiB
TypeScript
Czysty Zwykły widok Historia

import http from 'http'
2023-03-18 07:37:36 +00:00
// import { measure } from 'measurement-protocol'
import matchRoute from 'my-way'
import { serveBadgeNext } from './serve-badge-next'
import serveDoc from './serve-doc'
import sentry from './sentry'
import type { NextApiRequest, NextApiResponse } from 'next'
import type { BadgenParams } from './types'
import { HTTPError } from 'got'
export type PathArgs = NonNullable<ReturnType<typeof matchRoute>>
2023-02-05 11:20:03 +00:00
export type BadgenResult = Promise<BadgenParams>
export interface BadgenServeConfig {
title: string;
help?: string;
examples: { [url: string]: string };
2023-02-05 12:42:32 +00:00
handlers: { [pattern: string]: (pathArgs: PathArgs) => BadgenResult };
}
export function createBadgenHandler (badgenServerConfig: BadgenServeConfig) {
2023-02-05 11:20:03 +00:00
const { handlers, title, help, examples } = badgenServerConfig
async function nextHandler (req: NextApiRequest, res: NextApiResponse) {
let { pathname } = new URL(req.url || '/', `http://${req.headers.host}`)
if (pathname === '/favicon.ico') {
return res.end()
}
// Match badge handlers
let matchedArgs: PathArgs | null = null
const matchedScheme = Object.keys(handlers).find(scheme => {
return matchedArgs = matchRoute(scheme, decodeURI(pathname))
})
// Invoke badge handler
if (matchedArgs !== null && matchedScheme !== undefined) {
return await handlers[matchedScheme](matchedArgs).then(params => {
return serveBadgeNext(req, res, { params })
}).catch(error => {
const meta = { matchedArgs, matchedScheme }
return onBadgeHandlerError(meta, error, req, res)
})
}
if (matchRoute('/:name', pathname)) {
return serveDoc(badgenServerConfig)(req, res)
}
res.status(404).end()
}
2023-02-05 11:20:03 +00:00
nextHandler.meta = { title, examples, help, handlers }
return nextHandler
}
function onBadgeHandlerError (meta: any, err: Error | HTTPError, req: NextApiRequest, res: NextApiResponse) {
2023-02-05 12:42:32 +00:00
sentry.captureException(err)
console.error('BADGE_HANDLER_ERROR', err.message, req.url)
2023-02-05 12:42:32 +00:00
// Send user friendly response
const errorBadgeParams = {
subject: 'error',
status: '500',
color: 'red',
}
2023-02-05 12:42:32 +00:00
if (err instanceof HTTPError) {
errorBadgeParams.status = err.response.statusCode.toString()
}
if (err instanceof BadgenError) {
errorBadgeParams.status = err.status
}
res.setHeader('Error-Message', err.message)
return serveBadgeNext(req, res, {
code: 200,
params: errorBadgeParams,
})
}
function getBadgeStyle (req: http.IncomingMessage): string | undefined {
const host = req.headers['x-forwarded-host']?.toString() ?? req.headers.host ?? ''
return host.startsWith('flat') ? 'flat' : undefined
}
function simpleDecode (str: string): string {
return String(str).replace(/%2F/g, '/')
}
2023-06-10 01:25:28 +00:00
export class BadgenError {
public status: string // error badge param: status (required)
public color: string // error badge param: color
public code: number // status code for response
public message: string
2023-06-10 01:25:28 +00:00
constructor ({ status, color = 'grey', code = 500, message = '' }) {
2023-06-10 01:25:28 +00:00
this.status = status
this.color = color
this.code = code
this.message = message
2023-06-10 01:25:28 +00:00
}
}