feat(api): complete refactor of openapi routes
Some checks failed
Build and Push Docker Image / tests (push) Failing after 1m4s
Build and Push Docker Image / build (push) Has been skipped

This commit is contained in:
Lars Hampe 2024-11-06 22:20:10 +01:00
parent 95e00816c4
commit f161d6b468
7 changed files with 149 additions and 260 deletions

View File

@ -4,6 +4,10 @@ import { createRoute } from '@hono/zod-openapi'
import { and, eq } from 'drizzle-orm' import { and, eq } from 'drizzle-orm'
import { HTTPException } from 'hono/http-exception' import { HTTPException } from 'hono/http-exception'
import type { changelogApi } from '.'
import { verifyAuthentication } from '../utils/authentication'
import { openApiErrorResponses, openApiSecurity } from '../utils/openapi'
export const route = createRoute({ export const route = createRoute({
method: 'get', method: 'get',
path: '/:id', path: '/:id',
@ -20,39 +24,35 @@ export const route = createRoute({
}, },
description: 'Return changelog by id', description: 'Return changelog by id',
}, },
400: { ...openApiErrorResponses,
description: 'Bad Request',
},
500: {
description: 'Internal Server Error',
},
}, },
...openApiSecurity,
}) })
export const func = async ({ userId, id }: { userId: string; id: string }) => { export const registerChangelogById = (api: typeof changelogApi) => {
const result = await db.query.changelog.findFirst({ return api.openapi(route, async (c) => {
where: and(eq(changelog.userId, userId), eq(changelog.id, id)), const userId = await verifyAuthentication(c)
with: { const { id } = c.req.valid('param')
pages: {
with: { const result = await db.query.changelog.findFirst({
page: true, where: and(eq(changelog.userId, userId), eq(changelog.id, id)),
with: {
pages: {
with: {
page: true,
},
},
versions: {
orderBy: (changelog_version, { desc }) => [
desc(changelog_version.createdAt),
],
}, },
}, },
versions: { })
orderBy: (changelog_version, { desc }) => [
desc(changelog_version.createdAt), if (!result) {
], throw new HTTPException(404, { message: 'Not found' })
}, }
}, return c.json(ChangelogOutput.parse(result), 200)
}) })
if (!result) {
throw new HTTPException(404, { message: 'Not found' })
}
return result
}
export default {
route,
func,
} }

View File

@ -5,6 +5,10 @@ import {
} from '@boring.tools/schema' } from '@boring.tools/schema'
import { createRoute, type z } from '@hono/zod-openapi' import { createRoute, type z } from '@hono/zod-openapi'
import type { changelogApi } from '.'
import { verifyAuthentication } from '../utils/authentication'
import { openApiErrorResponses, openApiSecurity } from '../utils/openapi'
export const route = createRoute({ export const route = createRoute({
method: 'post', method: 'post',
path: '/', path: '/',
@ -23,32 +27,24 @@ export const route = createRoute({
}, },
description: 'Return created changelog', description: 'Return created changelog',
}, },
400: { ...openApiErrorResponses,
description: 'Bad Request',
},
500: {
description: 'Internal Server Error',
},
}, },
...openApiSecurity,
}) })
export const func = async ({ export const registerChangelogCreate = (api: typeof changelogApi) => {
userId, return api.openapi(route, async (c) => {
payload, const userId = await verifyAuthentication(c)
}: { const payload: z.infer<typeof ChangelogCreateInput> = await c.req.json()
userId: string
payload: z.infer<typeof ChangelogCreateInput>
}) => {
return await db
.insert(changelog)
.values({
...payload,
userId: userId,
})
.returning()
}
export default { const [result] = await db
route, .insert(changelog)
func, .values({
...payload,
userId: userId,
})
.returning()
return c.json(ChangelogCreateOutput.parse(result), 201)
})
} }

View File

@ -4,6 +4,10 @@ import { createRoute } from '@hono/zod-openapi'
import { and, eq } from 'drizzle-orm' import { and, eq } from 'drizzle-orm'
import { HTTPException } from 'hono/http-exception' import { HTTPException } from 'hono/http-exception'
import type { changelogApi } from '.'
import { verifyAuthentication } from '../utils/authentication'
import { openApiErrorResponses, openApiSecurity } from '../utils/openapi'
export const route = createRoute({ export const route = createRoute({
method: 'delete', method: 'delete',
path: '/:id', path: '/:id',
@ -17,29 +21,25 @@ export const route = createRoute({
}, },
description: 'Removes a changelog by id', description: 'Removes a changelog by id',
}, },
400: { ...openApiErrorResponses,
description: 'Bad Request',
},
500: {
description: 'Internal Server Error',
},
}, },
...openApiSecurity,
}) })
export const func = async ({ userId, id }: { userId: string; id: string }) => { export const registerChangelogDelete = async (api: typeof changelogApi) => {
const result = await db return api.openapi(route, async (c) => {
.delete(changelog) const userId = await verifyAuthentication(c)
.where(and(eq(changelog.userId, userId), eq(changelog.id, id))) const id = c.req.param('id')
.returning()
if (!result) { const [result] = await db
throw new HTTPException(404, { message: 'Not found' }) .delete(changelog)
} .where(and(eq(changelog.userId, userId), eq(changelog.id, id)))
.returning()
return result if (!result) {
} throw new HTTPException(404, { message: 'Not found' })
}
export default {
route, return c.json(GeneralOutput.parse(result), 200)
func, })
} }

View File

@ -1,135 +1,28 @@
import { OpenAPIHono } from '@hono/zod-openapi' import { OpenAPIHono } from '@hono/zod-openapi'
import { cors } from 'hono/cors' import { cors } from 'hono/cors'
import type { Variables } from '..' import type { Variables } from '..'
import { verifyAuthentication } from '../utils/authentication' import type { ContextModule } from '../utils/sentry'
import { type ContextModule, captureSentry } from '../utils/sentry' import { registerChangelogById } from './byId'
import ById from './byId'
import { changelogCommitApi } from './commit' import { changelogCommitApi } from './commit'
import Create from './create' import { registerChangelogCreate } from './create'
import Delete from './delete' import { registerChangelogDelete } from './delete'
import List from './list' import { registerChangelogList } from './list'
import Update from './update' import { registerChangelogUpdate } from './update'
import version from './version' import version from './version'
const app = new OpenAPIHono<{ Variables: Variables }>() export const changelogApi = new OpenAPIHono<{ Variables: Variables }>()
const module: ContextModule = { const module: ContextModule = {
name: 'changelog', name: 'changelog',
} }
app.use('*', cors()) changelogApi.use('*', cors())
app.route('/commit', changelogCommitApi) changelogApi.route('/commit', changelogCommitApi)
app.route('/version', version) changelogApi.route('/version', version)
app.openapi(ById.route, async (c) => {
const userId = await verifyAuthentication(c)
try {
const id = c.req.param('id')
const result = await ById.func({ userId, id })
return c.json(result, 200)
} catch (error) {
return captureSentry({
c,
error,
module,
user: {
id: userId,
},
})
}
})
app.openapi(List.route, async (c) => { registerChangelogById(changelogApi)
const userId = await verifyAuthentication(c) registerChangelogCreate(changelogApi)
try { registerChangelogDelete(changelogApi)
const result = await List.func({ userId }) registerChangelogUpdate(changelogApi)
return c.json(result, 200) registerChangelogList(changelogApi)
} catch (error) {
return captureSentry({
c,
error,
module,
user: {
id: userId,
},
})
}
})
app.openapi(Create.route, async (c) => { export default changelogApi
const userId = await verifyAuthentication(c)
try {
const [result] = await Create.func({
userId,
payload: await c.req.json(),
})
return c.json(result, 201)
} catch (error) {
return captureSentry({
c,
error,
module,
user: {
id: userId,
},
})
}
})
app.openapi(Delete.route, async (c) => {
const userId = await verifyAuthentication(c)
try {
const id = c.req.param('id')
const result = await Delete.func({ userId, id })
if (result.length === 0) {
return c.json({ message: 'Changelog not found' }, 404)
}
return c.json({ message: 'Changelog removed' })
} catch (error) {
return captureSentry({
c,
error,
module,
user: {
id: userId,
},
})
}
})
app.openapi(Update.route, async (c) => {
const userId = await verifyAuthentication(c)
try {
const id = c.req.param('id')
if (!id) {
return c.json({ message: 'Changelog not found' }, 404)
}
const result = await Update.func({
userId,
payload: await c.req.json(),
id,
})
if (result.length === 0) {
return c.json({ message: 'Changelog not found' }, 404)
}
return c.json(result)
} catch (error) {
return captureSentry({
c,
error,
module,
user: {
id: userId,
},
})
}
})
export default app

View File

@ -3,6 +3,10 @@ import { ChangelogListOutput } from '@boring.tools/schema'
import { createRoute } from '@hono/zod-openapi' import { createRoute } from '@hono/zod-openapi'
import { eq } from 'drizzle-orm' import { eq } from 'drizzle-orm'
import type { changelogApi } from '.'
import { verifyAuthentication } from '../utils/authentication'
import { openApiErrorResponses, openApiSecurity } from '../utils/openapi'
export const route = createRoute({ export const route = createRoute({
method: 'get', method: 'get',
path: '/', path: '/',
@ -16,40 +20,37 @@ export const route = createRoute({
}, },
description: 'Return changelogs for current user', description: 'Return changelogs for current user',
}, },
400: { ...openApiErrorResponses,
description: 'Bad Request',
},
500: {
description: 'Internal Server Error',
},
}, },
...openApiSecurity,
}) })
export const func = async ({ userId }: { userId: string }) => { export const registerChangelogList = (api: typeof changelogApi) => {
const result = await db.query.changelog.findMany({ return api.openapi(route, async (c) => {
where: eq(changelog.userId, userId), const userId = await verifyAuthentication(c)
with: {
versions: true,
commits: {
columns: { id: true },
},
},
orderBy: (changelog, { asc }) => [asc(changelog.createdAt)],
})
return result.map((changelog) => { const result = await db.query.changelog.findMany({
const { versions, commits, ...rest } = changelog where: eq(changelog.userId, userId),
return { with: {
...rest, versions: true,
computed: { commits: {
versionCount: versions.length, columns: { id: true },
commitCount: commits.length, },
}, },
} orderBy: (changelog, { asc }) => [asc(changelog.createdAt)],
})
const mappedData = result.map((changelog) => {
const { versions, commits, ...rest } = changelog
return {
...rest,
computed: {
versionCount: versions.length,
commitCount: commits.length,
},
}
})
return c.json(ChangelogListOutput.parse(mappedData), 200)
}) })
} }
export default {
route,
func,
}

View File

@ -6,6 +6,10 @@ import {
import { createRoute, type z } from '@hono/zod-openapi' import { createRoute, type z } from '@hono/zod-openapi'
import { and, eq } from 'drizzle-orm' import { and, eq } from 'drizzle-orm'
import { HTTPException } from 'hono/http-exception' import { HTTPException } from 'hono/http-exception'
import type { changelogApi } from '.'
import { verifyAuthentication } from '../utils/authentication'
import { openApiErrorResponses, openApiSecurity } from '../utils/openapi'
import { redis } from '../utils/redis' import { redis } from '../utils/redis'
export const route = createRoute({ export const route = createRoute({
@ -26,43 +30,32 @@ export const route = createRoute({
}, },
description: 'Return updated changelog', description: 'Return updated changelog',
}, },
400: { ...openApiErrorResponses,
description: 'Bad Request',
},
500: {
description: 'Internal Server Error',
},
}, },
...openApiSecurity,
}) })
export const func = async ({ export const registerChangelogUpdate = (api: typeof changelogApi) => {
userId, return api.openapi(route, async (c) => {
payload, const userId = await verifyAuthentication(c)
id, const id = c.req.param('id')
}: { const payload: z.infer<typeof ChangelogUpdateInput> = await c.req.json()
userId: string
payload: z.infer<typeof ChangelogUpdateInput>
id: string
}) => {
const [result] = await db
.update(changelog)
.set({
...payload,
})
.where(and(eq(changelog.id, id), eq(changelog.userId, userId)))
.returning()
if (!result) { const [result] = await db
throw new HTTPException(404, { message: 'Not found' }) .update(changelog)
} .set({
...payload,
})
.where(and(eq(changelog.id, id), eq(changelog.userId, userId)))
.returning()
if (result.pageId) { if (!result) {
redis.del(result.pageId) throw new HTTPException(404, { message: 'Not found' })
} }
return result
} if (result.pageId) {
redis.del(result.pageId)
export default { }
route, return c.json(ChangelogUpdateOutput.parse(result), 200)
func, })
} }

View File

@ -13,6 +13,12 @@ export const ChangelogOutput = z
example: 'This is a changelog', example: 'This is a changelog',
}), }),
versions: z.array(VersionOutput).optional(), versions: z.array(VersionOutput).optional(),
isSemver: z.boolean().openapi({
example: true,
}),
isConventional: z.boolean().openapi({
example: true,
}),
computed: z computed: z
.object({ .object({
versionCount: z.number().openapi({ versionCount: z.number().openapi({