Module: Authlogic::ORMAdapters::ActiveRecordAdapter::ActsAsAuthentic::Credentials

Defined in:
lib/authlogic/orm_adapters/active_record_adapter/acts_as_authentic/credentials.rb

Overview

Credentials

Handles any credential specific code, such as validating the login, encrpyting the password, etc.

Class Methods

  • friendly_unique_token - returns a random string of 20 alphanumeric characters. Used when resetting the password. This is a more user friendly token then a long Sha512 hash.

Instance Methods

  • {options[:password_field]}=(value) - encrypts a raw password and sets it to your crypted_password_field. Also sets the password_salt to a random token.

  • valid_{options[:password_field]}?(password_to_check) - checks is the password is valid. The password passed must be the raw password, not encrypted.

  • reset_{options[:password_field]} - resets the password using the friendly_unique_token class method

  • reset_{options[:password_field]}! - calls reset_password and then saves the record

Instance Method Summary collapse

Instance Method Details

#acts_as_authentic_with_credentials(options = {}) ⇒ Object



20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
# File 'lib/authlogic/orm_adapters/active_record_adapter/acts_as_authentic/credentials.rb', line 20

def acts_as_authentic_with_credentials(options = {})
  acts_as_authentic_without_credentials(options)
  
  if options[:validate_fields]
    email_name_regex  = '[\w\.%\+\-]+'
    domain_head_regex = '(?:[A-Z0-9\-]+\.)+'
    domain_tld_regex  = '(?:[A-Z]{2}|com|org|net|edu|gov|mil|biz|info|mobi|name|aero|jobs|museum)'
    email_field_regex ||= /\A#{email_name_regex}@#{domain_head_regex}#{domain_tld_regex}\z/i
    
    if options[:validate_login_field]
      case options[:login_field_type]
      when :email
        validates_length_of options[:login_field], {:within => 6..100}.merge(options[:login_field_validates_length_of_options])
        validates_format_of options[:login_field], {:with => email_field_regex, :message => "should look like an email address."}.merge(options[:login_field_validates_length_of_options])
      else
        validates_length_of options[:login_field], {:within => 2..100}.merge(options[:login_field_validates_length_of_options])
        validates_format_of options[:login_field], {:with => /\A\w[\w\.\-_@ ]+\z/, :message => "should use only letters, numbers, spaces, and .-_@ please."}.merge(options[:login_field_validates_format_of_options])
      end
      
      validates_uniqueness_of options[:login_field], {:allow_blank => true}.merge(options[:login_field_validates_uniqueness_of_options].merge(:if => "#{options[:login_field]}_changed?".to_sym))
    end
    
    if options[:validate_password_field]
      validates_presence_of options[:password_field], {:on => :create}.merge(options[:password_field_validates_presence_of_options])
      validates_confirmation_of options[:password_field], options[:password_field_validates_confirmation_of_options].merge(:if => "#{options[:crypted_password_field]}_changed?".to_sym)
      validates_presence_of "#{options[:password_field]}_confirmation", :if => "#{options[:crypted_password_field]}_changed?"
    end
    
    if options[:validate_email_field] && options[:email_field]
      validates_length_of options[:email_field], {:within => 6..100}.merge(options[:email_field_validates_length_of_options])
      validates_format_of options[:email_field], {:with => email_field_regex, :message => "should look like an email address."}.merge(options[:email_field_validates_format_of_options])
      validates_uniqueness_of options[:email_field], options[:email_field_validates_uniqueness_of_options].merge(:if => "#{options[:email_field]}_changed?".to_sym)
    end
  end
  
  attr_reader options[:password_field]
  
  class_eval <<-"end_eval", __FILE__, __LINE__
    def self.friendly_unique_token
      chars = ("a".."z").to_a + ("A".."Z").to_a + ("0".."9").to_a
      newpass = ""
      1.upto(20) { |i| newpass << chars[rand(chars.size-1)] }
      newpass
    end
    
    def #{options[:password_field]}=(pass)
      return if pass.blank?
      @#{options[:password_field]} = pass
      self.#{options[:password_salt_field]} = self.class.unique_token
      self.#{options[:crypted_password_field]} = #{options[:crypto_provider]}.encrypt(obfuscate_password(@#{options[:password_field]}))
    end
    
    def valid_#{options[:password_field]}?(attempted_password)
      return false if attempted_password.blank? || #{options[:crypted_password_field]}.blank? || #{options[:password_salt_field]}.blank?
      (#{options[:crypto_provider]}.respond_to?(:decrypt) && #{options[:crypto_provider]}.decrypt(#{options[:crypted_password_field]}) == attempted_password + #{options[:password_salt_field]}) ||
        (!#{options[:crypto_provider]}.respond_to?(:decrypt) && #{options[:crypto_provider]}.encrypt(obfuscate_password(attempted_password)) == #{options[:crypted_password_field]})
    end
    
    def reset_#{options[:password_field]}
      friendly_token = self.class.friendly_unique_token
      self.#{options[:password_field]} = friendly_token
      self.#{options[:password_field]}_confirmation = friendly_token
    end
    alias_method :randomize_password, :reset_password
    
    def reset_#{options[:password_field]}!
      reset_#{options[:password_field]}
      save_without_session_maintenance(false)
    end
    alias_method :randomize_password!, :reset_password!
    
    private
      def obfuscate_password(raw_password)
        if #{options[:act_like_restful_authentication].inspect}
          [REST_AUTH_SITE_KEY, raw_password, #{options[:password_salt_field]}, REST_AUTH_SITE_KEY].join("--")
        else
          raw_password + #{options[:password_salt_field]}
        end
      end
  end_eval
end