Convex CommunityCCConvex Community
Powered by
RJR
Convex Community•3y ago•
19 replies
RJ

Declarative schema validations and branding

I'm sure this is a big topic that you all have thought a lot about, but I've found myself more and more wishing that I could declare constraints for specific fields in my schema. It would be really neat (I think) if Convex supported something like this:

items: defineTable({
  quantity: s.number().validate((n) => n >= 0)
})
items: defineTable({
  quantity: s.number().validate((n) => n >= 0)
})


Such that whenever using
db.insert
db.insert
,
db.patch
db.patch
or
db.replace
db.replace
, any validations would be run on new or modified fields, and an error would be raised if any fail (or, better yet, the validation failure is [even optionally] reflected in the type signature).

I imagine there are much fancier alternatives to this, but this seems to have some nice advantages nonetheless:

1. It's effectively the same as what a user would have to do within each of their individual mutation functions if they're trying to enforce this invariant in a more ad-hoc way (the only option at the moment). By "effectively the same" I mean it is computationally identical (JS/TS code being executed at runtime), but a much nicer API.
2. It's what many ORMs do and have done for a long time (e.g. Ruby's Rails and Elixir's Phoenix, at least)—which implies a) that it's generally a useful pattern and also b) that it's familiar to people.
3. It doesn't require changing the database layer to implement (obviously database-layer guarantees are much nicer, but are, I imagine, also much harder—or at least more work).

In a similar vein, I've often found myself wanting to be able to more easily declare that e.g. a
s.string()
s.string()
is an ISO 8601 date. It would be cool to extend the above with type branding, such that you could have something like:

export default defineSchema({
  items: defineTable({
    quantity: positiveInteger
  })
});

const positiveInteger = s.number()
  .validate((n) => n >= 0)
  .brand("PositiveInteger");

export type PositiveInteger = Infer<typeof positiveInteger> // number & Brand<"PositiveInteger">
export default defineSchema({
  items: defineTable({
    quantity: positiveInteger
  })
});

const positiveInteger = s.number()
  .validate((n) => n >= 0)
  .brand("PositiveInteger");

export type PositiveInteger = Infer<typeof positiveInteger> // number & Brand<"PositiveInteger">


These would both be huge DX upgrades for me!
Convex Community banner
Convex CommunityJoin
Join the Convex Discord! Explore Convex, the backend platform that keeps your app in sync.
15,550Members
Resources
Recent Announcements

Similar Threads

Was this page helpful?
Recent Announcements
james

Hi @everyone with everything going on today I forgot to post on discord about our EU launch! We have a region in Dublin now! You can start using it immediately, even for folks on the free plan. https://news.convex.dev/we-finally-got-our-eu-visa/ Stay tuned for more regions but also stay tuned for more infra improvements where we improve latency for teams no matter where their servers are located. Just at the tip of the iceberg re all the optimizations we have lined up.

james · 2mo ago

Wayne

Hi @everyone happy Monday. Components Authoring [Challenge](https://www.convex.dev/components/challenge) updates! Meet the second batch of community-approved components **Firecrawl Scrape **- Scrape any URL and get clean markdown, HTML, screenshots, or structured JSON - with durable caching and reactive queries. https://www.convex.dev/components/firecrawl-scrape Built by: Gitmaxd **Durable Agents **- A Convex component for building durable AI agents with an async tool loop. https://www.convex.dev/components/durable-agents Built by: Siegfried **Convex Debouncer** - A server-side debouncing component for debouncing expensive operations like LLM calls, metrics computation, or any heavy processing that should only run after a period of inactivity. https://www.convex.dev/components/debouncer Built by: Ilya **DatabaseChat **- A Convex component for adding natural language database queries to your app. https://www.convex.dev/components/database-chat Built by: Nick **Transloadit** - A Convex component for creating Transloadit Assemblies, handling resumable uploads with status, and persisting status/results in Convex. https://www.convex.dev/components/transloadit Built by: Kvz **Loops** - A Convex component for integrating with Loops.so email marketing platform. https://www.convex.dev/components/loops Built by: Bobby The [challenge](https://www.convex.dev/components/challenge) is now ongoing, so keep building, and we'll keep rewarding. Thanks, everyone!

Wayne · 2mo ago

Liz C

Hi everyone! Have you ever wanted to get your hands on some convex swag? Well you're in luck! We just launched our Convex swag store. Check it out here ---> https://store.convex.dev/

Liz C · 2mo ago

Similar Threads

Schema and functions advice
✪ Flamakæ✪✪ Flamakæ / support-community
13mo ago
Relation between App Schema and Component Schema (betterAuth)
ViperVViper / support-community
5mo ago
Convex api not working and causing problem in user validations from the convex tables
RoninRRonin / support-community
2y ago
updating Schema: Schema validation failed
ShinobiSShinobi / support-community
3y ago