Interface-Oriented Design: The Foundation of Clean Architecture

Introduction

Building software that survives change is one of the biggest challenges in engineering.
Clean Architecture is a popular solution, offering a way to organize code so the “core” business logic is independent of frameworks, databases, or user interfaces. But what truly enables this independence is a design mindset called Interface-Oriented Design (IOD).


What is Interface-Oriented Design?

Interface-Oriented Design means:

  • Defining behaviors as interfaces (or abstract classes), not as concrete implementations.
  • Every dependency your business logic needs is represented as an interface—a contract that describes what should be done, not how.
  • Concrete implementations (e.g., a MySQL database, an email sender, a payment service) “plug in” by implementing these interfaces.

What is Clean Architecture?

Clean Architecture (by Robert C. Martin, “Uncle Bob”) organizes your code in layers:

  • Core Layer:
    Pure business rules and logic—no knowledge of database, web frameworks, or external systems.
  • Interface/Boundary Layer:
    Interfaces (contracts) that define what outside services must provide (e.g., repositories, gateways).
  • Infrastructure Layer:
    Actual implementations (database adapters, API clients, frameworks) that satisfy the interfaces.

The most important rule:

Dependencies always point inward—only the outer layers depend on the inner layers, never the other way around.


Why is IOD the DNA of Clean Architecture?

Clean Architecture enforces that core logic must not depend on outside details.
How does it enforce this?
By requiring all communication between layers to happen via interfaces.

If you don’t use interfaces, the core will inevitably become coupled to frameworks and infrastructure—breaking the Clean Architecture principle.


Mermaid.js Diagram: Clean Architecture & Interface-Oriented Design

flowchart TD
    subgraph OuterLayer [Framework/Infrastructure Layer]
        A["Web Controller"]
        B["Database Adapter"]
        C["Payment Adapter"]
    end
    subgraph Interfaces
        D["OrderRepository (interface)"]
        E["PaymentGateway (interface)"]
    end
    subgraph Core [Business Logic / Use Case]
        F["OrderService"]
        G["PaymentService"]
    end

    A --> D
    B --> D
    C --> E
    F --> D
    G --> E

Explanation of Diagram:

  • OuterLayer:
    • Components like web controllers, database adapters, and payment adapters.
    • These are the “details” of frameworks and technology.
  • Interfaces Layer:
    • OrderRepository, PaymentGateway, and similar abstractions.
    • These act as contracts between the core and the outside world.
  • Core Layer:
    • Pure business logic: order processing, payment rules, calculations.
    • Depends only on interfaces, never on actual web, database, or payment code.

Practical Example

Suppose your core logic needs to save orders and process payments.
In IOD/CA, you do this:

# In core/business logic
class OrderRepository(ABC):
    @abstractmethod
    def save(self, order): ...

class PaymentGateway(ABC):
    @abstractmethod
    def pay(self, amount): ...

class OrderService:
    def __init__(self, order_repo: OrderRepository, payment_gateway: PaymentGateway):
        self.order_repo = order_repo
        self.payment_gateway = payment_gateway

    def place_order(self, order, payment_amount):
        self.order_repo.save(order)
        self.payment_gateway.pay(payment_amount)

Elsewhere, in the outer layer:

class MySQLOrderRepository(OrderRepository):
    def save(self, order): 
        # implementation details

class StripeGateway(PaymentGateway):
    def pay(self, amount): 
        # implementation details
  • You can easily replace MySQLOrderRepository with MongoOrderRepository without changing your business logic.
  • You can unit-test OrderService by passing in a fake or mock implementation.

Key Benefits

  • Flexibility:
    Swap databases, APIs, or frameworks with minimal changes.
  • Testability:
    Core logic can be tested in isolation using mock interfaces.
  • Maintainability:
    The system is resilient to tech changes and easy to extend.

Conclusion

Interface-Oriented Design is the essential technique that enables Clean Architecture’s structure and power.
By programming to interfaces and enforcing clear, abstract boundaries, you ensure your system’s core remains pure, robust, and ready for anything the future brings.


Get in Touch with us

Chat with Us on LINE

iiitum1984

Speak to Us or Whatsapp

(+66) 83001 0222

Related Posts

Our Products