Ruby on Rails - April 2024
Sajjad Umar
Senior Backend Engineer | Ruby on Rails | Manager RubyConf. Pakistan | Author of Ruby on Rails for Agile Web Development | Desi Developer | Building adex.world
Greetings and welcome to the April edition of the "Ruby on Rails Monthly" newsletter! As we kick off another month, let's take a moment to reflect on recent events and the significance they hold. Ramadan, a time of fasting, reflection, and spiritual growth, has just concluded, and we extend warm wishes to all who observed this sacred month. Additionally, we celebrate Eid-ul-Fitar, marking the joyous culmination of Ramadan's devotion and dedication. With these meaningful observances behind us, let's embark on a new chapter in our journey through the world of Ruby on Rails. Here's to a month filled with learning, collaboration, and exciting developments in our coding endeavors!
I’m Sajjad Umar , your trusted Desi Developer , and I’m thrilled to share some exciting news with you. I've recently completed the final chapter of my upcoming book, "Ruby on Rails for Agile Web Development." This project has been a labor of love, spanning nearly 10 months of dedicated effort to ensure its clarity and accessibility. While I don't have an official release date just yet, rest assured that your copy will be available soon. Keep an eye out for further updates!
With that said, let's dive straight into the latest updates from the Ruby on Rails world.
Rails World Site for 2024 is now live
The website for Rails World happening on September 26 & 27 in Toronto, Canada is now live. Tickets will go on sale this month.
Read all the details here .
Rails Guides Gets a Facelift
When Rails 7.0 landed in December 2021, it came with a fresh new homepage and a new boot screen. The design of the guides, however, has remained largely untouched since 2009 - a point which hasn’t gone unnoticed (we heard your feedback).
With all of the work right now going into removing complexity from the Rails framework and making the documentation consistent, clear, and up-to-date, it was time to tackle the design of the guides and make them equally modern, simple, and fresh.
If you find a bug or wish to submit a suggestion, you can open a discussion on GitHub .
Check out the new design at edge rails guides here.
Solid Queue & Mission Control Gems are Now Officially Part of Rails
Solid Queue has graduated from its incubation under basecamp/ to its future home under Rails/ in anticipation of becoming the default queue in Rails 8.
Read about Solid Queue here.
Read about Mission Control here.
Fixed union select parentheses
This pull request changes Arel::Visitors::ToSql so that SELECT statements in Union and UnionAll nodes get enclosed in parentheses to avoid syntax errors.
To do this, the existing Arel::Visitors::PostgreSQL#grouping_parentheses is generalized and moved to Arel::Visitors::ToSql, where it is now used by infix_value_with_paren.
Read all the details here .
Fixed copying virtual columns when altering a table in sqlite3
When Rails alters a SQLite table, it creates a new table and copies the structure and data from the old one to it.
The problem is that virtual columns are incorrectly copied (copied as classic columns). This PR fixes that.
Read all the details here.
Fixed ActiveJob::EnqueueAfterTransactionCommit API
perform_later is supposed to return the Job instance on success, and false on error.
When the enqueue is automatically delayed, it is impossible to predict if the actual queueing will succeed, but for backward compatibility reasons, it's best to assume it will.
If needed, it is possible to hold onto the job instance and check for #successfully_enqueued? after the transaction has been completed.
Read all the details here.
Allowed to register transaction callbacks outside of a record
A fairly common mistake with Rails is to enqueue a job from inside a transaction, and a record as an argument, which then leads to a RecordNotFound error when picked up by the queue.
This is even one of the arguments advanced for job runners backed by the database such as solid_queue, delayed_job or good_job. But relying on this is undesirable as it makes the Active Job abstraction leaky, and if in the future you need to migrate to another backend or even move the queue to a separate database, you may experience a lot of race conditions of the sort.
But more generally, being able to defer work to after the current transaction has been a missing feature of Active Record. Right now the only way to do it is from a model callback, and this forces moving things in Active Record models that are better done elsewhere.
Also, some 3rd party gems are adding this capability using monkey patches. It's not a reason to upstream the capability, but it's proof that there is a demand for it.
As a part of this PR, ActiveRecord::Base.transaction now yields an ActiveRecord::Transaction object, which allows to register callbacks on it.
Article.transaction do |transaction|
article.update(published: true)
transaction.after_commit do
PublishNotificationMailer.with(article: article).deliver_later
end
end
Also, Added ActiveRecord::Base.current_transaction which also allows to register callbacks on it.
Article.current_transaction.after_commit do
PublishNotificationMailer.with(article: article).deliver_later
end
This PR also adds ActiveRecord.after_all_transactions_commit callback.
Useful for code that may run either inside or outside a transaction and needs to perform works after the state changes have been properly persisted.
def publish_article(article)
article.update(published: true)
ActiveRecord.after_all_transactions_commit do
PublishNotificationMailer.with(article: article).deliver_later
end
end
Read all the details here.
Added query count to template rendering instrumentation
There is often a need to quickly see how many SQL queries the current action produced. For example, to quickly check if N+1 was solved or if the caching is working and so the # of queries reduced, etc. This can be done manually by inspecting the logs and counting the number of queries, but for large actions with tens of hundreds of SQL queries, this is not a simple task.
This PR enhances the log output to also print the query counts.
# Before
Completed 200 OK in 3804ms (Views: 41.0ms | ActiveRecord: 33.5ms | Allocations: 112788)
# After
Completed 200 OK in 3804ms (Views: 41.0ms | ActiveRecord: 33.5ms (2 queries, 1 cached) | Allocations: 112788)
Read all the details here.
Added the ability to ignore counter cache columns while they are backfilling
Starting to use counter caches on existing large tables can be troublesome because the column values must be backfilled separately from the column addition (to not lock the table for too long) and before the use of :counter_cache (otherwise, methods like size/any?/etc, which use counter caches internally, can produce incorrect results). People usually use database triggers or callbacks on child associations while backfilling before introducing a counter cache configuration to the association.
Now after this PR, to safely backfill the column, while keeping the column updated with child records added/removed, use:
class Comment < ApplicationRecord
belongs_to :post, counter_cache: { active: false }
end
While the counter cache is not "active", the methods like size/any?/etc will not use it, but get the results directly from the database. After the counter cache column is backfilled, simply remove the { active: false } part from the counter cache definition, and it will now be used by the mentioned methods.
Read all the details here.
Now it is easier to retry ActionableErrors when running tests
This PR allows Actionable Errors encountered when running tests to be retried.
Migrations are pending. To resolve this issue, run:
bin/rails db:migrate
You have 1 pending migration:
db/migrate/20240201213806_add_a_to_b.rb
Run pending migrations? [Yn] Y # <------------ Interective Terminal
== 20240201213806 AddAToB: migrating =========================================
== 20240201213806 AddAToB: migrated (0.0000s) ================================
Running 7 tests in a single process (parallelization threshold is 50)
Run options: --seed 22200
# Running:
.......
Finished in 0.243394s, 28.7600 runs/s, 45.1942 assertions/s.
7 runs, 11 assertions, 0 failures, 0 errors, 0 skips
This feature is only available in an interactive terminal.
Read all the details here.
Rails now raise a named exception in AbstractMysqlAdapter when DB reports an invalid version
If the MySQL database provides an invalid version string, it will now trigger the raising of the ActiveRecord::ActiveRecordError.
Read all the details here.
Made ActiveSupport::BacktraceCleaner copy filters and silencers on dup and clone
This PR makes ActiveSupport::BacktraceCleaner copy filters and silencers on dup and clone. Previously the copy would still share the internal silencers and filters array, causing the state to leak.
Read all the details here.
Added config.active_record.permanent_connection_checkout setting
This PR adds controls on whether ActiveRecord::Base.connection raises an error, emits a deprecation warning, or neither.
ActiveRecord::Base.connection checkouts a database connection from the pool and keep it leased until the end of the request or job. This behavior can be undesirable in environments that use many more threads or fibers than there are available connections.
This configuration can be used to track down and eliminate code that calls ActiveRecord::Base.connection and migrate it to use ActiveRecord::Base.with_connection instead.
领英推荐
The default behavior remains unchanged, and there are no plans to change the default.
Read all the details here.
Eliminated remaining uses of lease_connection inside Active Record
Part of moving towards adding a new config.active_record.permanent_connection_checkout setting discussed above, this change eliminates the remaining uses of lease_connection inside Active Record API.
Read all the details here.
This PR ensures all needed options are added to the association options list unconditionally
If we make a simple typo like this (t(h)rough):
class User < ApplicationRecord
has_many :courses
has_many :assignments, trough: :courses
end
It raises the following error:
Unknown key: :trough. Valid keys are: :class_name, :anonymous_class, :primary_key, :foreign_key, :dependent, :validate, :inverse_of, :strict_loading, :query_constraints, :autosave, :before_add, :after_add, :before_remove, :after_remove, :extend, :counter_cache, :join_table, :index_errors (ArgumentError)
through is not present in the error which we expect to be noted.
This PR fixes all of these issues for HasOne and HasMany association builders resulting in the following error:
Unknown key: :trough. Valid keys are: :class_name, :anonymous_class, :primary_key, :foreign_key, :dependent, :validate, :inverse_of, :strict_loading, :query_constraints, :autosave, :before_add, :after_add, :before_remove, :after_remove, :extend, :counter_cache, :join_table, :index_errors, :through (ArgumentError)
Read all the details here.
Retry known idempotent SELECT queries on connection-related exceptions
This PR makes two types of queries retry-able by opting into our allow_retry flag:
These changes ensure that queries we know are safe to retry can be retried automatically.
Read all the details here.
Don't enqueue jobs to process a preview image if no variant requires it
This pull request resolves the issue where previewable attachments that didn't specify any variants or variants that required pre-processing still attempted to queue a job for processing the preview image.
Read all the details here.
Allowed primary_key: association option to be composite
Association's primary_key can be composite when derived from associated class primary_key or query_constraints. But we don't allow setting it explicitly even though Rails is already capable of supporting it.
This commit allows primary_key association option to be an array to support this behavior.
Read all the details here.
Do not build View Watcher until the first updated? check
Currently initialization of every Rails::Engine leads to the creation of a new View watcher when the engine prepends its paths.
ActiveSupport.on_load(:action_controller) { prepend_view_path(views) if respond_to?(:prepend_view_path) }
This contributes to the time it takes to perform the first cold request on a lazy-loaded application.
prepend_view_path -> _build_view_paths -> cast_file_system_resolvers -> file_system_resolver_hooks.each(&:call) -> rebuild_watcher
This change delays the initialization of the View Watcher until the first updated? check is performed which leads to only one initialization of the watcher.
Read all the details here.
Do not try to alias on key update when raw SQL is supplied
Following https://github.com/rails/rails/pull/51274/files#r1520277424 , we started getting mixed queries when using raw SQL. For example:
Model.upsert_all(..., on_duplicate: Arel.sql("x=VALUES(x)"))
INSERT table (...) VALUES (...) as values_alias ON DUPLICATE KEY UPDATE x=VALUES(x)
and
Model.upsert_all(..., on_duplicate: Arel.sql("x=x"))
INSERT table (...) VALUES (...) as values_alias ON DUPLICATE KEY UPDATE x=x
Both cases cause:
ActiveRecord::StatementInvalid: Trilogy::ProtocolError: 1052: Column 'x' in field list is ambiguous
In the case of raw SQL, using the old syntax without values aliases (previous behavior) is better since we can't determine whether it is aliased or not. For example:
Model.upsert_all(..., on_duplicate: Arel.sql("x=VALUES(x)"))
INSERT table (...) VALUES (...) ON DUPLICATE KEY UPDATE x=VALUES(x)
Read all the details here.
Memoize key_provider from key or deterministic key_provider if any
Previously, this memoization was removed which led to a performance hit for encrypted attributes. This PR conditionally adds it back.
Read all the details here.
Updated Astana with a Western Kazakhstan timezone
This Pull Request changes Astana (capital of Kazakhstan) to point to the Western Kazakhstan TZInfo identifier Asia/Almaty. After this change, the identifier better matches the expected timezone and UTC offset.
Read all the details here.
railties: configured sanitizer vendor in 7.1 defaults more robustly
To prevent a situation where Rails::HTML::Sanitizer isn't loaded yet, causing the sanitizer vendor to remain as Rails::HTML4 instead of being set to Rails::HTML5 as intended in Rails 7.1, measures have been taken.
Read all the details here.
Supported custom blob key in ActiveStorage::Blob#compose similar to other Blob APIs
Starting from Rails 6.1, Active Storage has enabled the provision of a custom key when attaching a new file. This pull request introduces support for customizing the name of the underlying service object when utilizing the compose class method.
Read all the details here.
Preserved encoding on truncate_bytes
This pull request resolves a problem where String#truncate_bytes might return a string with encoding different from the one being truncated.
Read all the details here.
As we wrap up this month's edition of the "Ruby on Rails Monthly" newsletter, I hope you found the updates insightful and inspiring for your coding endeavors. Remember to stay engaged with the community, keep learning, and continue exploring the exciting world of Ruby on Rails. I'll be back next month with more updates, insights, and resources to fuel your journey. Until then, happy coding!
If you want to support this newsletter, consider subscribing to the paid version on substack here:
Alternatively, you can buy me a coffee here: https://ko-fi.com/sumar7