The common architectural patterns are all pretty much the same. Layered architecture, MVC, Hexagonal architecture, Onion Architecture, Clean architecture… they all do two things:
- Define domains of responsibility
- Put functional dependencies in order
Look at these depictions of some of the patterns:
What you find are shapes (rectangles, circles) with the responsibility to produce part of the required behavior of a software. All models forsee certain responsibilities like interaction with the user, persisting data, representing and working with domain data etc. Some patterns are more detailed, some are less. That way they are more or less applicable to concrete requirements.
And those shapes are explicitly or implicitly connected in service relationships. One shape uses another; one shape depends on another. Sometimes arrows denote these relationships, sometimes mere location (e.g. a shape on top of another means “top uses bottom”).
These uses-relationships are cleanly directed and there are no cycles.
This is all nice and well. It helped us along quite a while.
But again and again I’ve seen developers scratching their heads. What about stored procedures in a database? They implement business logic, but don’t reside in the Business Layer. What about JavaScript validation logic? It’s part of the Presentation Layer, but implements business logic. Or what if I don’t want to go down the DDD path with Domain Entities and Services? Maybe the application is too small for this.
Most of all, though, what boggles the mind of developers, as far as I can see, are the functional dependencies between all those responsibilities. It’s deep hierarchies which won’t go away by switching to IoC and DI.
Looking at a function on any level of this hierarchy is an exercise in archaeology. Put on your head lamp, get out the brush, be careful while you uncover mysterious code hieroglyphs of the past. There is so much to learn from those 5,000 lines of code… And there, look, another tunnel leading you to yet another level of code.
As useful the architectural patterns are, they are only coarse grained and they leave us with a very crude idea of the fundamental structure of software. According to them, software is just a deep hierarchy of services with particular responsibilities calling other services calling other services etc. for help.
Thus the DI container has become the developers best friend. And his second best friend has become the mock framework.
This is bothering me.
Certainly every software system is made up of aspects which should be encapsulated in modules. There might be a UI aspect or a persistence aspect or a use case aspect or an encryption aspect or a reporting aspect etc. etc. They are all a matter of the Single Responsibility Principle, which I’ve already put under the microscope.
At this point I don’t want to argue with the established architectural patterns about the responsibilities they perceive as inevitable, be that Business Layer or Repository or Controller etc.
Let’s call them behavioral responsibilities. Because each responsibility is about a functional or non-functional aspect of the behavior the customer wants a software to show.
But what I question is, whether the fundamental notion of software being just a uniform hierarchy of services is useful. And service being some kind of behavioral responsibility to be used by another service.
Essentially there is no distinction. Look at the code in a function in a Presentation Layer or a Use Case or a Controller or a Respository. You won’t see a difference – except in behavioral purpose. All functions on all levels of the dependency tree contain logic and calls to other functions.
“That’s how it is. That’s the natur of software, isn’t it?”, you might say. But I beg to differ. This is just how we structure software today. It’s a mush. And that’s why there are functions with 5,000 or even 10,000 lines of code. There is no technical limit to how many LOC a function can have. And there is no artificial rule about how many LOC it should have (except for some recommendations). It’s all up to us developers. So if push comes to shove a method just grows – one line at a time. Because it can.
This is what has been happening for the past 50 years. And it has been happening despite all well meant design patterns or architecture patterns.
Why’s that? My guess, it’s because all those patterns do not question the basic assumption of all levels of dependency being created equal.
Patterns so far have been concerned with defining certain behavioral responsibilities and nicely ordered dependencies between them. That’s all.
So I suggest, if we don’t like the current state of codebase affairs, then we should try something new. Here’s what:
Formal responsibilities
I’d like to propose to search for the fundamental architecture of software in another dimension. The current architectural patterns are just evolutionary steps in the same dimension. So let’s step out of it. Let’s turn 90° and start moving down another path.
It’s one thing to search for responsibilities in the behavioral domain. That’s necessary, but also it is limited. Like the Flatlanders baffled by the effects of a 3D object passing through their 2D space, we’re again and agin baffled by the temptation to name some class “manager” or “coordinator”. It doesn’t sound right.
Locked in our current view of software design the need for managercoordinatorcontroller classes seems like a symptom of bad object-orientation.
But what if we broaden our view, step out of our dimension or behavioral responsibilities?
Let me introduce three formal responsibilities. I call them formal, because they are not concerned with creating a certain behavior. These formal responsibilities are orthogonal to all behavioral responsibilities.
Operation
Operation I call the responsibility of any logic. (With logic being transformations/expressions, control statements, hardware access/API calls.)
Any module containing just logic, is an operation. What an operation does, whether it stores data in a file, calculates a price, parses a string, stuffs data into a view is of no concern.
It’s implicit in that operations only contain data. However, since this is so important, let me state it also explicitly: Operations may not call any other operation. Operations don’t know of each other. There is no functional dependency between operations. Calling a function is not part of the definition of logic (see above).
Operations just execute their own logic which works on data. They are IO-processors: given some input they produce output (and possibly side effects).
Data
Giving structure to data is a responsibility separate from operating on data. Operations of course work on data – but they are not data.
This might sound opposed to object-orientation, but it is not. Data structures may provide services to work with them. Those services just should be limited to maintaining structure and consistency of the data.
If something is data, then that’s what its functionality should be limited to. If, on the other hand, something has data, it’s free to operate on it in any way.
Integration
Since operations don’t know each other, there needs to be some other party to form a visible behavior from all those small operational behaviors. That’s what I call integration.
Integration is the responsibility to put pieces together. Nothing more, nothing less. That means, integration does not perform any logic. From this also follows, integration is not functionally dependent on operations or other integration.
Formal dependencies
The formal responsibilities (or aspects) are independent of any domain. They neither suggest nor deny there should exist adapters or controllers or use cases or business logic modules in a software. They are also independent of any requirements scenario. Software can (and should) be structured according to the formal aspects if it’s a game, a web application, a batch processor, a desktop application or a mobile application or some service running on a device.
The formal aspects are truly universal. They define fundamental formal responsibilities to be separated into modules of their own. Put operations into other modules than integration or data.
And interestingly what goes away if you do this are functional dependencies between the modules of an application.
If you decide to have some operation module for displaying data, and some for doing business calculations, and some for persisting data… then those modules won’t be calling each other.
They just float around independently. Only maybe sharing some data.
As you can see, operations depend on data. Of course that’s necessary. Otherwise there would be no purpose for software. Logic needs to have raw material to process.
But data is not functionality. It’s, well, data. It’s structure.
Finally those operations floating around need to be wired-up to work in cooperation towards a common goal: some larger behavior of which each is just a part.
This is done by integration on possibly many levels.
Integration depends on operations and other integration. But also this dependency is not “bad”, because it’s no functional dependency.
Integration depends on operations and other integration. But also this dependency is not “bad”, because it’s no functional dependency.
The dependencies here are just formal. Or empty, if you will. They don’t have to do with any functional or non-functional efficiency requirements.
Separating What from How
I call this universal separation of formal responsibilities IODA Architecture. There are several strata of Integration sitting on top of one stratum of Operations using Data – and APIs from outside the scope of the requirements to at least cause any tangible effects on hardware.
Or if your more inclined towards “onions” here’s another one 😉
Or this one, depending on how you want to look at hit. You could say, the environment interacts with some integration, so they need to be on the surface. Or you could say, the environment is only accessible through APIs, so they need to be on the surface.
In any case, operations are about how behavior is created. Logic is imperative, it’s nitty gritty details.
Integration on the other hand defines what is supposed to happen. It does not contain logic, it thus is not imperative, but declarative. Like SQL is declarative. Integration assumes the stuff it integrates just to work. If that assumption is valid, then it promises to wire it up into something larger and also correctly functioning.
Also data is about the what. No logic in data except to enforce a certain structure.
Self-similarity
I’ve long sought an image to symbolize the IODA architecture. And now I’ve found one which I like quite a bit: a Sierpinski triangle.
I think it fits because it’s a self-similar figure. It’s a triangle made up of triangles made up of triangles… It’s triangles all the way down 🙂 A fractal.
IODA is also self-similar: An operation on one level of abstraction can in fact be an IODA structure – which you only see when you zoom in.
Operations on a high level of abstraction are black boxes. They are leaves of the the behavioral tree – which works on data and uses APIs.
But if you think, an operation is too coarse grained, you may open it up at any time. Refine it. Zoom in. Then it’s decomposed into another “miniature” IODA hierarchy. The integration formerly integrating the operation with others then integrates the root integration of the operation.[1]
Summary
To escape from “dependency hell” it’s not enough to wave the IoC/DI wand. The problem is more fundamental. It’s the very assumption software should be built at deep hierarchies of functionally dependent modules that’s problematic.
Functional dependencies lead to ever growing methods. The SRP is too weak to keep developers from adding more lines to an already excessively long method.
This is different in a IODA architecture.[2] There simply is no way to write an operational method with more a dozen lines of code. Because any developer then will tend to extract some logic into a method to be called. But that would be a violation of the IODA rule of not calling any (non-API) functions from an operation. So if a method is extracted from an operation’s logic the operation has to be turned into an integration.
Integration methods are also short. Without any logic being allowed in an integration method it simply does not feel right to have more than a dozen lines of code in it. And it’s easy to extract an integration method if the number of lines grows.
IODA does not only get rid of functional dependencies – of which a palpable effect is, the need for mock frameworks drastically diminishes -, it also creates a force keeping module sizes small.
Software architecture is not a matter of whether a Presentation Layer may only call a Business Layer or Use Cases may call Entities or the other way around. There should not be any service dependencies between these or other kinds of behavioral responsibilities in the first place.
Which behavioral responsibilities there should be is only a function of the concrete requirements. But what is truly universal and unchanging in my view is the distinction between integration, operation, data – and whatever framework APIs deemed helpful to get the job done.
- The same is true for data, by the way. Data structures are black boxes to operations. But since they can contain logic (to enforce their structure and consistency), they might not just consist of operations but a whole IODA hierarchy. ↩
- Which is not just a theoretical thought, but experience from several years of designing software based on IODA. ↩