Created
June 13, 2014 03:31
-
-
Save ShepBook/dcd4e3c312acf1f6e096 to your computer and use it in GitHub Desktop.
Update form_for without password
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Originally from http://stackoverflow.com/questions/7083575/updating-user-attributes-without-requiring-password/8790881#8790881 | |
The correct answer no-longer works for rails 4. I believe my answer is the cleanest and the most versatile that will work whenever you want to leave out any attributes (not just the password). This approach will be needed if you want to update the separate attributes of any model in a number of different places. | |
For example, if you want to do what Stack Overflow does and have the passwords updatable via a security page, the profile image updatable via the user show view and the bulk of a user's information updatable via a user edit view. | |
1) Extend the hash class with a class method to delete blank values. We will use this method to remove blank values that are not being updated but are still present in the params hash: | |
1a) Create a hash.rb file in your lib directory, under an ext directory: | |
command line | |
$ mkdir lib/ext | |
$ touch lib/ext/hash.rb | |
1b) Inside hash.rb, 'create' a Hash class and create a .delete_blanks! method: | |
lib/ext/hash.rb | |
class Hash | |
def delete_blanks! | |
delete_if { |k, v| v.nil? } | |
end | |
end | |
1c) Require this file (and your entire lib directory) into the rails referencing it in an initializer: | |
config/boot.rb | |
# other things such as gemfiles are required here, left out for brevity | |
Dir['lib/**/*.rb'].each { |f| load(f) } # requires all .rb files in the lib directory | |
2) Inside the users#update action, implement our shiny new delete_blanks! class method to remove the attributes we're not updating from the params hash. Then, update the user instance via the update_attributes method, *not the update method! | |
2a) Firstly, let's use the delete_blanks! method to fix our user_params hash: | |
app/controllers/users_controller.rb | |
new_params = user_params.delete_blanks! | |
2b) And now let's update the instance using the update_attributes method, (again, not the update method): | |
app/controllers/users_controller.rb | |
@user.update_attributes(new_params) | |
Here's how the finished users#update action should look: | |
app/controllers/users_controller.rb | |
def update | |
new_params = user_params.delete_blanks! | |
if @user.update_attributes(new_params) | |
redirect_to @user, notice: 'User was successfully updated.' | |
else | |
render action: 'edit' // or whatever you want to do | |
end | |
end | |
3) In the User model, add the if: :<attribute> option to all of your validations. This is to make sure the validation is only triggered if the attribute is present in the params hash. Our delete_blanks! method will have removed the attribute from the params hash, so the validation for password, for example, won't be run. It's also worth noting that delete_blanks! only removes hash entries with a value of nil, not those with empty strings. So if someone leaves out the password on the user create form (or any form with a field for the password), a presence validation will take effect because the :password entry of the hash won't be nil, it'll be an empty string: | |
3a) Use the if: option on all validations: | |
app/models/user.rb | |
VALID_EMAIL_REGEX = /[a-zA-Z0-9_.+-]+@[a-zA-Z0-9-]+\.[a-zA-Z0-9\-.]/ | |
validates :first_name, presence: true, if: :first_name | |
validates :last_name, presence: true, if: :last_name | |
validates :user_name, presence: true, if: :user_name | |
validates :email, presence: true, | |
uniqueness: { case_sensitive: false }, | |
format: { with: VALID_EMAIL_REGEX }, if: :email | |
validates :password, length: { minimum: 6, maximum: 10 }, if: :password | |
And that's it. Now the user model can be updated over many, many different forms all over your app. Presence validations for an attribute still come into play on any form that contains a field for it, e.g. the password presence validation still would come into play in the user#create view. | |
This may seem more verbose than other answers, but I believe this is the most robust way. You can update in isolation an infinite number of attributes for User instances, on an infinite amount of different models. Just remember when you want to do this with a new model you need to repeat the steps 2a), 2b) and 3a) |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment