diff --git a/apps/api/src/access-token/create.ts b/apps/api/src/access-token/create.ts index d4ac959..11dbd44 100644 --- a/apps/api/src/access-token/create.ts +++ b/apps/api/src/access-token/create.ts @@ -3,8 +3,10 @@ import { access_token, db } from '@boring.tools/database' import { AccessTokenCreateInput, AccessTokenOutput } from '@boring.tools/schema' import { createRoute, type z } from '@hono/zod-openapi' import { HTTPException } from 'hono/http-exception' + import type { accessTokenApi } from '.' import { verifyAuthentication } from '../utils/authentication' +import { openApiErrorResponses, openApiSecurity } from '../utils/openapi' export const route = createRoute({ method: 'post', @@ -24,13 +26,9 @@ export const route = createRoute({ }, description: 'Commits created', }, - 400: { - description: 'Bad Request', - }, - 500: { - description: 'Internal Server Error', - }, + ...openApiErrorResponses, }, + ...openApiSecurity, }) export const registerAccessTokenCreate = (api: typeof accessTokenApi) => { @@ -53,6 +51,6 @@ export const registerAccessTokenCreate = (api: typeof accessTokenApi) => { throw new HTTPException(404, { message: 'Not Found' }) } - return c.json(result, 200) + return c.json(AccessTokenOutput.parse(result), 201) }) } diff --git a/apps/api/src/access-token/delete.ts b/apps/api/src/access-token/delete.ts index 10b209b..3cd0585 100644 --- a/apps/api/src/access-token/delete.ts +++ b/apps/api/src/access-token/delete.ts @@ -3,27 +3,25 @@ import { GeneralOutput } from '@boring.tools/schema' import { createRoute, type z } from '@hono/zod-openapi' import { and, eq } from 'drizzle-orm' import { HTTPException } from 'hono/http-exception' + import type { accessTokenApi } from '.' import { verifyAuthentication } from '../utils/authentication' +import { openApiErrorResponses, openApiSecurity } from '../utils/openapi' export const route = createRoute({ method: 'delete', path: '/:id', tags: ['access-token'], responses: { - 201: { + 200: { content: { 'application/json': { schema: GeneralOutput }, }, description: 'Removes a access token by id', }, - 400: { - description: 'Bad Request', - }, - 500: { - description: 'Internal Server Error', - }, + ...openApiErrorResponses, }, + ...openApiSecurity, }) export const registerAccessTokenDelete = (api: typeof accessTokenApi) => { @@ -40,6 +38,6 @@ export const registerAccessTokenDelete = (api: typeof accessTokenApi) => { throw new HTTPException(404, { message: 'Not Found' }) } - return c.json(result, 200) + return c.json(GeneralOutput.parse({ message: 'Access token deleted' }), 200) }) } diff --git a/apps/api/src/access-token/list.ts b/apps/api/src/access-token/list.ts index 8da4bf3..8f1f306 100644 --- a/apps/api/src/access-token/list.ts +++ b/apps/api/src/access-token/list.ts @@ -3,8 +3,10 @@ import { AccessTokenListOutput } from '@boring.tools/schema' import { createRoute } from '@hono/zod-openapi' import { eq } from 'drizzle-orm' import { HTTPException } from 'hono/http-exception' + import type { accessTokenApi } from '.' import { verifyAuthentication } from '../utils/authentication' +import { openApiErrorResponses, openApiSecurity } from '../utils/openapi' const route = createRoute({ method: 'get', @@ -19,13 +21,9 @@ const route = createRoute({ }, description: 'Return version by id', }, - 400: { - description: 'Bad Request', - }, - 500: { - description: 'Internal Server Error', - }, + ...openApiErrorResponses, }, + ...openApiSecurity, }) export const registerAccessTokenList = (api: typeof accessTokenApi) => { @@ -44,6 +42,6 @@ export const registerAccessTokenList = (api: typeof accessTokenApi) => { token: at.token.substring(0, 10), })) - return c.json(mappedData, 200) + return c.json(AccessTokenListOutput.parse(mappedData), 200) }) } diff --git a/apps/api/src/changelog/commit/create.ts b/apps/api/src/changelog/commit/create.ts index 5c506ad..e7e9247 100644 --- a/apps/api/src/changelog/commit/create.ts +++ b/apps/api/src/changelog/commit/create.ts @@ -3,8 +3,10 @@ import { CommitCreateInput, CommitCreateOutput } from '@boring.tools/schema' import { createRoute, type z } from '@hono/zod-openapi' import { and, eq } from 'drizzle-orm' import { HTTPException } from 'hono/http-exception' + import type { changelogCommitApi } from '.' import { verifyAuthentication } from '../../utils/authentication' +import { openApiErrorResponses, openApiSecurity } from '../../utils/openapi' export const route = createRoute({ method: 'post', @@ -24,13 +26,9 @@ export const route = createRoute({ }, description: 'Commits created', }, - 400: { - description: 'Bad Request', - }, - 500: { - description: 'Internal Server Error', - }, + ...openApiErrorResponses, }, + ...openApiSecurity, }) export const registerCommitCreate = (api: typeof changelogCommitApi) => { @@ -64,6 +62,6 @@ export const registerCommitCreate = (api: typeof changelogCommitApi) => { throw new HTTPException(404, { message: 'Not Found' }) } - return c.json(result, 200) + return c.json(CommitCreateOutput.parse(result), 201) }) } diff --git a/apps/api/src/changelog/commit/list.ts b/apps/api/src/changelog/commit/list.ts index 64f29a1..20f46fa 100644 --- a/apps/api/src/changelog/commit/list.ts +++ b/apps/api/src/changelog/commit/list.ts @@ -4,8 +4,10 @@ import { createRoute } from '@hono/zod-openapi' import { and, eq, isNotNull, isNull } from 'drizzle-orm' import { HTTPException } from 'hono/http-exception' import { P, match } from 'ts-pattern' + import type { changelogCommitApi } from '.' import { verifyAuthentication } from '../../utils/authentication' +import { openApiErrorResponses, openApiSecurity } from '../../utils/openapi' const route = createRoute({ method: 'get', @@ -23,13 +25,9 @@ const route = createRoute({ }, description: 'Return version by id', }, - 400: { - description: 'Bad Request', - }, - 500: { - description: 'Internal Server Error', - }, + ...openApiErrorResponses, }, + ...openApiSecurity, }) export const registerCommitList = (api: typeof changelogCommitApi) => { @@ -80,6 +78,6 @@ export const registerCommitList = (api: typeof changelogCommitApi) => { throw new HTTPException(404, { message: 'Not Found' }) } - return c.json(commits, 200) + return c.json(CommitListOutput.parse(commits), 200) }) } diff --git a/apps/api/src/index.ts b/apps/api/src/index.ts index 38d393d..7e49d94 100644 --- a/apps/api/src/index.ts +++ b/apps/api/src/index.ts @@ -22,6 +22,7 @@ export type Variables = { export const app = new OpenAPIHono<{ Variables: Variables }>({ defaultHook: handleZodError, + strict: false, }) // app.use( @@ -33,6 +34,14 @@ export const app = new OpenAPIHono<{ Variables: Variables }>({ app.onError(handleError) app.use('*', cors()) app.use('/v1/*', authentication) +app.openAPIRegistry.registerComponent('securitySchemes', 'AccessToken', { + type: 'http', + scheme: 'bearer', +}) +app.openAPIRegistry.registerComponent('securitySchemes', 'Clerk', { + type: 'http', + scheme: 'bearer', +}) app.route('/v1/user', user) app.route('/v1/changelog', changelog) diff --git a/apps/api/src/page/byId.ts b/apps/api/src/page/byId.ts index 5a8bcbf..f1659df 100644 --- a/apps/api/src/page/byId.ts +++ b/apps/api/src/page/byId.ts @@ -5,6 +5,7 @@ import { and, eq } from 'drizzle-orm' import { HTTPException } from 'hono/http-exception' import { verifyAuthentication } from '../utils/authentication' +import { openApiErrorResponses, openApiSecurity } from '../utils/openapi' import type { pageApi } from './index' const route = createRoute({ @@ -24,13 +25,9 @@ const route = createRoute({ }, description: 'Return changelog by id', }, - 400: { - description: 'Bad Request', - }, - 500: { - description: 'Internal Server Error', - }, + ...openApiErrorResponses, }, + ...openApiSecurity, }) export const registerPageById = (api: typeof pageApi) => { @@ -60,6 +57,6 @@ export const registerPageById = (api: typeof pageApi) => { changelogs: changelogs.map((log) => log.changelog), } - return c.json(mappedResult, 200) + return c.json(PageOutput.parse(mappedResult), 200) }) } diff --git a/apps/api/src/page/create.ts b/apps/api/src/page/create.ts index 08022b4..26c32fc 100644 --- a/apps/api/src/page/create.ts +++ b/apps/api/src/page/create.ts @@ -4,6 +4,7 @@ import { createRoute, type z } from '@hono/zod-openapi' import { HTTPException } from 'hono/http-exception' import { verifyAuthentication } from '../utils/authentication' +import { openApiErrorResponses, openApiSecurity } from '../utils/openapi' import type { pageApi } from './index' const route = createRoute({ @@ -27,13 +28,9 @@ const route = createRoute({ }, description: 'Return changelog by id', }, - 400: { - description: 'Bad Request', - }, - 500: { - description: 'Internal Server Error', - }, + ...openApiErrorResponses, }, + ...openApiSecurity, }) export const registerPageCreate = (api: typeof pageApi) => { @@ -42,6 +39,7 @@ export const registerPageCreate = (api: typeof pageApi) => { const { changelogIds, ...rest }: z.infer = await c.req.json() + const [result] = await db .insert(page) .values({ @@ -50,6 +48,10 @@ export const registerPageCreate = (api: typeof pageApi) => { }) .returning() + if (!result) { + throw new HTTPException(404, { message: 'Not Found' }) + } + // TODO: implement transaction if (changelogIds.length > 0) { await db.insert(changelogs_to_pages).values( @@ -59,10 +61,7 @@ export const registerPageCreate = (api: typeof pageApi) => { })), ) } - if (!result) { - throw new HTTPException(404, { message: 'Not Found' }) - } - return c.json(result, 200) + return c.json(PageOutput.parse(result), 200) }) } diff --git a/apps/api/src/page/delete.ts b/apps/api/src/page/delete.ts index eb0fadf..456a737 100644 --- a/apps/api/src/page/delete.ts +++ b/apps/api/src/page/delete.ts @@ -3,8 +3,10 @@ import { GeneralOutput, PageByIdParams } from '@boring.tools/schema' import { createRoute } from '@hono/zod-openapi' import { and, eq } from 'drizzle-orm' import { HTTPException } from 'hono/http-exception' + import type { pageApi } from '.' import { verifyAuthentication } from '../utils/authentication' +import { openApiErrorResponses, openApiSecurity } from '../utils/openapi' const route = createRoute({ method: 'delete', @@ -23,27 +25,25 @@ const route = createRoute({ }, description: 'Removes a changelog by id', }, - 400: { - description: 'Bad Request', - }, - 500: { - description: 'Internal Server Error', - }, + ...openApiErrorResponses, }, + ...openApiSecurity, }) export const registerPageDelete = (api: typeof pageApi) => { return api.openapi(route, async (c) => { const userId = await verifyAuthentication(c) const { id } = c.req.valid('param') + const result = await db .delete(page) .where(and(eq(page.userId, userId), eq(page.id, id))) .returning() + if (!result) { throw new HTTPException(404, { message: 'Not Found' }) } - return c.json(result, 200) + return c.json({}, 200) }) } diff --git a/apps/api/src/page/index.ts b/apps/api/src/page/index.ts index 2b7c135..89fe4b6 100644 --- a/apps/api/src/page/index.ts +++ b/apps/api/src/page/index.ts @@ -1,6 +1,7 @@ import { OpenAPIHono } from '@hono/zod-openapi' import { timing } from 'hono/timing' import type { Variables } from '..' +import { handleZodError } from '../utils/errors' import type { ContextModule } from '../utils/sentry' import { registerPageById } from './byId' import { registerPageCreate } from './create' @@ -9,7 +10,9 @@ import { registerPageList } from './list' import { registerPagePublic } from './public' import { registerPageUpdate } from './update' -export const pageApi = new OpenAPIHono<{ Variables: Variables }>() +export const pageApi = new OpenAPIHono<{ Variables: Variables }>({ + defaultHook: handleZodError, +}) pageApi.use('*', timing()) const module: ContextModule = { name: 'page', diff --git a/apps/api/src/page/list.ts b/apps/api/src/page/list.ts index 4e1be99..8722ccb 100644 --- a/apps/api/src/page/list.ts +++ b/apps/api/src/page/list.ts @@ -1,16 +1,17 @@ import { db, page } from '@boring.tools/database' +import { PageListOutput } from '@boring.tools/schema' import { createRoute } from '@hono/zod-openapi' import { and, eq } from 'drizzle-orm' - -import { PageListOutput } from '@boring.tools/schema' import { HTTPException } from 'hono/http-exception' + import { verifyAuthentication } from '../utils/authentication' +import { openApiErrorResponses, openApiSecurity } from '../utils/openapi' import type { pageApi } from './index' const route = createRoute({ method: 'get', tags: ['page'], - description: 'Get a page list', + description: 'Get a list of pages', path: '/', responses: { 200: { @@ -21,13 +22,9 @@ const route = createRoute({ }, description: 'Return changelog by id', }, - 400: { - description: 'Bad Request', - }, - 500: { - description: 'Internal Server Error', - }, + ...openApiErrorResponses, }, + ...openApiSecurity, }) export const registerPageList = (api: typeof pageApi) => { @@ -42,6 +39,6 @@ export const registerPageList = (api: typeof pageApi) => { throw new HTTPException(404, { message: 'Not Found' }) } - return c.json(result, 200) + return c.json(PageListOutput.parse(result), 200) }) } diff --git a/apps/api/src/page/public.ts b/apps/api/src/page/public.ts index 761abc1..2aabd2d 100644 --- a/apps/api/src/page/public.ts +++ b/apps/api/src/page/public.ts @@ -1,18 +1,18 @@ import { changelog_version, db, page } from '@boring.tools/database' +import { PagePublicOutput, PagePublicParams } from '@boring.tools/schema' import { createRoute } from '@hono/zod-openapi' import { eq } from 'drizzle-orm' -import { endTime, setMetric, startTime } from 'hono/timing' - -import { PagePublicOutput, PagePublicParams } from '@boring.tools/schema' import { HTTPException } from 'hono/http-exception' +import { endTime, startTime } from 'hono/timing' +import { openApiErrorResponses } from '../utils/openapi' import { redis } from '../utils/redis' import type { pageApi } from './index' const route = createRoute({ method: 'get', tags: ['page'], - description: 'Get a page', + description: 'Get a page by id for public view', path: '/:id/public', request: { params: PagePublicParams, @@ -24,14 +24,9 @@ const route = createRoute({ schema: PagePublicOutput, }, }, - description: 'Return changelog by id', - }, - 400: { - description: 'Bad Request', - }, - 500: { - description: 'Internal Server Error', + description: 'Get a page by id for public view', }, + ...openApiErrorResponses, }, }) @@ -96,6 +91,6 @@ export const registerPagePublic = (api: typeof pageApi) => { } redis.set(id, JSON.stringify(mappedResult), { EX: 60 }) - return c.json(mappedResult, 200) + return c.json(PagePublicOutput.parse(mappedResult), 200) }) } diff --git a/apps/api/src/page/update.ts b/apps/api/src/page/update.ts index 8a467b8..c240639 100644 --- a/apps/api/src/page/update.ts +++ b/apps/api/src/page/update.ts @@ -9,6 +9,7 @@ import { import { and, eq } from 'drizzle-orm' import { HTTPException } from 'hono/http-exception' import { verifyAuthentication } from '../utils/authentication' +import { openApiErrorResponses, openApiSecurity } from '../utils/openapi' import { redis } from '../utils/redis' import type { pageApi } from './index' @@ -34,13 +35,9 @@ const route = createRoute({ }, description: 'Return changelog by id', }, - 400: { - description: 'Bad Request', - }, - 500: { - description: 'Internal Server Error', - }, + ...openApiErrorResponses, }, + ...openApiSecurity, }) export const registerPageUpdate = (api: typeof pageApi) => { @@ -60,6 +57,10 @@ export const registerPageUpdate = (api: typeof pageApi) => { .where(and(eq(page.userId, userId), eq(page.id, id))) .returning() + if (!result) { + throw new HTTPException(404, { message: 'Not Found' }) + } + // TODO: implement transaction if (changelogIds) { if (changelogIds.length === 0) { @@ -80,12 +81,8 @@ export const registerPageUpdate = (api: typeof pageApi) => { } } - if (!result) { - throw new HTTPException(404, { message: 'Not Found' }) - } - redis.del(id) - return c.json(result, 200) + return c.json(PageOutput.parse(result), 200) }) } diff --git a/apps/api/src/statistic/get.ts b/apps/api/src/statistic/get.ts index de5f110..9d7530b 100644 --- a/apps/api/src/statistic/get.ts +++ b/apps/api/src/statistic/get.ts @@ -2,8 +2,10 @@ import { changelog, db, page } from '@boring.tools/database' import { StatisticOutput } from '@boring.tools/schema' import { createRoute } from '@hono/zod-openapi' import { eq } from 'drizzle-orm' + import type { statisticApi } from '.' import { verifyAuthentication } from '../utils/authentication' +import { openApiErrorResponses, openApiSecurity } from '../utils/openapi' const route = createRoute({ method: 'get', @@ -16,13 +18,9 @@ const route = createRoute({ }, description: 'Return user', }, - 400: { - description: 'Bad Request', - }, - 500: { - description: 'Internal Server Error', - }, + ...openApiErrorResponses, }, + ...openApiSecurity, }) export const registerStatisticGet = (api: typeof statisticApi) => { @@ -108,6 +106,6 @@ export const registerStatisticGet = (api: typeof statisticApi) => { }, } - return c.json(mappedData, 200) + return c.json(StatisticOutput.parse(mappedData), 200) }) } diff --git a/apps/api/src/user/get.ts b/apps/api/src/user/get.ts index 52cc980..bee1e71 100644 --- a/apps/api/src/user/get.ts +++ b/apps/api/src/user/get.ts @@ -2,7 +2,9 @@ import { db, user as userDb } from '@boring.tools/database' import { UserOutput } from '@boring.tools/schema' import { createRoute } from '@hono/zod-openapi' import { eq } from 'drizzle-orm' + import type { userApi } from '.' +import { openApiErrorResponses, openApiSecurity } from '../utils/openapi' const route = createRoute({ method: 'get', @@ -15,13 +17,9 @@ const route = createRoute({ }, description: 'Return user', }, - 400: { - description: 'Bad Request', - }, - 500: { - description: 'Internal Server Error', - }, + ...openApiErrorResponses, }, + ...openApiSecurity, }) export const registerUserGet = (api: typeof userApi) => { @@ -35,6 +33,6 @@ export const registerUserGet = (api: typeof userApi) => { throw new Error('User not found') } - return c.json(result, 200) + return c.json(UserOutput.parse(result), 200) }) } diff --git a/apps/api/src/user/webhook.ts b/apps/api/src/user/webhook.ts index dd63990..56c9cbe 100644 --- a/apps/api/src/user/webhook.ts +++ b/apps/api/src/user/webhook.ts @@ -4,7 +4,9 @@ import { UserOutput, UserWebhookInput } from '@boring.tools/schema' import { createRoute, type z } from '@hono/zod-openapi' import { HTTPException } from 'hono/http-exception' import { Webhook } from 'svix' + import type userApi from '.' +import { openApiErrorResponses, openApiSecurity } from '../utils/openapi' const route = createRoute({ method: 'post', @@ -24,13 +26,9 @@ const route = createRoute({ }, description: 'Return success', }, - 400: { - description: 'Bad Request', - }, - 500: { - description: 'Internal Server Error', - }, + ...openApiErrorResponses, }, + ...openApiSecurity, }) const userCreate = async ({ @@ -72,7 +70,7 @@ export const registerUserWebhook = (api: typeof userApi) => { case 'user.created': { const result = await userCreate({ payload: verifiedPayload }) logger.info('Clerk Webhook', result) - return c.json(result, 200) + return c.json(UserOutput.parse(result), 200) } default: throw new HTTPException(404, { message: 'Webhook type not supported' }) diff --git a/apps/api/src/utils/createRoute.ts b/apps/api/src/utils/createRoute.ts new file mode 100644 index 0000000..352aa65 --- /dev/null +++ b/apps/api/src/utils/createRoute.ts @@ -0,0 +1,26 @@ +import { type RouteConfig, createRoute as route } from '@hono/zod-openapi' +import { openApiErrorResponses } from './openapi' + +type CreateRoute = { + method: 'get' | 'post' | 'put' | 'delete' + tags: string[] + description: string + path: string + responses: RouteConfig['responses'] + request?: RouteConfig['request'] +} + +export const createRoute = (input: CreateRoute) => + route({ + ...input, + responses: { + ...input.responses, + ...openApiErrorResponses, + }, + security: [ + { + Clerk: [], + AccessToken: [], + }, + ], + }) diff --git a/apps/api/src/utils/openapi.ts b/apps/api/src/utils/openapi.ts new file mode 100644 index 0000000..26d6a5e --- /dev/null +++ b/apps/api/src/utils/openapi.ts @@ -0,0 +1,68 @@ +import { createErrorSchema } from './errors' + +export const openApiSecurity = { + security: [ + { + Clerk: [], + AccessToken: [], + }, + ], +} + +export const openApiErrorResponses = { + 400: { + description: + 'The server cannot or will not process the request due to something that is perceived to be a client error (e.g., malformed request syntax, invalid request message framing, or deceptive request routing).', + content: { + 'application/json': { + schema: createErrorSchema('BAD_REQUEST').openapi('ErrBadRequest'), + }, + }, + }, + 401: { + description: + 'The client must authenticate itself to get the requested response.', + content: { + 'application/json': { + schema: createErrorSchema('UNAUTHORIZED').openapi('ErrUnauthorized'), + }, + }, + }, + 403: { + description: + 'The client does not have the necessary permissions to access the resource.', + content: { + 'application/json': { + schema: createErrorSchema('FORBIDDEN').openapi('ErrForbidden'), + }, + }, + }, + 404: { + description: "The server can't find the requested resource.", + content: { + 'application/json': { + schema: createErrorSchema('NOT_FOUND').openapi('ErrNotFound'), + }, + }, + }, + 409: { + description: + 'The request could not be completed due to a conflict mainly due to unique constraints.', + content: { + 'application/json': { + schema: createErrorSchema('CONFLICT').openapi('ErrConflict'), + }, + }, + }, + 500: { + description: + "The server has encountered a situation it doesn't know how to handle.", + content: { + 'application/json': { + schema: createErrorSchema('INTERNAL_SERVER_ERROR').openapi( + 'ErrInternalServerError', + ), + }, + }, + }, +} diff --git a/apps/app/src/routes/page.$id.edit.lazy.tsx b/apps/app/src/routes/page.$id.edit.lazy.tsx index c101375..fca09b8 100644 --- a/apps/app/src/routes/page.$id.edit.lazy.tsx +++ b/apps/app/src/routes/page.$id.edit.lazy.tsx @@ -40,7 +40,7 @@ const Component = () => { resolver: zodResolver(PageUpdateInput), defaultValues: { ...page.data, - changelogIds: page.data?.changelogs.map((log) => log.id), + changelogIds: page.data?.changelogs?.map((log) => log.id), }, }) const onSubmit = (values: z.infer) => { @@ -138,6 +138,10 @@ const Component = () => { key={changelog.id} onSelect={() => { const getIds = () => { + if (!field.value) { + return [changelog.id] + } + if (field.value?.includes(changelog.id)) { return field.value.filter( (id) => id !== changelog.id, diff --git a/apps/app/src/routes/page.$id.index.lazy.tsx b/apps/app/src/routes/page.$id.index.lazy.tsx index 2bf2632..32d4457 100644 --- a/apps/app/src/routes/page.$id.index.lazy.tsx +++ b/apps/app/src/routes/page.$id.index.lazy.tsx @@ -31,12 +31,14 @@ const Component = () => {
- Changelogs ({data.changelogs?.length}) + + Changelogs ({data.changelogs?.length ?? 0}) +
- {data.changelogs.map((changelog) => { + {data.changelogs?.map((changelog) => { return (
{data.description}

- {data.changelogs.length >= 2 &&
+ {data.changelogs?.length >= 2 &&

Changelogs

- {data.changelogs.map((changelog) => { + {data.changelogs?.map((changelog) => { if (changelog.versions && changelog.versions?.length < 1) { return null } @@ -65,7 +65,7 @@ const data: PageById = await response.json()
})}
} - {data.changelogs.length === 1 &&
+ {data.changelogs?.length === 1 &&

Changelog

{data.changelogs.map((changelog) => { if (changelog.versions && changelog.versions?.length < 1) { diff --git a/bruno/Changelog/Version/Create Auto.bru b/bruno/Changelog/Version/Create Auto.bru index eec0da4..2e8c329 100644 --- a/bruno/Changelog/Version/Create Auto.bru +++ b/bruno/Changelog/Version/Create Auto.bru @@ -17,6 +17,6 @@ auth:bearer { body:json { { "changelogId": "d83fe688-3331-4e64-9af6-318f82e511d4", - "commitIds": ["01cc79df-6d16-4496-b9ba-b238d686efc4"] + "version": "0.1.0" } } diff --git a/bruno/Page/By Id.bru b/bruno/Page/By Id.bru new file mode 100644 index 0000000..8cfad3f --- /dev/null +++ b/bruno/Page/By Id.bru @@ -0,0 +1,15 @@ +meta { + name: By Id + type: http + seq: 1 +} + +get { + url: {{API_URL}}/page/:id/public + body: none + auth: none +} + +params:path { + id: 74ab8978-e6d4-436e-98b8-fb6d6fde35fe +} diff --git a/bt-cli b/bt-cli new file mode 100755 index 0000000..d1139b9 Binary files /dev/null and b/bt-cli differ diff --git a/bun.lockb b/bun.lockb index 3edda30..78ac010 100755 Binary files a/bun.lockb and b/bun.lockb differ diff --git a/packages/database/package.json b/packages/database/package.json index 9a2a7ac..fb6c3c8 100644 --- a/packages/database/package.json +++ b/packages/database/package.json @@ -12,7 +12,7 @@ "devDependencies": { "@types/bun": "latest", "@types/pg": "^8.11.10", - "drizzle-kit": "^0.27.1" + "drizzle-kit": "^0.28.0" }, "peerDependencies": { "typescript": "^5.0.0" diff --git a/packages/logger/package.json b/packages/logger/package.json index c64a69c..ee3e671 100644 --- a/packages/logger/package.json +++ b/packages/logger/package.json @@ -11,6 +11,7 @@ "dependencies": { "@logtail/node": "^0.5.0", "@logtail/winston": "^0.5.0", - "winston": "^3.14.2" + "winston": "^3.14.2", + "winston-loki": "^6.1.3" } } \ No newline at end of file diff --git a/packages/logger/src/index.ts b/packages/logger/src/index.ts index 6725115..15fb249 100644 --- a/packages/logger/src/index.ts +++ b/packages/logger/src/index.ts @@ -1,6 +1,7 @@ import { Logtail } from '@logtail/node' import { LogtailTransport } from '@logtail/winston' import winston from 'winston' +import LokiTransport from 'winston-loki' // Create a Winston logger - passing in the Logtail transport export const logger = winston.createLogger({ @@ -9,6 +10,14 @@ export const logger = winston.createLogger({ new winston.transports.Console({ format: winston.format.json(), }), + new LokiTransport({ + host: 'http://localhost:9100', + labels: { app: 'api' }, + json: true, + format: winston.format.json(), + replaceTimestamp: true, + onConnectionError: (err) => console.error(err), + }), ], }) diff --git a/packages/schema/src/access-token/base.ts b/packages/schema/src/access-token/base.ts index 4ae0358..88a1bee 100644 --- a/packages/schema/src/access-token/base.ts +++ b/packages/schema/src/access-token/base.ts @@ -7,6 +7,6 @@ export const AccessTokenOutput = z }), token: z.string().optional(), name: z.string(), - lastUsedOn: z.string().or(z.date()), + lastUsedOn: z.string().or(z.date()).optional().nullable(), }) .openapi('Access Token') diff --git a/packages/schema/src/changelog/base.ts b/packages/schema/src/changelog/base.ts index 3bcb430..c86cf2e 100644 --- a/packages/schema/src/changelog/base.ts +++ b/packages/schema/src/changelog/base.ts @@ -13,13 +13,15 @@ export const ChangelogOutput = z example: 'This is a changelog', }), versions: z.array(VersionOutput).optional(), - computed: z.object({ - versionCount: z.number().openapi({ - example: 5, - }), - commitCount: z.number().openapi({ - example: 10, - }), - }), + computed: z + .object({ + versionCount: z.number().openapi({ + example: 5, + }), + commitCount: z.number().openapi({ + example: 10, + }), + }) + .optional(), }) .openapi('Changelog') diff --git a/packages/schema/src/page/base.ts b/packages/schema/src/page/base.ts index 3cc7967..0ad9e3c 100644 --- a/packages/schema/src/page/base.ts +++ b/packages/schema/src/page/base.ts @@ -1,4 +1,5 @@ import { z } from '@hono/zod-openapi' +import { ChangelogOutput } from '../changelog' export const PageOutput = z .object({ @@ -8,5 +9,6 @@ export const PageOutput = z title: z.string(), description: z.string().optional(), icon: z.string(), + changelogs: z.array(ChangelogOutput).optional(), }) .openapi('Page') diff --git a/packages/schema/src/page/list.ts b/packages/schema/src/page/list.ts index 9f6b337..9a9bc77 100644 --- a/packages/schema/src/page/list.ts +++ b/packages/schema/src/page/list.ts @@ -1,9 +1,4 @@ import { z } from '@hono/zod-openapi' -import { ChangelogOutput } from '../changelog' import { PageOutput } from './base' -export const PageListOutput = z.array( - PageOutput.extend({ - changelogs: z.array(ChangelogOutput), - }), -) +export const PageListOutput = z.array(PageOutput)