EOW Dev Log #1: Circular Dependency Nightmare & Colocation Wins ๐Ÿš€

EOW Dev Log #1: Circular Dependency Nightmare & Colocation Wins ๐Ÿš€

12 days ago

3 mins read

This 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:

๐Ÿ“‚ What is Feature Colocation?

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.

๐Ÿ›‘ Preventing Circular Dependencies

To stay ahead of circular dependency issues from the start, I:

Howeverโ€ฆ neither of them warned me. Not even once.

๐Ÿšจ The Issue Appears

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). ๐Ÿ˜…

๐Ÿ” The Real Problem: Circular Dependencies

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:

๐Ÿง What Caused It?

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.

๐Ÿ› ๏ธ The Fix

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)
  1. ๐Ÿšซ No mutual imports!

  2. ๐Ÿ’ก Extract mutual imports into a dedicated module to break cycles.

  3. โžก๏ธ 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.

๐ŸŽฏ Lessons Learned

  • 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.

    • Vercel catches what local dev setups might miss โ€” deploy early to uncover hidden problems.

๐Ÿ’ก Final Thoughts

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? ๐Ÿ‘€

Share you thoughts.

Enjoyed this article?

Subscribe to my YouTube channel for coding tutorials, live project builds, and tech discussions.

Subscribe