Type-Driven vs Object-Driven Development
Full disclosure: I am a big fan of Type-Driven development.
any
in order to dodge the hassles, while others go all in on fully typed projects that are beautifully precise but can be tough to maintain. Then there’s the middle ground: those who know enough TypeScript to avoid as any
but still write code in a practical, human-friendly way.
No matter where you fall on that spectrum, if you have even a little familiarity with TypeScript, you’ve likely formed an opinion - consciously or otherwise - on how to define entities. That’s what this article is all about. Different organizations, open-source projects, and libraries adopt various methods for defining entities. We’ll explore several approaches and, most importantly, discuss their pros and cons.
Building a Social Network
To avoid the usual car-and-plane comparisons, let’s define entities for a fictional social network called NotMySpace. In the codebase for this social network, we have these entities:organization->users
array.
Object-Driven Development
Imagine you’re defining theUser
entity. You might use a library like Zod, which follows an object-driven development approach. You specify an object at runtime (the Zod schema), and then infer the type from that object:
// Abstract code
const UserSchema = z
.object({
email: z.string().min(4),
passwordHash: z.string(),
image: z.string().url(),
});
type User = z.infer<typeof UserSchema>
Here, UserSchema
is a Zod schema instance that comes with its own methods - such as parse
or safeParse
- which you can use to validate any object against the schema. Meanwhile, User
is simply the inferred TypeScript type, which can be used for function arguments, return values, class properties, etc. This exemplifies “object-driven development” because you define the runtime object first and then derive the TypeScript type from it.
Pros of Object-Driven Development
Cons of Object-Driven Development
Type-Driven Development
On the flip side, you can define your types right away, before writing any runtime code. This strategy lets you think through your data models in a purely static way. Once you have your types, you can start building the logic around them. For instance:export interface User {
email: string
passwordHash: string
image: string
};
export interface Organization {
name: string
image: string
yearCreated: number
users: User[]
};
export interface Session {
accessToken: string
ttl: number
userId: string
};
You might then generate or write runtime validation code only when you need it - if you need it at all.
Pros of Type-Driven Development
Cons of Type-Driven Development
Choosing the Right Approach (or Blending Them)
Which approach is better? As I mentioned earlier, I'm a strong believer in Type-Driven development. In my view, types are the most effective way to share architectural ideas and pinpoint boundaries in a project. From my experience, teams that frequently run into type and compiler issues often are in projects where no one invested time in defining clear, descriptive types. Defining types in TypeScript goes far beyond writingname: string
. It can involve everything from string unions to advanced type transformations. Developers must be careful in this process, dedicating time up front to outline exactly how data will flow through their system. After all, the code itself should merely support what those types describe. This way, even newcomers can quickly understand the project’s core logic just by reviewing the types.
In many real-world scenarios, you might do both: define certain critical domains or public-facing entities with an object-driven approach, but keep smaller, short-lived data structures as simple TypeScript types. This hybrid approach may work well if you clearly communicate where and how each method should be used. From my perspective, this works only if you have a really small team. Generally, it's not worth it.
Conclusion
Whether you prefer object-driven development (using libraries like Zod) or type-driven development (leaning on TypeScript’s type system, leveraging type support in libraries Joi), each approach has its role. Object-driven development works in smaller semi-complex systems that are usually not maintained by bigger teams. It definitely makes you spend less time on writing new features in the system. Pre-defined types on the other hand work as single source of truth, but might feel verbose for smaller or transient objects.