No release in over 3 years
Low commit activity in last 3 years
Provides `ActiveRecord::Relation#one!`, a convenience method that asserts a relation contains exactly one record and returns it. If there are no records or more than one, an `ActiveRecord::NoRecordFound` or `ActiveRecord::MultipleRecordsFound` exception is raised, respectively.
2005
2006
2007
2008
2009
2010
2011
2012
2013
2014
2015
2016
2017
2018
2019
2020
2021
2022
2023
2024
2025
 Dependencies

Development

Runtime

 Project Readme

activerecord-one

Provides ActiveRecord::Relation#one!, a convenience method that asserts a relation contains exactly one record and returns it. If there are no records or more than one, an ActiveRecord::NoRecordFound or ActiveRecord::MultipleRecordsFound exception is raised, respectively.

#one! reads as follows:

def one!
  case size
  when 0
    raise ActiveRecord::NoRecordFound
  when 1
    first
  else
    raise ActiveRecord::MultipleRecordsFound
  end
end

Why not use find_by?

find_by doesn't verify uniqueness. The Rails documentation for the method says:

Finds the first record matching the specified conditions. There is no implied ordering so if order matters, you should specify it yourself.

Therefore, given the following records:

#<Record id: 1, name: "Alice", color: "red">,
#<Record id: 2, name: "Bob", color: "blue">,
#<Record id: 3, name: "Bob", color: "green">

The following expression has undefined behavior:

Record.find_by(name: "Bob")    # could return record 2 or record 3!

The record returned is whatever the database decides is first. There's no way to tell from the code alone which it will be, or that such a thing has even happened. In other words, find_by is only safe when you specify the order of rows, or have confidence that no more than one row will match—usually thanks to a uniqueness constraint in the database.

If you expect only one row to match but don't have a database uniqueness constraint (perhaps due to non-unique legacy data, for example), you would need to do something like:

raise unless Record.where(name: "Bob").count == 1
Record.find_by(name: "Bob")

Or, with this gem:

Record.where(name: "Bob").one!

Shouldn't this be provided by Rails?

Probably. See rails/rails#26206.

Installation

Add this line to your application's Gemfile:

gem 'activerecord-one'

And then execute:

$ bundle

Or install it yourself as:

$ gem install activerecord-one

License

This gem is dedicated to the public domain. In jurisdictions where this is not possible, this gem is licensed to all under the least restrictive terms possible, and all copyright rights are waived.