Skip to content

Instantly share code, notes, and snippets.

@Cauen
Last active November 21, 2024 13:46
Show Gist options
  • Save Cauen/98a4aa9956419ee370a50f1e8b9e5bbe to your computer and use it in GitHub Desktop.
Save Cauen/98a4aa9956419ee370a50f1e8b9e5bbe to your computer and use it in GitHub Desktop.
Pothos Realtime Subscription with Prisma query optimization and context cache
// Idea from https://github.com/hayes/pothos/issues/1349
import { builder } from '@/schema/builder'
import { pubsub } from '@/pubsub'
import { withFilter } from 'graphql-subscriptions'
import { queryFromInfo } from '@pothos/plugin-prisma'
import crypto from 'crypto'
import { Context } from '@/context'
import { prisma } from '@orm/db'
function hasher(value: any) {
const hash = crypto
.createHash('sha256')
.update(JSON.stringify(value))
.digest('hex')
return hash
}
const trigger = 'general-smart-subscriptions-refetch'
function publishTrigger() {
console.log('PUBLISHING trigger', trigger)
pubsub.publish(trigger, {})
}
setInterval(() => {
publishTrigger()
}, 1000)
type Payload = Record<'', ''>
type Variables = Record<'', ''>
builder.subscriptionFields((t) => {
return {
codeDevRealtimeApiKeys: t.prismaField({
type: ['ApiKey'],
subscribe: (...subProps) => {
return withFilter(
() => {
// AUTO RESPOND AFTER CONNECTION https://ralexanderson.com/blog/subscription-only-graphql-data
process.nextTick(() => {
publishTrigger()
})
return pubsub.asyncIterator(trigger)
},
async (event: Payload, variables: Variables, ctx: Context, info) => {
const query = queryFromInfo({
info,
context: ctx,
typeName: 'ApiKey',
})
const res = await prisma.apiKey.findMany({
// complex args
where: {
description: {
contains: 'Teste -',
},
},
...query,
})
const hashedRes = hasher(res)
const hashedReq = hasher({
_operatioName: 'codeDevRealtimeApiKeys',
...variables,
})
const lastRes = ctx.subscriptionResponses?.[hashedReq]
if (lastRes?.hashedRes === hashedRes) {
console.log('SAME', {
lastRes,
responses: ctx.subscriptionResponses,
user: ctx.user,
ctxKeys: Object.keys(ctx),
})
return false
}
ctx.subscriptionResponses = { [hashedReq]: { hashedRes, res } }
console.log({ subRes: ctx.subscriptionResponses })
return true
},
)(...subProps) as any as AsyncIterable<Payload>
},
// args: ... complex args
async resolve(include, parent, args, context, info) {
const hashedReq = hasher({
_operatioName: 'codeDevRealtimeApiKeys',
...args,
})
const resCopy = structuredClone(
context.subscriptionResponses[hashedReq].res,
)
delete context.subscriptionResponses[hashedReq].res
return resCopy
},
}),
}
})
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment