(comments inline)
Hello
I'm wondering if someone can put me back on the right track?
I've started playing with Rails for the last 2 days while reading
through the Agile Web Dev with Rails book. I'm in Chapter 7 - Unit
Testing and Validation. The book demos work fine, but I'm now also
doing my own sample app to reinforce what I'm reading at the same
time.
I'm trying to do some Unit Tests on my User Model to check the length
of the password attribute.
Some of the assertions are failing unexpectedly and I'm now at a loss.
Can someone please help me find what I'm overlooking?
Here's the Terminal output from running the Unit Tests:
********************************
turgs:snapshots tim$ rake test:units
Loaded suite /usr/local/lib/ruby/gems/1.9.1/gems/rake-0.9.2/lib/rake/
rake_test_loader
Started
....FF.
Finished in 0.531600 seconds.
1) Failure:
test_Password_must_be_long_enough(UserTest) [/users/tim/Sites/
snapshots/test/unit/user_test.rb:31]:
Failed assertion, no message given.
2) Failure:
test_Password_must_be_shorter(UserTest) [/users/tim/Sites/snapshots/
test/unit/user_test.rb:47]:
Failed assertion, no message given.
7 tests, 25 assertions, 2 failures, 0 errors, 0 skips
Test run options: --seed 21159
rake aborted!
Command failed with status (1): [/usr/local/bin/ruby -I"lib:test" "/
usr/loc...]
Tasks: TOP => test:units
(See full trace by running task with --trace)
turgs:snapshots tim$
********************************
Here's the User model code:
********************************
class User < ActiveRecord::Base
has_many :accounts, :dependent => :destroy
validates :email, :presence => true
validates :name, :presence => true
validates :password, :presence => true, :length =>
{ :minimum => 8, :maximum => 2000, :message => 'should be between 8
and 2000 characters' }
validates :security_question, :presence => true
validates :security_answer, :presence => true
validates :mobile_phone, :length =>
{ :minimum => 8, :maximum => 30, :message => 'should be between 8 and
30 characters'}
end
********************************
Here's the Unit Test user_test.rb file:
********************************
require 'test_helper'
class UserTest < ActiveSupport::TestCase
test "User attributes must not be empty" do
user = User.new
assert user.invalid?
assert user.errors[:email].any?
assert user.errors[:name].any?
assert user.errors[:password].any?
assert user.errors[:security_question].any?
assert user.errors[:security_answer].any?
assert user.errors[:mobile_phone].any?
end
test "Password must be long enough" do
user = User.new(:email => "adam@example.com",
:name => "Adam",
:security_question => "What's my name?",
:security_answer => "Adam")
user.password = "1234567"
assert user.invalid?
assert_equal "should be between 8 and 2000 characters",
user.errors[:password].join('; ')
user.password = "12345678"
assert user.valid? # this assertion is failing
This test is meant to test one thing -- that a password must be at least 8 characters. Testing "user.valid?" tests much more than that. That could be false for any number of reasons -- perhaps you have a validation that rejects "example.com" as a valid email address.
I would test that at least one error exists on the :password attribute for the user and if you want, take it a step farther and test that the type of error is :too_short (see 5.1.2 of Rails Internationalization (I18n) API — Ruby on Rails Guides for what i'm talking about)
I would also break this up into individual tests...
test "user is invalid when password is too short"
test "user is invalid when password is too long"
test "user is valid when password is just right"
user.password = "123456789"
assert user.valid?
end
test "Password must be shorter" do
user = User.new(:email => "eve@example.com",
:name => "Eve",
:security_question => "What's my name?",
:security_answer => "Eve")
user.password = "12345678901234567890...01234567890123456789" #
this is 1999 chars long, I've just cut it for this email.
You might try this for readability.
user.password = "x" * 1999 # results in a string of x's of length 1999.
Good luck!