Good code has a lot to do with how readable it is. As developers, we more often read code than write it. As my Perl teacher told us many times: the flexibility of Perl's syntax was its best and worst trait at the same time. Ruby's syntax was influenced partly by Perl and is also quite flexible.
Whatever language you pick, set some guidelines to avoid overusing a language's flexibility. Style guides for Ruby abound on the web, and it's not difficult to pick a style nowadays. But there is not much point in having copious debates on style and whether a proposed change follows a guide. Style enforcement is best left to a tool.
Such tools are called linters and static code analyzers. The de facto standard for Ruby in that category is RuboCop. Here follows an introduction to RuboCop: what it is, how it helps developers, how to use it, and some key practical use cases.
What Is RuboCop for Ruby?
First, RuboCop is a great tool to lint a code base and ensure a specific coding style is followed (such as indentation, spacing, naming conventions, etc.). It's heavily configurable on that level through many rules, called "cops". Each can be activated (or not) and tailored (for example, the number of characters per line).
Ensuring a specific coding style is followed helps reduce developers' cognitive load in writing and reading code. It's not a matter of code looking "nice"; it's rather a matter of code looking similar to the rest of the code base.
The second side of RuboCop is its ability to analyze code statically for quick insights into code complexity and potential code issues, such as the misuse of variables.
As RuboCop is a little utility, it's easily run within a CI pipeline and in any developer's IDE. Thus, RuboCop's feedback can be integrated into the key places where it matters: when code is being written and before it's merged into the code base.
How to Use RuboCop for Ruby
Style is a matter of taste, so it is entirely subjective. Remember the main point behind using a linter: it's just there to help follow a chosen style because it gives us more capacity to do what matters in any project.
By default, RuboCop will enforce the style defined in the Ruby Community Style Guide. We can tailor it to our specific tastes and context, but let's rely on this basic set of rules to learn how to use RuboCop.
Simple Use of RuboCop
Adding RuboCop to a project is as simple as adding the rubocop
gem to the development and test group in the Gemfile.
group :development, :test do gem 'rubocop', require: false end
There are also several complimentary gems you can use, including:
Those last two will be useful for many of you.
Once the gem, or gems, is installed, we can run rubocop
from within the root folder of the Ruby application.
$> rubocop *
This will print out a report of all "offenses" and how many can be automatically fixed.
For example:
$> rubocop * Inspecting 807 files ...CCC.................... Offenses: app/controllers/api/messages_controller.rb:46:25: C: Naming/MethodParameterName: Method parameter must be at least 3 characters long. 80 files inspected, 20 offenses detected, 7 offenses autocorrectable
Let's dissect the only offense we have kept in this example.
RuboCop's documentation explains that this issue:
Checks method parameter names for how descriptive they are. It is highly configurable.
This is followed by an explanation of the configuration that can be made for this specific rule (we will cover that soon).
What will be of interest to most are the examples of good and bad styles, as checked by this rule:
# bad def bar(varOne, varTwo) varOne + varTwo end # With `AllowNamesEndingInNumbers` set to false def foo(num1, num2) num1 * num2 end # With `MinNameLength` set to number greater than 1 def baz(a, b, c) do_stuff(a, b, c) end # good def bar(thud, fred) thud + fred end def foo(speed, distance) speed * distance end def baz(age_a, height_b, gender_c) do_stuff(age_a, height_b, gender_c) end
We can apply this advice and rewrite the code to match the required style. RuboCop can often fix the issue it has found (we will look at this a bit later).
Configuring Rules in RuboCop
By default, out of the box, RuboCop comes with a default set of pre-configured rules. The documentation will tell you Rubocop's default rules.
Configuring RuboCop for a project is done through the .rubocop.yml
file at the project's root. This can be tailored further in sub-folders, but I wouldn't recommend it.
And, of course, it's also possible to have system-wide configuration files within the user's home folder: ~/.rubocop.yml
or ~/.config/rubocop/config.yml
.
Here is a simple example of such a configuration:
Layout/LineLength: Max: 99
Let's reuse the previous rule: Naming/MethodParameterName
.
Naming/MethodParameterName: MinNameLength: 3 AllowedNames: - as - at - in - ip - id - to
There are a lot of things that can be tailored in the RuboCop configuration file. The documentation can help you pick the rules you want to use, but a few other options are available to give you a starting point beyond the default rules.
The Ruby style guide matches RuboCop's default configuration, for example. But you can also find other guides like the Relaxed Ruby Style, and some companies - such as Shopify - share theirs.
Arguments in RuboCop
RuboCop can be run without arguments for a basic check, but we can add on a few arguments for different effects:
-a
: to auto-correct offenses, but only when it's safe to do so.-A
: to auto-correct offenses, even when it's not safe to do so.-E
: to display extra details for each offense listed.-S
: to display style guide URLs in the offense messages.
Those are the main options we might be interested in. -E
and -S
are great to add whenever you run RuboCop to check the code base. They both give more details for developers to figure out what to do.
-a
and -A
are handy to quickly take care of the "small" issues, but beware of the automatic side. It might do well enough for minor style offenses, such as spaces, but other things might not be that simple.
Bending the Law
In some rare cases, you might want a RuboCop rule or rules to be disabled around one or several lines in a file. This is generally frowned upon, as it creates a precedent many developers won't hesitate to replicate. Yet it might still be useful to know how to do this hack for certain cases.
The way to disable a rule and then re-enable it is to use a pair of comments:
# rubocop:disable Layout/ClassStructure, Style/AndOr [...] # rubocop:enable Layout/ClassStructure, Style/AndOr
This will disable and enable two rules: Layout/ClassStructure
, and Style/AndOr
.
In the previous example, you can also disable whole sections (or departments) of rules by using the department name: Layout
or Style
.
Finally, you can also disable all the rules in one go.
# rubocop:disable all [...] # rubocop:enable all
RuboCop in Practice
Now that we have seen what it is and how to use it, let's review how we might use RuboCop in our day to day.
Local Uses
As discussed before, a linter's purpose is to ensure a standard style is used throughout a code base. Standards are how we help each other work together without putting up complex conversion and compensation schemes. When building a house, standards are the way every carpenter knows how to work together. The actual details will be a bit different from one house to another, but, using the standard, the work different carpenters do will match.
RuboCop helps software engineers work together by keeping their code up to a certain standard style.
The simplest way to use RuboCop is to run it before making a commit or once in a while before pushing commits to a remote repository. As described in the first part of this post, using rubocop *
will run a check of the code base and display any offenses for you to fix.
You can then use the -a
or -A
option on another run to get some or all offenses fixed by RuboCop.
$> rubocopo -a
You can also fix them yourself, of course.
Within an IDE
Some IDEs and text editors have RuboCop plugins, so you can get notices about style offenses while working directly on code. The only issue with these is the configuration, which can be fiddly.
Integration with CI
It's important to run RuboCop within CI pipelines.
Usually, we run linters such as RuboCop as part of a pipeline's first steps to notify the author of code changes about any offense quickly. We also consider such infractions important, so they will block the Pull or Merge request in its tracks. A merge should not be possible if the style is not correct.
If your project relies on GitHub's Actions, you can find a RuboCop action. You can build similar steps for any CI/CD pipeline out there. The principle is simple: run RuboCop against the code base as an early block within the pipeline; if there is an infraction, it will return a non-zero code, and the CI will treat that as a failure. The output can be used more easily as an artifact.
The -f
option will allow you to tailor the output format. This can be practical when running RuboCop in the CI pipeline: in markdown, HTML, or GitHub. I'd advise starting with markdown or HTML, but you should also consider the fairly readable default format.
Adopting RuboCop in a New Project
Many projects start without a coding style or diverge from one for some reason. Getting such code bases back in the fold is often challenging and hard work.
The first run of RuboCop in such a context will usually output a fairly long list. Trying to solve all offenses at once is foolish. Instead, RuboCop allows us to generate a to-do list.
Running RuboCop with the --auto-gen-config
option will generate the todo
file. This file will disable any rule that is not respected. It's located within the project folder .rubocop_todo.yml
.
Disabling all the rules that are not respected rather than fixing them might seem counterproductive. But we don't stop here. This step ensures we don't face a mountain of work. We can remove a rule from the ignore list and fix all offenses against it, working our way, one rule at a time, through the to-do list and the code base.
This approach — of ignoring infractions to one or more rules within multiple files and then fixing the infractions one by one in many files — is doable, but will require a lot of back and forth between files. You might end up changing several files and several lines multiple times over the course of the process.
Another method is to use a "per file" approach. Using the following syntax within the to-do list file, you will ignore infractions to all cops in the files listed.
AllCops: Exclude: - "app/models/user.rb" - "app/controllers/users_controller.rb"
You can work your way through those files one by one: removing the file from the list, running RuboCop against it, and then fixing all offenses.
Whichever approach you pick, you must remember this is a step towards improvement through continuous learning and effort. You need a strategy and vision that works well with that: include enough Slack time within the team schedule, or set specific times aside to work on this issue.
Making RuboCop Easier to Use with Standard Ruby
RuboCop and coding style guides have helped us determine standards for each project and team. We should only debate what code is doing, not how it looks.
Some Ruby developers have made the point that we can even forego adjusting RuboCop's configuration. The idea is simple: after so many years working with Ruby, we more or less know how Ruby code should look, so a style that works for most will be quite okay.
This approach is known as Standard Ruby. It can also be completed with plugins, including one for Ruby on Rails projects.
Wrapping Up
RuboCop is, alongside Bundler, a good friend to any Ruby developer. As we have seen, it's easy to run it locally to get a list of offenses against a project's style standard.
We have also seen how RuboCop can fix a lot of those offenses on its own, and how to work our way through a long list of offenses if we start from a large backlog.
RuboCop is a tool in your toolbox; code style is only meant to help a team work together and shouldn't be a perpetual topic of debate. Yet, we have also seen that if you find it tedious to figure out a style for your project, you should look into projects such as Standard Ruby.
Happy coding!
P.S. If you'd like to read Ruby Magic posts as soon as they get off the press, subscribe to our Ruby Magic newsletter and never miss a single post!