Treat Rails for What it Is and Organize Code By Structural Purpose
October 25, 2022 📬 Get My Weekly Newsletter ☞
If I had one piece of advice for using Rails, it is to treat Rails for what it is, not what you might like it to be. This was the subject of my talk at Rails Conf 2022, but it has a few practical implications for, among other things, how to organize the code in your Rails app.
No matter how simple your app is, it will have business logic, and you will have a boundary
between your controllers/jobs/mailboxes/tasks and that logic. The code that defines that
boundary is called a service layer. It should go in app/services
, because Rails organizes
code by structural purpose, not domain purpose. Following Rails conventions will make
development of your app easier to sustain over time.
An App’s Architecture Starts with Code Organization
The reason the question “where does this code go?” is important to answer is because it’s the foundation of the application’s architecture. If developers don’t know (or can’t agree on) where code is supposed to go, sustained development is going to be difficult.
Rails is a framework that will manage most of the classes you need to build an app: HTTP, Email, database, background jobs, etc. Rails does not provide any specific way to manage your core domain or business logic. The only feature it provides is the ability to auto-load classes in app/«whatever»
.
The easier it is to answer the question “where does the code go?” the easier it will be to work on your app over time and through change (in requirements, team, etc). If answering this question is difficult, change is harder. Questions that are difficult to answer create more friction than questions that are easy to answer, and you want to reduce friction, especially around questions that must be answered before coding can start.
Fortunately, Rails does provide an easier answer for almost all of the code you have to
write: controllers go in app/controllers
, Active Records go in app/models
, Mailers go in app/mailers
, and so on. Developers don’t need to do a lot of analysis to figure out where that sort of code goes. So it should be with business logic.
Business (or perhaps domain) logic doesn’t fit into any of the Rails-managed classes, and Rails doesn’t provide an answer for where this code goes. I find it useful to acknowledge the boundary between Rails-managed classes and business logic, and I find the best term for this boundary to be service layer.
Business Logic Should be Encapsulated
Even if you don’t explicitly define a boundary between your controllers and business logic, it doesn’t mean it doesn’t exist, at least conceptually. This boundary is called a service layer, which Martin Fowler defines thusly, emphasis mine:
A Service Layer defines an application’s boundary and its set of available operations from the perspective of interfacing client layers. It encapsulates the application’s business logic, controlling transactions and coordinating responses in the implementation of its operations.
The highlighted section is important. The service layer encapsulates the business logic from “clients”. In a Rails app, a client is a controller, mailbox, task, or background job. They invoke business logic and interpret its results. The service layer is where this happens.
This does not imply that the service layer contain all the logic. It is just a boundary. It encapsulate whatever the logic is.
Rails developers often fail to create an explicit service layer, and have methods littered all over the place—often on Active Records—that trigger (and implement) business logic. This is needlessly confusing and hard to manage over time. Having instead a single place where business logic is invoked makes everything easier.
Inside app/services
, because of encapsulation, you are free to organize the code however you
like. If you prefer stateless procedures, you can do that. If you prefer a rich collection of
objects passing messages, you can do that, too. If you need to create subdirectories for domain
concepts, you can do that as well. This is the primary benefit of encapsulation and, because
this code is tucked into app/services
, it also is consistent with Rails’ conventions.
What About Decorators and Other Classes?
Most apps should not need more than app/services
plus what Rails gives you, but if you do end
up having a lot of classes that conform to some structural purpose, you can certainly create a
directory in app
to store them.
This is what Rails intends you to do, and creating a directory like app/decorators
is a clear way to communicate that there is a concept of a decorator, and that the way it is constructed is important to be consistent.
This provides an easy answer to “where do decorators go?” and is also consistent with how Rails wants you to structure your code. The more easy answers your architecture provides, the better.
What About lib
?
There is a second type of code that is particular to your app but doesn’t fall into a
Rails-managed class or business logic, and that is the code that should go in lib
. This
code is often infrastructure-type code like middleware or plugins. lib
can also hold code you
intend to extract as a gem in the future.
This convention follows the policy we’ve been discussing: easy answer to where code goes. If you
need to create some code that is not business logic and does not go in a Rails-managed class, it
goes in lib
.
What About Organizing by Domain Concept?
There are advantages to organizing code by domain concept instead of structural purpose. For
example, you might want app/shopping
to contain all the code about purchasing from your store
and app/reporting
to contain all the reporting.
To entirely organize your app this way requires quite the configuration feat with Rails and would obviate many of Rails’ benefits. It also creates a far more difficult-to-answer question of where code goes. Is there an existing concept where this new code should go? Is there an existing concept that is close, and if we rename it would this code go there? Or, does adding this code to an existing concept make that concept too complex such that it requires splitting up into two smaller concepts?
These questions can be hard to answer, especially if the app is undergoing rapid change. You may
not know what concepts the app will need or if a concept will be developed beyond the initial
feature. When organizing by structural purpose you can always safely put the code in
app/controllers
or app/services
or wherever, and organize it later.
The ability to organize later is powerful: it’s much easier to organize code that exists and is tested than it is to try to predict where code should go in the future. The contents of your structurally-based folders will show you exactly what concepts are important and which ones aren’t.
When app
contains structurally-organized directories and those directories contain
domain-organized code, you get the best of both worlds. You can group by domain concept, but
also easily answer the question of where code goes. You also have an escape hatch if the
question is too hard to answer: put it in the top of the relevant directory.
Treat Rails for What it Is
Rails is a web framework that organizes code by structural purpose. It provides rudimentary tools
for adding your own new structural purposes, and does not prescribe how the code inside should be
organized. Thus, define your service layer explicitly in app/services
, then organize the code
in there—as well as your domain logic—however you see fit. If you always try to treat Rails for
what it is—not what you might like it to be—your app’s architecture will provide a solid
foundation for sustainable development.