12 days ago
3 mins readThis week, I faced circular dependency issues head-on while working on my bootstrapped real estate startup project.
A deep dive into feature colocation architecture revealed some unexpected blind spots.
Hereโs how it went:
Feature colocation means grouping related files inside feature-specific folders rather than scattering them across different directories.
This keeps:
Business logic close to UI & API calls
Code more maintainable & scalable
After reading TkDodoโs article on Colocation (where he also referenced Kent C. Doddsโ take on colocation), I structured my project like this:
/features/property
โโโ server.ts
โโโ client.ts
โโโ consts.ts
โโโ model.ts
โโโ types.ts
โโโ components/
โโโ hooks/
All logic flows through these feature folders. Seemed perfect โ until it wasnโt.
To stay ahead of circular dependency issues from the start, I:
Set up import/no-cycle ESLint rules.
Installed madge to detect dependency loops.
Howeverโฆ neither of them warned me. Not even once.
Everything seemed fine โ until I pushed to GitHub after working on multiple features. ๐
The PR triggered a Vercel deploymentโฆ
Boom ๐ฅ A build error: Invalid ESLint rule.
Turns out, my ESLint plugins config was wrong (I used an array instead of an object). ๐
After fixing ESLint, I ran pnpm build
locallyโฆ and guess what?
Over 100 files had circular dependency issues! ๐ตโ๐ซ
I knew circular deps existed in theory but never had to deal with them at this scale (100+ files).
Below is what the circular dependency error looks like after running pnpm build
:
My feature assets โ like consts.ts
, types.ts
, model.ts
โ were cross-importing each other, causing a loop.
Example:
// consts.ts
import { SomeType } from "./types";
// types.ts
import { SOME_CONST } from "./consts";
// model.ts
import { SomeType } from "./types";
import { SOME_CONST } from "./consts";
This caused a self-referencing loop, breaking the build.
I enforced a strict import hierarchy:
types.ts โก๏ธ consts.ts โก๏ธ model.ts
// โ
Define types first; they should not import from consts (types.ts)
// โ
Constants can import types but not models (consts.ts)
// โ
Models can import both types and constants (model.ts)
๐ซ No mutual imports!
๐ก Extract mutual imports into a dedicated module to break cycles.
โก๏ธ Clear, one-directional data flow.
After implementing the above fixes, the project finally builds successfully locally โ
โ And on Vercel
In total, I made changes to 147 files to resolve the circular dependency issue.
Feature colocation reduces mental load & makes refactoring easier.
Always validate ESLint rule configurations before trusting them
Prototype errors early to verify expected behavior.
Push more often โ frequent commits & early deployments help catch issues sooner.
Despite the struggles, colocation made my architecture cleaner & more scalable. Iโd still choose it again for future projects.
Have you ever dealt with circular dependencies at scale? How did you fix them? ๐
Enjoyed this article?
Subscribe to my YouTube channel for coding tutorials, live project builds, and tech discussions.
Subscribe