Upgrading Rails from 4.2 to 5.2, 6.0 - collected notes

zealot128

Stefan Wienert

Posted on January 14, 2020

Upgrading Rails from 4.2 to 5.2, 6.0 - collected notes

Recently, I've upgraded a bunch of apps to Rails 5.2 and 6.0, I want to share some common issues and the process I follow:

General upgrade process

  1. Have a quick overview of the Rails upgrade guide, to get a feeling what kind of things changed between versions:
  2. I change the Rails requirement in Gemfile to the next minor version, like gem "rails", "~> 5.0.0", Maybe also bump the Ruby version to the required one, if your current one is too old. But I try to usually do it separately, because newer Ruby versions can also break your app.
  3. Now comes the bundle update rails, This will probably fail because some other Gems collide. If those Gems have version constraints in your Gemfile, relax that. Try adding more and more Gems to bundle update rails rack sass-rails etc., Use a bundle update without Gems only as last resort, as this will update ALL Gems at once and usually break a lot of things.
    In my experience, common Gems that have to be upgraded together with Rails, are:

    • paginators (will_paginate)
    • form gems (simple-form)
    • ActiveRecord Gems (paper_trail, acts-as-taggable-on)
    • Admin-interfaces (Active Admin)
    • Asset-Pipeline Gems (sass, sass-rails, font-awesome-rails)
    • bootsnap, rspec-rails
  4. Try to boot app, like bundle exec rails c, Fix any errors that happen, then

  5. Run rails app:update. The will launch a command line wizard, that will interactively compare all the config files of your app to that of a newly generated Rails app. My tip: Try to adjust your files in a editor to match the Rails standard as closest as possible. E.g. our config/environments/development.rb looks almost the same like the generated. All custom config by us is then on the bottom of the file and can be easily moved between upgrades later. Alternatively, use Rails-Diff to compare config changes between versions.

  6. Try to run tests, fix deprecations

  7. Run development server and click around

  8. Push to CI. Hint: Try to make deprecations more visible, make them an error, and eager load on test (to remove bugs that happen by invalid files that the development system didn't catch):

# config/environments/test.rb

    config.eager_load = ENV['CI'].present?
    config.active_support.deprecation = ENV['CI'].present? ? :raise : :stderr
    # Show the full stacktrace of deprecations, 
    #   e.g. middleware. Maybe put that line in application.rb after loading rails
    ActiveSupport::Deprecation.debug = true
  end
Enter fullscreen mode Exit fullscreen mode

Rails 5.0

Protected Attributes is gone!

If you previously have used attr_accessible and similar, that stuff is gone!

Finding possible occurences:

ag new.\*params app | grep -v permit
ag update.\*params app | grep -v permit
Enter fullscreen mode Exit fullscreen mode

Sometimes, e.g. for query models on a get request, this kind of pattern is useful:

# passing whole params
MyForm(params.permit!.to_h)

# passing only params, that might not be there on the first page load
MyForm(params[:my_form]&.permit!&.to_h)
Enter fullscreen mode Exit fullscreen mode

Important: Belongs_to required by default!

This will probably brake old apps. To reduce friction for future upgrades, adjust config/application.rb:

module MyApp
  class Application < Rails::Application
    # Initialize configuration defaults for originally generated Rails version.
    config.load_defaults 5.0
    ...

    config.active_record.belongs_to_required_by_default = false
  end
end
Enter fullscreen mode Exit fullscreen mode

Some later Rails generator brings this line into a config/initializers, but that seems to be having no effect, only adding to application.rb.

before_filter -> before_action

(Same with after_action)

sed -i 's/before_filter/before_action/g' `find app -type f`
sed -i 's/after_filter/after_action/g' `find app -type f`
Enter fullscreen mode Exit fullscreen mode

Controller tests/specs - get/post must be keyword arguments

In case you are using RSpec, just use this awesome Gems to convert everything:

gem install rails5-spec-converter
rails5-spec-converter
Enter fullscreen mode Exit fullscreen mode

This converts 95% of the scenarios, only very custom session/cookies stuff must be checked manually.

redirect_to :back deprecated

- redirect_to :back, alert: "whatever"
+ redirect_back fallback_location: '/', alert: "whatever"
Enter fullscreen mode Exit fullscreen mode

Rails 5.1

  • image_tag does not allow nil! nil is not a valid asset source, wrap all image_tag in an if my_model.attachment.present?
  • database_cleaner is not required for browser tests anymore (Rspec: System Tests)
  • add 'listen' gem to development/test group gem "listen"

response.success? -> response.sucessful?

sed -i 's/be_success$/be_successful/g' `ag be_success$ spec -l`
Enter fullscreen mode Exit fullscreen mode

Foreign Key mismatch

ActiveRecord::MismatchedForeignKey: Column `cooperation_id` on table `cooperation_data_points` does not match column `id` on `cooperations`, which has type `bigint(20)`. To resolve this issue, change the type of
the `cooperation_id` column on `cooperation_data_points` to be :bigint. (For example `t.bigint :cooperation_id`).
Original message: Mysql2::Error: Cannot add foreign key constraint: ALTER TABLE `cooperation_data_points` ADD CONSTRAINT `fk_rails_3979ee89c8`
FOREIGN KEY (`cooperation_id`)
  REFERENCES `cooperations` (`id`
Enter fullscreen mode Exit fullscreen mode
  • Problem: db/schema.rb does not specify correct primary key types (integer vs. bigint)
  • Solution: Run rails db:schema:dump

Rails 5.2

Arel.sql

All order and pluck columns should be checked and any non-trivial statement must be wrapped in Arel.sql:

- .order('length(name) asc').first
+ .order(Arel.sql('length(name) asc')).first
Enter fullscreen mode Exit fullscreen mode

Find occurences of order/pluck with a string, check if there is function call or even an SQL injection possibility :) (like order(params[:sort]))

  • ag "order\('"
  • ag 'order\("'
  • ag "pluck\('"
  • ag 'pluck\("'

Rails 6.0

One very obvious error was the introduction of host checking (against DNS rebinding attacks). We don't need that, as all our apps are proxied in production. In addition, we have dynamic hostnames in development for every developer, so we disable that:

# config/application.rb
config.hosts.clear
Enter fullscreen mode Exit fullscreen mode

Other things we noticed:

  • scope - cannot contain the class name itself, just plain calls to where/order etc. (no scope :active { User.where(active: true) })
  • update_attributes -> update simple sed
  • Tests/specs: content_type -> media_type
  • Gems:
    • in one project, that update line succeeded first: bundle update rails draper devise sass-rails font-awesome-rails annotate premailer-rails
    • other gems that produced errors while booting: bundle update coffee-rails slim-rails bullet pry-rails pry-rescue bootsnap airbrake rspec-rails
    • gem 'rspec-rails', '~> 4.0.0.beta2'
    • gem "cancancan", "~> 3.0"
    • will-paginate must be upgraded
  • If you are needing helpers in non-controller/view files, there was an older snippet on Stackoverflow, to use ActionView::Base.new, replace with: ActionView::Base.new(ActionView::LookupContext.new('.'), {})

Ruby 2.4+

Meanwhile, if you also upgrading Ruby, one error we ran into:

webmock > 2, vcr > 3

undefined method `<3, :continue_timeout=>nil, :debug_output=>nil}:Hash
Enter fullscreen mode Exit fullscreen mode

If you had running VCR version less than 3 and used HTTP Basic auth, you need to convert your cassettes: https://gist.github.com/glaszig/9170b1cf2186674faeead74a68606c5d

Deprecations Fixnum/Bignum -> Integer

  • Substitute your own usages of "Fixnum" with Integer.
  • Upgrade all Gems with the error

So far, I will update that post if I find more common stuff.

💖 💪 🙅 🚩
zealot128
Stefan Wienert

Posted on January 14, 2020

Join Our Newsletter. No Spam, Only the good stuff.

Sign up to receive the latest update from our blog.

Related