Unions help us define a value as one or more types. Sometimes this makes data easier to rationalize and makes it easier to represent our reality and the domains we’re exploring. A common one I think about is how the United States uses the word state
, where in Canada they use the word province
to roughly describe the same thing. The union of the two being something like a stateOrProvinceSpecificer: "state" | "province"
, a union of two constant strings.
Sometimes our data just better fits as a structured union and even across primitive types like boolean | number
or interface types like article | photo
. Here’s an deep-dive into unions within prisma. Prisma currently doesn’t support union types in the schema, but there are some ways we can work around this.
Follow along with the repo here https://github.com/reggi/demo-prisma-union
Creating the Schema
Touching on our state
vs province
example, and borrowing from this schema here, let’s say we wanted to create a Place
table that is the union of a City
or a Country
. Perhaps we’re building a map were the specificity doesn’t need to matter, or a set of destinations. Here both are nullable which allows us to use them as unions, but also opens up the database to some other unintended circumstances, in this case both can be null, and both can be set 😬. We won’t explore options for remedying this in this article, but there are other provisions that can be explored at the postgres level, as well as the API layer to ensure this doesn’t happen.
model Place {
id Int @default(autoincrement()) @id
city City? @relation(fields: [cityId], references: [id])
cityId Int?
country Country? @relation(fields: [countryId], references: [id])
countryId Int?
}
We can expand out the City
and Country
models as well. Here I also added a bit of recursion a City
also has an associated Country
.
model City {
id Int @default(autoincrement()) @id
name String @unique
places Place[]
country Country @relation(fields: [countryId], references: [id])
countryId Int
}
model Country {
id Int @default(autoincrement()) @id
name String @unique
places Place[]
cities City[]
}
With all of this placed neatly within schema.prisma
we can run npx prisma migrate dev --name init
to migrate the schema and setup the database. I’m using a local postgres database. My .env
file has something like this:
DATABASE_URL="postgresql://thomasreggi:thomasreggi@localhost:5432/prisma-demo?schema=public"
Seeding the Database
Now that our database is ready, and the Prisma types have been generated we can seed the database with some dummy data. I’ve created a script that will create two cities, New York
and Berlin
and three countries United States
, France
and Germany
. To seed the database run npm tsx seed.ts
.
Querying the Union
Now we can get to the juicy part. We can query the Place
table and massage the Typescript types to more accurately represent the union we want to work with. We want to include
both City
and Country
but also include the recursive aspect of City.Country
, this can be down with an added nested include.
const result = await prisma.place.findFirst({
where: { id },
include: {
city: {
include: {
country: true
}
},
country: true
}
})
What’s super useful now that we’ve written this out is that prisma will automatically define the return type of the find when you hover over result
you get this:
type PlaceResult = (Prisma.Place & {
country: Prisma.Country | null;
city: (Prisma.City & {
country: Prisma.Country;
}) | null;
}) | null
Note: Here I importer prisma as
import Prisma from '@prisma/client'
that way I keep prisma types scoped neatly withinPrisma.
and can define my own versions of all the types without confusion.
Given that we want to explore a Place
being a union of both City
and Country
this result
type isn’t exactly that. It’s an object that can either have a country
or a city
, in fact it could have both or neither, which is problematic. We can fix this by wrapping the create
and find
functions and ensuring that when things are created in the database and when things are read out the structure of the data is as we’d expect.
Creating a Typesafe Union
Because the example here of City
and Country
is a union of interfaces there’s an advantage to knowing whether or not the Place is a City
or a Country
from a runtime perspective. One way to do this is to have add a type
property to the interfaces. This will allow us to conditionally do different things depending on if it’s a City
or a Country
. We can wrap the prisma types like this:
import Prisma from '@prisma/client'
type Country = Prisma.Country & { type: 'country' }
type City = Prisma.City & { type: 'city', country: Country }
type Place = City | Country
Here I’m importing prisma like this import Prisma from '@prisma/client'
and not import { Country, City, Place} from '@prisma/client'
because if we imported the types directly we’d be colliding with them in creating these new union-compatible alternatives.
Voila! Place
is now a perfect union of City
and Country
.
Now that we have all the necessary types we can reuse the type return from Prisma’s find PlaceResult
that we defined above and transform it. This converts the object with .city
or .country
to a union without added type
property which will come in handy in a bit.
function transformPlaceUnion (result: PlaceResult): Place | null {
if (result?.city) {
return {
...result.city,
type: 'city',
country: { type: 'country', ...result.city.country }
}
}
if (result?.country) return { ...result.country, type: 'country'}
return null
}
Now we can write a new find
function putting all we’ve learned together:
async function find (id: number): Promise<Place | null> {
const result = await prisma.place.findFirst({
where: { id},
include: {
city: {
include: {
country: true
}
},
country: true
}
})
return transformPlaceUnion(result)
}
Now when we call this find
we get a place
variable that has the type Place | null
and we can easily access that country only for city
.
const place = await find(1)
if (!place) {
console.log('Place not found')
return
} else {
console.log(place.name)
console.log(place.type)
}
if (place.type === 'city') {
// ✅ type safety works here
console.log(place.country.name)
}
if (place.type === 'country') {
// ❌ type safety works here and will throw error country doesn't have a country
// console.log(place.country.name)
}
Conclusion
Unions help us structure our data to be more in alignment with how we see the world, and how our domain models function. By making two columns nullable and transforming the return value from the database we can explore unions in Typescript.