Tip #14 - Custom Error Messages in Validations
Tue Apr 22 18:10:20 -0700 2008
If you use Rails, you sometimes get a situation where the custom error messages just don’t work, here is how you can fix it…
Say you have a model with a field called ‘country_iso’ which specifies the ISO value of the country the person belongs to.
Database design concerns aside (you should really have it as country_id and a separate country_iso field) you might be stuck with this situation from a legacy database (like I was).
As this is a required field, you set the following in your model:
1 2 3 |
class User < ActiveRecord::Base validates_presence_of :country_iso end |
So, you go ahead and make a form which presents the country as just a select box, because you don’t want your users to worry about if Thailand is TH or TL, but then you don’t specify a country and the error message comes back to the user in nice big letters:
Country Iso can’t be blank
Ugh!
Aside from the fact that Iso is capitalized badly, how many of YOUR users know that it stands for “International Standards Organization” and that a Country ISO is usually a two letter representation of the country? I’d wager about 3 of them have some idea :)
So you go in and change the validates line to:
1 2 3 |
class User < ActiveRecord::Base validates_presence_of :country_iso, :message => "Country can't be blank" end |
Do your test again, and now you get:
Country Iso Country can’t be blank
Nup.. that’d didn’t really fix it.
Fortunately, the fix is easy:
1 2 3 4 5 |
class User < ActiveRecord::Base validate do |user| user.errors.add_to_base("Country can't be blank") if user.country_iso.blank? end end |
Now, reload your view and you’ll get:
Country can’t be blank
Much better!
blogLater
Mikel
Leave a Reply
Latest posts
- Put your mailer where the action is!
- Why Force a Choice?
- How to make an RSS feed in Rails
- Rails 3 Routing with Rack
- Bundle me some Rails
- Helping out in Haiti
- Watch your self
- Is Rails 3.0 a Game Changer?
- Where did the scripts go?
- validates :rails_3, :awesome => true
- New Rails Version 3.0 Guides Online
- New ActionMailer API in Rails 3.0
- Mail gem version 2 released
- How to rename a Rails 3 Application
- Rails 3.0 Examples
- DECCA Driving Day
- Mail now merged into ActionMailer
- Tip #29 - Stop a Mongrel (or any) Service in Windows
- Ruby on Rails Tips Page
- Monitoring a DAHDI or Zaptel Channel
- Mail gets some compliments!
- Rails Unit Tests: uninitialized constant error
- New Mail gem released
- Mail and Bounced Emails
- Mail, TMail, The Future of Ruby Email Handling
- Custom Music on Hold for Asterisk
- Always getting an invalid authenticity token error
- Windows ipconfig does not show anything
- FreeBSD rc scripts
- How to monitor a logged in professional
- TMail Moves to GIT
- Funny...
- How to reset a sequence with PostgreSQL
- OpenBSD RAID and Temp Sensors on HP Proliant DL 360 and 380 Series
- Terminator - Timeout without Mercy
- Tip #28 - Separate the things that change from the things that stay the same
- Fortune...
- Examples of Behaviour Spec'n
- Tip #27 - Spec a Behaviour, Not an Implementation
- Tip #26 - Start Small
- Tip #25 - Logging is your friend...
- Tip #24 - Being clever in specs is for dummies
- Tip #23 - Know your fundamentals
- Convert Visual Basic / Microsoft Long Integer Color Values to CSS RGB Format
- It's amazing what you find...
- RSpec Story xhr problem
- Tip #22 - How to ask a question about Rails
- Tip #21 - Developer Info On Every Page
- And now for something completely different...
- Rails 2.1 is out
Latest comments
Tag Cloud
AJAX ARGH! ActiveRecord Ajax Apache Apple Asterisk Australia Copy Database Development Feedburner Gem server Google Human Rights Javascript L. Ron Hubbard MS SQL Server MacOSX Mail Mephisto Not Programming OpenBSD Opensource Performance Personal Integrity PostgreSQL Programming Prototype Puzzle RDoc REST RESTful Rails RSPec RSpec Rails Rails Tips Rspec Ruby Ruby on Rails Ruby on Rails Tips Ruby on rails Tips SQL SQLServer SVN Scientologist Scientology Site Stats Soekris Soekris net5501 TMail Textmate Tips Windows World about mikel anti drug apache contributing daemon documentation drugs illustrator javascript lambda mail mephisto newspapers nitro open source opensource photoshop productivity programming railscasts rspec ruby ruby on rails rubyforge scientology seo sitemap sqlserver tips tmail tom cruise unix tricks vector graphicsArchives
- November 2009 (1)
- October 2009 (2)
- September 2009 (2)
- August 2009 (0)
- July 2009 (1)
- June 2009 (0)
- May 2009 (1)
- April 2009 (0)
- March 2009 (0)
- February 2009 (0)
- January 2009 (2)
- December 2008 (0)
- November 2008 (5)
- October 2008 (0)
- September 2008 (1)
- August 2008 (0)
- July 2008 (2)
- June 2008 (13)
- May 2008 (7)
- April 2008 (18)
- March 2008 (8)
- February 2008 (5)
- January 2008 (7)
- December 2007 (20)
- November 2007 (22)

Wed May 07 03:57:55 -0700 2008
Hey Great Tip! Thanks for the post. Any idea how to make sure that the form field remains tagged properly with the appropriate error class so that the css rule for an error field will be applied properly? ~Ben
Sun May 11 17:46:07 -0700 2008
Ben: You could do http://henrik.nyh.se/2007/12/full-error-messages-without-prepended-attribute-name.
I believe another way would be to add_to_base and then add a nil error to the attribute: user.errors.add(:country_iso, nil)
nil errors should not be listed in text but still get you the CSS error highlighting.
Sun May 11 19:49:58 -0700 2008
Ben: You could also do http://henrik.nyh.se/2007/12/change-displayed-column-name-in-rails-validation-messages
Sun May 11 20:19:58 -0700 2008
Henrik: Nice solution! Go read Henrik’s solution, it’s quite neat!
Sun May 11 22:45:46 -0700 2008
Is there a way to change the table name as well for the error message? For example, I have an STI model “class Group < Assembly” so when there is an error saving a Group, the error message says “2 errors prohibited this assembly from being saved” but I would like it to say “3 errors prohibited this group from being saved”.
Wed May 14 04:02:43 -0700 2008
The way I do this is by specifying full error messages for all my attributes, and then using a custom FormBuilder that displays the messages with the field (i.e., I do not use error_messages_for). With this approach, you get the benefit of having the error messages associated with the invalid attribute.
I think the default messages are kind of like scaffolding: good for getting started, but I wouldn’t use them in a real app.
Sun May 18 00:31:29 -0700 2008
I would suggest better way:
class User < ActiveRecord::Base @@field_names = { :country_iso => ‘Country’ } cattr_reader :field_names
...
Why? Because you can use all validates_… methods and regular error processing (how are you going to use validates_uniqueness_of?). Drawback – translation is application-wide, so User.country_iso and Session.country_iso will be translated exactly same way. Bad for existing applications where you can not choose fields.
Sun May 18 01:12:08 -0700 2008
In fact I should have put this code to ApplicationController along with ActiveRecord::Errors.default_error_messages = { ...
Mon May 19 15:14:42 -0700 2008
Great! I’d like to ask if there is a way to change default texts in header and bellow header:
“5 errors prohibited this order from being saved There were problems with the following fields:”
Reason – localization into different language Thanx
Tue May 20 21:16:08 -0700 2008
yea Peter go check out this page: http://www.softiesonrails.com/2008/4/23/better-messages-for-activerecord-validation-errors
Wed May 21 09:56:15 -0700 2008
As for me I am using
begin @User.save! rescue Exception => e flash.now[ :user_errors ] = e.message render :action => :new end
Might not be the best thing of course, but resulting error message is concise and helps in case of other errors, not just database.
Wed May 21 10:09:28 -0700 2008
Still don’t know how to paste code. You can check here - http://pastie.textmate.org/201339
Meanwhile working example of my previous post can be found here (full translation of error messages & field names) – http://pastie.textmate.org/201340
I happened to be a bit more complex than I expected.
Wed Nov 26 11:32:12 -0800 2008
How to use it in error_messages_for ???