DEV Community

Cover image for Stop Writing Endpoints. Start Defining Systems.
Drew Marshall
Drew Marshall

Posted on

Stop Writing Endpoints. Start Defining Systems.

For a long time, I thought building APIs meant writing endpoints.

You know the pattern:

  • Define a route
  • Validate input
  • Query the database
  • Transform the result
  • Send a response

Do that over and over again.

Different routes. Same structure.


The Illusion of Control

Writing endpoints feels productive.

You’re in control of everything:

  • The logic
  • The validation
  • The data flow

But after a while, something becomes obvious:

You’re not building systems.

You’re repeating patterns.


The Real Problem

Most APIs look like this:

app.get('/users/:id', async (req, res) => {
  const id = req.params.id;

  if (!id) {
    return res.status(400).json({ error: 'Missing id' });
  }

  const user = await db.users.findById(id);

  if (!user) {
    return res.status(404).json({ error: 'Not found' });
  }

  return res.json(user);
});
Enter fullscreen mode Exit fullscreen mode

Now multiply that by:

  • Dozens of endpoints
  • Multiple resources
  • Different validation rules
  • Slight variations in logic

You end up with:

  • Repeated code
  • Inconsistent patterns
  • Hard-to-maintain systems

You’re Not Writing Logic. You’re Rewriting Structure.

Look closer at most endpoints.

They follow the same shape:

  1. Extract input
  2. Validate input
  3. Execute query
  4. Handle errors
  5. Return response

The structure doesn’t change.

Only the details do.

So why are we rewriting the structure every time?


The Shift: Define, Don’t Rewrite

Instead of writing endpoints…

Define them.

What if your API looked like this instead?

get:
  user:
    GetUserById:
      input:
        id: number
      where:
        id: $param.id
      response:
        id: number
        name: string
        email: string
Enter fullscreen mode Exit fullscreen mode

No route handler.

No repeated boilerplate.

Just a definition.


What This Changes

When you define systems instead of writing endpoints:

  • Structure becomes consistent
  • Validation becomes automatic
  • Queries become predictable
  • Behavior becomes visible

You’re no longer guessing how something works.

You can read it directly.


From Endpoints to Systems

Traditional approach:

  • Every endpoint is custom
  • Logic is scattered
  • Behavior is implicit

System-driven approach:

  • Endpoints follow a pattern
  • Logic is structured
  • Behavior is explicit

You move from “code-first” to “contract-first.”


Where the Code Goes

This doesn’t eliminate code.

It moves it.

Instead of writing endpoint logic repeatedly…

You write:

  • A compiler that reads definitions
  • A pipeline that executes them
  • A system that enforces rules

Code becomes the engine.

Not the repetition.


Example Flow

With a system-driven approach, a request might flow like this:

Request → Parse Definition → Validate → Build Query → Execute → Format Response
Enter fullscreen mode Exit fullscreen mode

The difference is:

  • The flow is constant
  • The behavior is defined in configuration

Why This Matters

Without this approach:

  • Every developer writes endpoints differently
  • Bugs are repeated across routes
  • Refactoring becomes painful

With this approach:

  • Patterns are enforced
  • Behavior is predictable
  • Systems scale cleanly

“Isn’t This Less Flexible?”

Yes.

And that’s the point.

Unlimited flexibility leads to:

  • Inconsistency
  • Complexity
  • Fragile systems

Constraints lead to:

  • Clarity
  • Stability
  • Speed

Where This Fits

This kind of system works best when:

  • You have repeated CRUD patterns
  • You want consistent APIs
  • You care about long-term maintainability

It doesn’t replace every use case.

But it replaces most of the boring, repetitive ones.


The Bigger Idea

This isn’t just about APIs.

It’s about how we build software.

Instead of:

  • Writing everything manually
  • Repeating patterns
  • Hoping for consistency

We can:

  • Define systems
  • Enforce structure
  • Let the engine handle execution

Final Thought

Writing endpoints feels like control.

But it’s often just repetition.

Defining systems feels restrictive at first.

But it leads to something better:

Clarity.

Consistency.

Scalability.

That’s why I stopped writing endpoints…

…and started defining systems.

Top comments (0)