Patolord
Patolord4mo ago

Architecture of larger convex apps.

Hey guys, I'm a solo developer working with Convex and some of my apps are getting bigger with complex business logic. I wanted a cleaner architecture approach but don't want to over-engineer since I'm working solo. I asked Claude for suggesting and it gave me the ones in the image, but I'm not experienced enough to tell if they are good or not. Would love to hear from seasoned devs about: Have you used any of these patterns with Convex? Which approach would you recommend? Any pitfalls to watch out for?
No description
7 Replies
ballingt
ballingt4mo ago
One pattern we've seen larger codebases use is splitting up the schema into smaller pieces, and using "accessor" functions to access it These images don't mean much to me What are the pain you're you're hitting? Maybe they can be addressed directly or the right pattern will be informed by what's a pain now.
Son
Son4mo ago
Do you have examples of this “accessor” patterns?
ballingt
ballingt4mo ago
Not handy, what would help to see? What types of programming are you familiar with? Accessor means getters and setters generally, here I more mean CRUD. If you have an organizations table, keep addMemberToOrg, getAllOrgMembers, etc. in one file or directory. Don't directly insert into the organizations table from outside of that directory, so you don't have to remember to check permissions or set foreign keys or enforce constraints from other places. Mostly write helper functions. API endpoints (like mutation(() => { ... })) should be short mostly call helper functions and might not belong in the organizations folder at all. Have you used Rails or Django? Think the class methods on the model classes.
Son
Son4mo ago
Thanks for the response! i’ve only used jsx based frameworks (JavaScript) and i understand what you’re getting at. Most of my queries and mutations just call helpers as you say. Im struggling to visualise what your get here though “If you have an organizations table, keep addMemberToOrg, getAllOrgMembers, etc. in one file or directory. Don't directly insert into the organizations table from outside of that directory, so you don't have to remember to check permissions or set foreign keys or enforce constraints from other places.“
ballingt
ballingt4mo ago
The idea is to have only set of ways to modify a logical entity. If you need to split the user created in addUser() into three separate tables, you'd like to have only one file if functions to change instead of feeling the codebase for everywhere the users table is accessed.
Son
Son4mo ago
Oh this makes sense! Thanks
ampp
ampp4mo ago
The best example of splitting up a schema is convex ai-town https://github.com/a16z-infra/ai-town/blob/main/convex/schema.ts
GitHub
ai-town/convex/schema.ts at main · a16z-infra/ai-town
A MIT-licensed, deployable starter kit for building and customizing your own version of AI town - a virtual town where AI characters live, chat and socialize. - a16z-infra/ai-town

Did you find this page helpful?