jabra
jabra3w ago

Hi All,

Hi All, I would appreciate help on best practices how to do a paginate & join query or a preferred way to query efficiently, avoid unnecessary filtering and calls to loadMore on the frontend. I'm trying not to use collect() even when querying with indexes since I want to stuff to continue to scale. I'm not tied to this design but I have 3 tables where
schema is vary simple
members: (podId, userId)
pods: (superPodId, name, ....) with search index
superPods: (name, ....)
schema is vary simple
members: (podId, userId)
pods: (superPodId, name, ....) with search index
superPods: (name, ....)
// I need way effecient way to paginate the pods that the user is a member of
export const getMemberPods = query({
args: {
superPodId: v.id('superPods'),
query: v.optional(
v.object({
name: v.optional(v.string()),
categoryName: v.optional(v.string()),
})
),
paginationOpts: paginationOptsValidator,
},
handler: async (ctx, args) => {
// get userId
...
const podsQuery =
query?.name && query.name !== ''
? ctx.db
.query('pods')
.withSearchIndex('search_name', (q) => q.search('name', query.name).eq('superPodId', superPodId))
: ctx.db
.query('pods')
.withIndex('by_superPodId', (q) => q.eq('superPodId', superPodId))
.order('desc')

const paginatedPods = await podsQuery.paginate(paginationOpts);
const memberPods = (
await Promise.all(
paginatedPods.page.map(async (pod) =>
(await getMember(ctx, pod._id, userId)) ? pod : null
)
)
).filter((pod): pod is Doc<'pods'> => pod !== null);
// post processing
....

return { ...paginatedPods, page: memberPods }
}


// Helper function (member schema is userId, podId)
const getMember = async (ctx: QueryCtx, podId: Id<'pods'>, userId: Id<'users'>) => {
return ctx.db
.query('members')
.withIndex('by_pod_id_user_id', (q) => q.eq('podId', podId).eq('userId', userId))
.unique()
}
// I need way effecient way to paginate the pods that the user is a member of
export const getMemberPods = query({
args: {
superPodId: v.id('superPods'),
query: v.optional(
v.object({
name: v.optional(v.string()),
categoryName: v.optional(v.string()),
})
),
paginationOpts: paginationOptsValidator,
},
handler: async (ctx, args) => {
// get userId
...
const podsQuery =
query?.name && query.name !== ''
? ctx.db
.query('pods')
.withSearchIndex('search_name', (q) => q.search('name', query.name).eq('superPodId', superPodId))
: ctx.db
.query('pods')
.withIndex('by_superPodId', (q) => q.eq('superPodId', superPodId))
.order('desc')

const paginatedPods = await podsQuery.paginate(paginationOpts);
const memberPods = (
await Promise.all(
paginatedPods.page.map(async (pod) =>
(await getMember(ctx, pod._id, userId)) ? pod : null
)
)
).filter((pod): pod is Doc<'pods'> => pod !== null);
// post processing
....

return { ...paginatedPods, page: memberPods }
}


// Helper function (member schema is userId, podId)
const getMember = async (ctx: QueryCtx, podId: Id<'pods'>, userId: Id<'users'>) => {
return ctx.db
.query('members')
.withIndex('by_pod_id_user_id', (q) => q.eq('podId', podId).eq('userId', userId))
.unique()
}
Thanks
27 Replies
erquhart
erquhart3w ago
Realistic expectations question: about how many pods do you expect the average user to be a member of, roughly? And about how many pods would a user realistically be a part of on the high end?
jabra
jabraOP3w ago
Good point, designers will use the pods for their clients and students Designers on my platform teach 3-4 a year I am just opening the client pods and I assume they will have 30 clients a year ~30 pods a year for each designer
erquhart
erquhart3w ago
How many pods do you expect in a super pod Eg., could a super pod scale to be many hundreds of pods quickly Also a little confused on member schema having a podId field if members can have multiple pods, guessing I'm missing something there Ah a user can have multiple member records
jabra
jabraOP3w ago
The public pods can be in the many hundreds and will grow For the public pods I can easily paginate on visibility field for now But I will have the same problem if the user want filter on category if the podCategories table which will have categoryId and podId
erquhart
erquhart3w ago
If a pod can only have a single super pod id, I would consider adding super pod id to the member schema. A bit of strategic denormalization can make things much simpler in Convex. Not required at all, though. I really don't expect you to hit performance issues with what you're describing.
jabra
jabraOP3w ago
That’s easy to do
erquhart
erquhart3w ago
That would make fetching all pods a user is a member of within a given super pod work just fine as an indexed query with collect().
jabra
jabraOP3w ago
So you are saying can do the following, collect , collect and … and then paginate on the final result and I will hit no issues
erquhart
erquhart3w ago
For the member pods of a given user, within a single super pod (although I don't know what that represents, but assuming it's a meaningful delineator), I doubt you would need pagination. You would query on the members table first, not pods index that query on user id, pod id, and super pod id
jabra
jabraOP3w ago
But for the sake of the discussion to understand convex when I really scale how should I approach the join paginate Is it with aggregates, flat tables I’m also trying to understand convex Since I’m not hitting those limits now
erquhart
erquhart3w ago
It really depends on the complexity of the query, but paginating plus mapping goes quite a long way.
jabra
jabraOP3w ago
But it will affect how I think about schemas etc…. Let say you paginate and filter like I did above
erquhart
erquhart3w ago
I have pagination that maps multiple levels deep on multiple relations per record, plus the same kind of optional searching that you're doing in your example and it works really well
jabra
jabraOP3w ago
Let’s say I solved my issue and added to the members table superpodId. The filter is what throwing me off , imagine I filter on categories so when I paginate, if I have a batch 10 then the first batch can be empty and need to call loadmore on the client
erquhart
erquhart3w ago
Yeah that's true - here's a relevant comment on that: https://discord.com/channels/1019350475847499849/1332861193764999229/1333676309628325982 So there's a few ways you can go about this, but it really depends on your actual use case. I'd really recommend keeping it simple and worrying about limits when you see queries start to take time. Convex is really fast.
jabra
jabraOP3w ago
Seems the best way is a materialized view
jabra
jabraOP3w ago
No description
erquhart
erquhart3w ago
Yeah that would be worth the effort if it was necessary, I agree. If you're getting into search complexity with filters and such, I could definitely see that being a good investment out the gate just for that use case. I've been keeping my search scenarios simple because I'm lazy lol, but I know it's coming
jabra
jabraOP3w ago
Exactly, thanks for brainstorming and the link, helped a lot
ian
ian3w ago
Also a teaser: something from @lee is coming out in the next couple days, but is already on 0.1.72: https://github.com/get-convex/convex-helpers/tree/main/packages/convex-helpers#composable-querystreams
GitHub
convex-helpers/packages/convex-helpers at main · get-convex/convex-...
A collection of useful code to complement the official packages. - get-convex/convex-helpers
jabra
jabraOP2w ago
this is awesome I used it and this is exactly and I needed Thanks @lee and @ian
lee
lee2w ago
Merging Streams of Convex data
New convex-helpers are available now for fetching streams of documents, merging them together, filtering them them out, and paginating the results. Wi...
Translate SQL into Convex Queries
Here’s a cheatsheet with examples of conversions between SQL queries and Convex queries. This article is geared towards developers (and LLMs) who have...
jabra
jabraOP2w ago
@lee great articles small fix: In this query you are missing
query('messages')
query('messages')
before
.withIndex
.withIndex
jabra
jabraOP2w ago
No description
lee
lee2w ago
ty! @ian pls fix
jabra
jabraOP2w ago
@ian @lee any idea when stream will support withSearchIndex ? https://github.com/get-convex/convex-helpers/blob/main/packages/convex-helpers/server/stream.ts#L602
withSearchIndex(_indexName: any, _searchFilter: any): any {
throw new Error("Cannot paginate withSearchIndex");
}
withSearchIndex(_indexName: any, _searchFilter: any): any {
throw new Error("Cannot paginate withSearchIndex");
}
GitHub
convex-helpers/packages/convex-helpers/server/stream.ts at main · ...
A collection of useful code to complement the official packages. - get-convex/convex-helpers
ian
ian2w ago
Let's put things like this in the community support channel so they're easier for others to find, and/or feature requests on that repo. Thanks!

Did you find this page helpful?