feat(api): add redis cache for public page
Some checks failed
Build and Push Docker Image / build (push) Has been cancelled
Build and Push Docker Image / tests (push) Has been cancelled

This commit is contained in:
Lars Hampe 2024-11-05 15:33:27 +01:00
parent 52829bd127
commit 8657228e02
13 changed files with 109 additions and 3 deletions

View File

@ -15,6 +15,7 @@
"@scalar/hono-api-reference": "^0.5.149", "@scalar/hono-api-reference": "^0.5.149",
"convert-gitmoji": "^0.1.5", "convert-gitmoji": "^0.1.5",
"hono": "^4.6.3", "hono": "^4.6.3",
"redis": "^4.7.0",
"semver": "^7.6.3", "semver": "^7.6.3",
"svix": "^1.36.0", "svix": "^1.36.0",
"ts-pattern": "^5.5.0" "ts-pattern": "^5.5.0"

View File

@ -6,6 +6,7 @@ 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 { redis } from '../utils/redis'
export const route = createRoute({ export const route = createRoute({
method: 'put', method: 'put',
@ -55,6 +56,9 @@ export const func = async ({
throw new HTTPException(404, { message: 'Not found' }) throw new HTTPException(404, { message: 'Not found' })
} }
if (result.pageId) {
redis.del(result.pageId)
}
return result return result
} }

View File

@ -9,6 +9,7 @@ import { createRoute, type z } from '@hono/zod-openapi'
import { and, eq, inArray } from 'drizzle-orm' import { and, eq, inArray } from 'drizzle-orm'
import { HTTPException } from 'hono/http-exception' import { HTTPException } from 'hono/http-exception'
import semver from 'semver' import semver from 'semver'
import { redis } from '../../utils/redis'
export const create = createRoute({ export const create = createRoute({
method: 'post', method: 'post',
@ -90,6 +91,10 @@ export const createFunc = async ({
}) })
.returning() .returning()
if (changelogResult.pageId) {
redis.del(changelogResult.pageId)
}
await db await db
.update(changelog_commit) .update(changelog_commit)
.set({ versionId: versionCreateResult.id }) .set({ versionId: versionCreateResult.id })

View File

@ -16,6 +16,7 @@ import semver from 'semver'
import type { changelogVersionApi } from '.' import type { changelogVersionApi } from '.'
import { verifyAuthentication } from '../../utils/authentication' import { verifyAuthentication } from '../../utils/authentication'
import { commitsToMarkdown } from '../../utils/git/commitsToMarkdown' import { commitsToMarkdown } from '../../utils/git/commitsToMarkdown'
import { redis } from '../../utils/redis'
export const route = createRoute({ export const route = createRoute({
method: 'post', method: 'post',
@ -141,6 +142,10 @@ export const registerVersionCreateAuto = (api: typeof changelogVersionApi) => {
.set({ versionId: versionCreateResult.id }) .set({ versionId: versionCreateResult.id })
.where(isNull(changelog_commit.versionId)) .where(isNull(changelog_commit.versionId))
if (changelogResult.pageId) {
redis.del(changelogResult.pageId)
}
return c.json(versionCreateResult, 201) return c.json(versionCreateResult, 201)
} }
@ -164,6 +169,10 @@ export const registerVersionCreateAuto = (api: typeof changelogVersionApi) => {
.set({ versionId: versionCreateResult.id }) .set({ versionId: versionCreateResult.id })
.where(isNull(changelog_commit.versionId)) .where(isNull(changelog_commit.versionId))
if (changelogResult.pageId) {
redis.del(changelogResult.pageId)
}
return c.json(versionCreateResult, 201) return c.json(versionCreateResult, 201)
}) })
} }

View File

@ -8,6 +8,7 @@ import { GeneralOutput } from '@boring.tools/schema'
import { createRoute } from '@hono/zod-openapi' 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 { redis } from '../../utils/redis'
export const remove = createRoute({ export const remove = createRoute({
method: 'delete', method: 'delete',
@ -67,6 +68,10 @@ export const removeFunc = async ({
.set({ versionId: null }) .set({ versionId: null })
.where(eq(changelog_commit.versionId, id)) .where(eq(changelog_commit.versionId, id))
if (findChangelog.pageId) {
redis.del(findChangelog.pageId)
}
return db return db
.delete(changelog_version) .delete(changelog_version)
.where(and(eq(changelog_version.id, id))) .where(and(eq(changelog_version.id, id)))

View File

@ -3,6 +3,7 @@ import { VersionUpdateInput, VersionUpdateOutput } from '@boring.tools/schema'
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 { redis } from '../../utils/redis'
export const update = createRoute({ export const update = createRoute({
method: 'put', method: 'put',
@ -70,5 +71,9 @@ export const updateFunc = async ({
.where(and(eq(changelog_version.id, id))) .where(and(eq(changelog_version.id, id)))
.returning() .returning()
if (findChangelog.pageId) {
redis.del(findChangelog.pageId)
}
return versionUpdateResult return versionUpdateResult
} }

View File

@ -1,4 +1,5 @@
import { OpenAPIHono } from '@hono/zod-openapi' import { OpenAPIHono } from '@hono/zod-openapi'
import { timing } from 'hono/timing'
import type { Variables } from '..' import type { Variables } from '..'
import type { ContextModule } from '../utils/sentry' import type { ContextModule } from '../utils/sentry'
import { registerPageById } from './byId' import { registerPageById } from './byId'
@ -9,7 +10,7 @@ import { registerPagePublic } from './public'
import { registerPageUpdate } from './update' import { registerPageUpdate } from './update'
export const pageApi = new OpenAPIHono<{ Variables: Variables }>() export const pageApi = new OpenAPIHono<{ Variables: Variables }>()
pageApi.use('*', timing())
const module: ContextModule = { const module: ContextModule = {
name: 'page', name: 'page',
} }

View File

@ -1,9 +1,12 @@
import { changelog_version, db, page } from '@boring.tools/database' import { changelog_version, db, page } from '@boring.tools/database'
import { createRoute } from '@hono/zod-openapi' import { createRoute } from '@hono/zod-openapi'
import { eq } from 'drizzle-orm' import { eq } from 'drizzle-orm'
import { endTime, setMetric, startTime } from 'hono/timing'
import { PagePublicOutput, PagePublicParams } from '@boring.tools/schema' import { PagePublicOutput, PagePublicParams } from '@boring.tools/schema'
import { HTTPException } from 'hono/http-exception' import { HTTPException } from 'hono/http-exception'
import { redis } from '../utils/redis'
import type { pageApi } from './index' import type { pageApi } from './index'
const route = createRoute({ const route = createRoute({
@ -35,6 +38,15 @@ const route = createRoute({
export const registerPagePublic = (api: typeof pageApi) => { export const registerPagePublic = (api: typeof pageApi) => {
return api.openapi(route, async (c) => { return api.openapi(route, async (c) => {
const { id } = c.req.valid('param') const { id } = c.req.valid('param')
const cache = await redis.get(id)
if (cache) {
c.header('Cache-Control', 'public, max-age=86400')
c.header('X-Cache', 'HIT')
return c.json(JSON.parse(cache), 200)
}
startTime(c, 'database')
const result = await db.query.page.findFirst({ const result = await db.query.page.findFirst({
where: eq(page.id, id), where: eq(page.id, id),
@ -70,6 +82,8 @@ export const registerPagePublic = (api: typeof pageApi) => {
}, },
}) })
endTime(c, 'database')
if (!result) { if (!result) {
throw new HTTPException(404, { message: 'Not Found' }) throw new HTTPException(404, { message: 'Not Found' })
} }
@ -81,6 +95,7 @@ export const registerPagePublic = (api: typeof pageApi) => {
changelogs: changelogs.map((log) => log.changelog), changelogs: changelogs.map((log) => log.changelog),
} }
redis.set(id, JSON.stringify(mappedResult), { EX: 60 })
return c.json(mappedResult, 200) return c.json(mappedResult, 200)
}) })
} }

View File

@ -9,6 +9,7 @@ import {
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 { verifyAuthentication } from '../utils/authentication' import { verifyAuthentication } from '../utils/authentication'
import { redis } from '../utils/redis'
import type { pageApi } from './index' import type { pageApi } from './index'
const route = createRoute({ const route = createRoute({
@ -83,6 +84,8 @@ export const registerPageUpdate = (api: typeof pageApi) => {
throw new HTTPException(404, { message: 'Not Found' }) throw new HTTPException(404, { message: 'Not Found' })
} }
redis.del(id)
return c.json(result, 200) return c.json(result, 200)
}) })
} }

View File

@ -0,0 +1,9 @@
import { createClient } from 'redis'
export const redis = createClient({
password: import.meta.env.REDIS_PASSWORD,
url: import.meta.env.REDIS_URL,
})
redis.on('error', (err) => console.log('Redis Client Error', err))
await redis.connect()

View File

@ -5,6 +5,8 @@ import { logger } from '@boring.tools/logger'
declare module 'bun' { declare module 'bun' {
interface Env { interface Env {
POSTGRES_URL: string POSTGRES_URL: string
REDIS_PASSWORD: string
REDIS_URL: string
CLERK_WEBHOOK_SECRET: string CLERK_WEBHOOK_SECRET: string
CLERK_SECRET_KEY: string CLERK_SECRET_KEY: string
CLERK_PUBLISHABLE_KEY: string CLERK_PUBLISHABLE_KEY: string
@ -12,7 +14,7 @@ declare module 'bun' {
} }
} }
const TEST_VARIABLES = ['POSTGRES_URL'] const TEST_VARIABLES = ['POSTGRES_URL', 'REDIS_URL', 'REDIS_PASSWORD']
const DEVELOPMENT_VARIABLES = [ const DEVELOPMENT_VARIABLES = [
...TEST_VARIABLES, ...TEST_VARIABLES,

BIN
bun.lockb

Binary file not shown.

View File

@ -8,4 +8,51 @@ services:
- POSTGRES_PASSWORD=postgres - POSTGRES_PASSWORD=postgres
- POSTGRES_USER=postgres - POSTGRES_USER=postgres
- POSTGRES_DB=postgres - POSTGRES_DB=postgres
cache:
container_name: boring_redis
image: redis:7.4.1-alpine
restart: always
ports:
- '6379:6379'
command: redis-server --save 20 1 --loglevel warning --requirepass development
loki:
image: grafana/loki:latest
ports:
- "9100:3100"
command: -config.file=/etc/loki/local-config.yaml
promtail:
image: grafana/promtail:latest
volumes:
- /var/log:/var/log
command: -config.file=/etc/promtail/config.yml
grafana:
environment:
- GF_PATHS_PROVISIONING=/etc/grafana/provisioning
- GF_AUTH_ANONYMOUS_ENABLED=true
- GF_AUTH_ANONYMOUS_ORG_ROLE=Admin
entrypoint:
- sh
- -euc
- |
mkdir -p /etc/grafana/provisioning/datasources
cat <<EOF > /etc/grafana/provisioning/datasources/ds.yaml
apiVersion: 1
datasources:
- name: Loki
type: loki
access: proxy
orgId: 1
url: http://loki:3100
basicAuth: false
isDefault: true
version: 1
editable: false
EOF
/run.sh
image: grafana/grafana:latest
ports:
- "9000:3000"