Ruby on Rails - April 2024
Ruby on Rails - Monthly by Sajjad Umar | April 2024

Ruby on Rails - April 2024

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:

  1. SELECT queries we construct by walking the Arel tree via #to_sql_and_binds. We use a new retryable attribute on collector classes, which defaults to true for most node types, but will be set to false for non-idempotent node types (functions, SQL literals, update/delete/insert statements, etc). The retryable value is returned from #to_sql_and_binds and used by #select_all and passed down the call stack, eventually reaching the adapter's #internal_exec_query method.
  2. #find and #find_by queries with known attributes. We set allow_retry: true in #cached_find_by, and pass this down to #find_by_sql and #_query_by_sql.

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:

Subscribe now

Alternatively, you can buy me a coffee here: https://ko-fi.com/sumar7




要查看或添加评论,请登录

Sajjad Umar的更多文章

  • Ruby on Rails - Oct 2024

    Ruby on Rails - Oct 2024

    The only Ruby on Rails newsletter you will ever need! Hey Ruby folks?—?this is a fascinating month for the Ruby on…

    1 条评论
  • Ruby on Rails - September 2024

    Ruby on Rails - September 2024

    Welcome to the September 2024 edition of Ruby on Rails Monthly, this is Sajjad Umar your own Desi Developer. I have a…

    1 条评论
  • Ruby on Rails - Aug 2024

    Ruby on Rails - Aug 2024

    It's the only Ruby on Rails newsletter you will ever need! Rails 7.2 is out! There has been close to 2,500 commits made…

  • Ruby on Rails - July 2024

    Ruby on Rails - July 2024

    The only Ruby on Rails newsletter you will ever need! Welcome to the latest edition of the Ruby on Rails Monthly…

  • Ruby on Rails?—?June 2024

    Ruby on Rails?—?June 2024

    The only Ruby on Rails Newsletter you will ever need! Welcome to the latest edition of the Ruby on Rails Monthly…

  • Ruby on Rails - May 2024

    Ruby on Rails - May 2024

    Greetings and welcome to the May edition of the "Ruby on Rails Monthly" newsletter! As we kick off another month of…

  • Ruby on Rails - March 2024

    Ruby on Rails - March 2024

    Welcome to the March edition of "Ruby on Rails, Monthly" newsletter for 2024! As we delve into this month's updates…

  • Ruby on Rails - February 2024

    Ruby on Rails - February 2024

    Greetings, As we dive into the February edition of the "Ruby on Rails, Monthly" newsletter for 2024, let's infuse our…

    1 条评论
  • Ruby on Rails - January 2024

    Ruby on Rails - January 2024

    Greetings, Happy New Year! Welcome to the first edition of the "Ruby on Rails, Monthly" newsletter for 2024! I hope…

    2 条评论
  • [FULL] Ruby on Rails - December 2023

    [FULL] Ruby on Rails - December 2023

    Greetings, Welcome to this month's edition of the "Ruby on Rails, Monthly" newsletter! As always, I'm Sajjad Umar, your…

社区洞察

其他会员也浏览了