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

129 wiersze
3.6 KiB
TypeScript

import http from 'http'
// 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>>
export type BadgenResult = Promise<BadgenParams>
export interface BadgenServeConfig {
title: string;
help?: string;
examples: { [url: string]: string };
handlers: { [pattern: string]: (pathArgs: PathArgs) => BadgenResult };
}
export function createBadgenHandler (badgenServerConfig: BadgenServeConfig) {
const { handlers, title, help, examples } = badgenServerConfig
async function nextHandler (req: NextApiRequest, res: NextApiResponse) {
let { pathname } = new URL(req.url || '/', `http://${req.headers.host}`)
measurementLogInvocation(req.headers?.host ?? 'badgen.net', pathname)
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)
// return res.send('TODO: serve doc page')
}
return res.status(404).end()
}
nextHandler.meta = { title, examples, help, handlers }
return nextHandler
}
function onBadgeHandlerError (meta: any, err: Error | HTTPError, req: NextApiRequest, res: NextApiResponse) {
sentry.captureException(err)
console.error('BADGE_HANDLER_ERROR', err.message, req.url)
// Send user friendly response
const errorBadgeParams = {
subject: 'error',
status: '500',
color: 'red',
}
if (err instanceof HTTPError) {
errorBadgeParams.status = err.response.statusCode.toString()
}
res.setHeader('Error-Message', err.message)
return serveBadgeNext(req, res, {
code: 200,
params: errorBadgeParams,
})
}
type MeasurementProtocolEvent = {
name: string;
params: Record<string, any>;
}
function measure (clientId: string, events: MeasurementProtocolEvent[]) {
const { GA_MEASUREMENT_ID, GA_API_SECRET } = process.env
if (!GA_MEASUREMENT_ID || !GA_API_SECRET) return
const searchParams = `measurement_id=${GA_MEASUREMENT_ID}&api_secret=${GA_API_SECRET}`
fetch(`https://www.google-analytics.com/mp/collect?${searchParams}`, {
method: "POST",
body: JSON.stringify({
client_id: clientId,
events,
})
})
}
function measurementLogInvocation (host: string, pathname: string) {
const { VERCEL_REGION = '0000' } = process.env
measure(VERCEL_REGION, [{
name: 'invocation',
params: {
host,
pathname,
name: pathname.split('/')[1]
}
}])
}
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, '/')
}