We aim to write code that is a pleasure to read, and we have a lot of opinions about how to do it well. Writing great code is an essential part of our programming culture, and we deliberately set a high bar for every code change anyone contributes. We care about how code reads, how code looks, and how code makes you feel when you read it.
We love discussing code. If you have questions about how to write something, or if you detect some smell you are not quite sure how to solve, please ask away to other programmers. A Pull Request is a great way to do this.
When writing new code, unless you are very familiar with our approach, try to find similar code elsewhere to look for inspiration.
In general, we prefer to use expanded conditionals over guard clauses.
# Bad
def todos_for_new_group
ids = params.require(:todolist)[:todo_ids]
return [] unless ids
@bucket.recordings.todos.find(ids.split(","))
end
# Good
def todos_for_new_group
if ids = params.require(:todolist)[:todo_ids]
@bucket.recordings.todos.find(ids.split(","))
else
[]
end
endThis is because guard clauses can be hard to read, especially when they are nested.
As an exception, we sometimes use guard clauses to return early from a method:
- When the return is right at the beginning of the method.
- When the main method body is not trivial and involves several lines of code.
def after_recorded_as_commit(recording)
return if recording.parent.was_created?
if recording.was_created?
broadcast_new_column(recording)
else
broadcast_column_change(recording)
end
endWe order methods in classes in the following order:
classmethodspublicmethods withinitializeat the top.privatemethods
We order methods vertically based on their invocation order. This helps us to understand the flow of the code.
class SomeClass
def some_method
method_1
method_2
end
private
def method_1
method_1_1
method_1_2
end
def method_1_1
# ...
end
def method_1_2
# ...
end
def method_2
method_2_1
method_2_2
end
def method_2_1
# ...
end
def method_2_2
# ...
end
endShould I call a method do_something or do_something!?
As a general rule, we only use ! for methods that have a correspondent counterpart without !. In particular, we don’t use ! to flag destructive actions. There are plenty of destructive methods in Ruby and Rails that do not end with !.
We don't add a newline under visibility modifiers, and we indent the content under them.
class SomeClass
def some_method
# ...
end
private
def some_private_method_1
# ...
end
def some_private_method_2
# ...
end
endIf a module only has private methods, we mark it private at the top and add an extra new line after but don't indent.
module SomeModule
private
def some_private_method
# ...
end
endWe model web endpoints as CRUD operations on resources (REST). When an action doesn't map cleanly to a standard CRUD verb, we introduce a new resource rather than adding custom actions.
# Bad
resources :cards do
post :close
post :reopen
end
# Good
resources :cards do
resource :closure
endIn general, we favor a vanilla Rails approach with thin controllers directly invoking a rich domain model. We don't use services or other artifacts to connect the two.
Invoking plain Active Record operations is totally fine:
class Cards::CommentsController < ApplicationController
def create
@comment = @card.comments.create!(comment_params)
end
endFor more complex behavior, we prefer clear, intention-revealing model APIs that controllers call directly:
class Cards::GoldnessesController < ApplicationController
def create
@card.gild
end
endWhen justified, it is fine to use services or form objects, but don't treat those as special artifacts:
Signup.new(email_address: email_address).create_identityAs a general rule, we write shallow job classes that delegate the logic itself to domain models:
- We typically use the suffix
_laterto flag methods that enqueue a job. - A common scenario is having a model class that enqueues a job that, when executed, invokes some method in that same class. In this case, we use the suffix
_nowfor the regular synchronous method.
module Event::Relaying
extend ActiveSupport::Concern
included do
after_create_commit :relay_later
end
def relay_later
Event::RelayJob.perform_later(self)
end
def relay_now
# ...
end
end
class Event::RelayJob < ApplicationJob
def perform(event)
event.relay_now
end
end