devise_invitable 2.0.9 → 2.0.10

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
data/README.rdoc DELETED
@@ -1,496 +0,0 @@
1
- = DeviseInvitable
2
- {<img src="https://badge.fury.io/rb/devise_invitable.svg"/>}[http://badge.fury.io/rb/devise_invitable]
3
- {<img src="https://github.com/scambra/devise_invitable/actions/workflows/ci.yml/badge.svg"/>}[https://github.com/scambra/devise_invitable/actions/workflows/ci.yml]
4
- {<img src="https://codeclimate.com/github/scambra/devise_invitable/badges/gpa.svg"/>}[https://codeclimate.com/github/scambra/devise_invitable]
5
-
6
- It adds support to Devise[https://github.com/plataformatec/devise] for sending invitations by email (it requires to be authenticated) and accept the invitation setting the password.
7
-
8
- == Requirements
9
-
10
- The latest version of DeviseInvitable works with Devise >= 4.6.
11
-
12
- If you want to use devise_invitable with earlier Devise releases (4.0 <= x < 4.6), use version 1.7.5.
13
-
14
- == Installation
15
-
16
- Install DeviseInvitable gem:
17
-
18
- gem install devise_invitable
19
-
20
- Add DeviseInvitable to your Gemfile:
21
-
22
- gem 'devise_invitable', '~> 2.0.0'
23
-
24
- === Automatic installation
25
-
26
- Run the following generator to add DeviseInvitable’s configuration option in the Devise configuration file (<tt>config/initializers/devise.rb</tt>):
27
-
28
- rails generate devise_invitable:install
29
-
30
- When you are done, you are ready to add DeviseInvitable to any of your Devise models using the following generator:
31
-
32
- rails generate devise_invitable MODEL
33
-
34
- Replace MODEL by the class name you want to add DeviseInvitable, like <tt>User</tt>, <tt>Admin</tt>, etc. This will add the <tt>:invitable</tt> flag to your model's Devise modules. The generator will also create a migration file (if your ORM supports them).
35
-
36
- === Manual installation
37
-
38
- Follow the walkthrough for Devise and after it's done, follow this walkthrough.
39
-
40
- ==== Devise Configuration
41
- Add <tt>:invitable</tt> to the <tt>devise</tt> call in your model (we’re assuming here you already have a User model with some Devise modules):
42
-
43
- class User < ActiveRecord::Base
44
- devise :database_authenticatable, :confirmable, :invitable
45
- end
46
-
47
- ==== ActiveRecord Migration
48
- Add <tt>t.invitable</tt> to your Devise model migration:
49
-
50
- create_table :users do
51
- ...
52
- ## Invitable
53
- t.string :invitation_token
54
- t.datetime :invitation_created_at
55
- t.datetime :invitation_sent_at
56
- t.datetime :invitation_accepted_at
57
- t.integer :invitation_limit
58
- t.integer :invited_by_id
59
- t.string :invited_by_type
60
- ...
61
- end
62
- add_index :users, :invitation_token, unique: true
63
-
64
- or for a model that already exists, define a migration to add DeviseInvitable to your model:
65
-
66
- def change
67
- add_column :users, :invitation_token, :string
68
- add_column :users, :invitation_created_at, :datetime
69
- add_column :users, :invitation_sent_at, :datetime
70
- add_column :users, :invitation_accepted_at, :datetime
71
- add_column :users, :invitation_limit, :integer
72
- add_column :users, :invited_by_id, :integer
73
- add_column :users, :invited_by_type, :string
74
- add_index :users, :invitation_token, unique: true
75
- end
76
-
77
- If you previously used devise_invitable with a <tt>:limit</tt> on <tt>:invitation_token</tt>, remove it:
78
-
79
- def up
80
- change_column :users, :invitation_token, :string, limit: nil
81
- end
82
-
83
- def down
84
- change_column :users, :invitation_token, :string, limit: 60
85
- end
86
-
87
- == Mongoid Field Definitions
88
- If you are using Mongoid, define the following fields and indexes within your invitable model:
89
-
90
- field :invitation_token, type: String
91
- field :invitation_created_at, type: Time
92
- field :invitation_sent_at, type: Time
93
- field :invitation_accepted_at, type: Time
94
- field :invitation_limit, type: Integer
95
-
96
- index( { invitation_token: 1 }, { background: true} )
97
- index( { invitation_by_id: 1 }, { background: true} )
98
-
99
- You do not need to define a <tt>belongs_to</tt> relationship, as DeviseInvitable does this on your behalf:
100
- belongs_to :invited_by, polymorphic: true
101
-
102
- Remember to create indexes within the MongoDB database after deploying your changes.
103
- rake db:mongoid:create_indexes
104
-
105
- == Model configuration
106
-
107
- DeviseInvitable adds some new configuration options:
108
-
109
- * <tt>invite_for</tt>: The period the generated invitation token is valid. After this period, the invited resource won't be able to accept the invitation. When <tt>invite_for</tt> is <tt>0</tt> (the default), the invitation won't expire.
110
-
111
- You can set this configuration option in the Devise initializer as follow:
112
-
113
- # ==> Configuration for :invitable
114
- # The period the generated invitation token is valid.
115
- # After this period, the invited resource won't be able to accept the invitation.
116
- # When invite_for is 0 (the default), the invitation won't expire.
117
- # config.invite_for = 2.weeks
118
-
119
- or directly as parameters to the <tt>devise</tt> method:
120
-
121
- devise :database_authenticatable, :confirmable, :invitable, invite_for: 2.weeks
122
-
123
- * <tt>invitation_limit</tt>: The number of invitations users can send. The default value of <tt>nil</tt> means users can send as many invites as they want, there is no limit for any user, <tt>invitation_limit</tt> column is not used. A setting of <tt>0</tt> means they can't send invitations. A setting <tt>n > 0</tt> means they can send <tt>n</tt> invitations. You can change <tt>invitation_limit</tt> column for some users so they can send more or less invitations, even with global <tt>invitation_limit = 0</tt>.
124
-
125
- * <tt>invite_key</tt>: The key to be used to check existing users when sending an invitation. You can use multiple keys. This value must be a hash with the invite key as hash keys, and values that respond to the <tt>===</tt> operator (including procs and regexes). The default value is looking for users by email and validating with <tt>Devise.email_regexp</tt>.
126
-
127
- * <tt>validate_on_invite</tt>: force a record to be valid before being actually invited.
128
-
129
- * <tt>resend_invitation</tt>: resend invitation if user with invited status is invited again. Enabled by default.
130
-
131
- * <tt>invited_by_class_name</tt>: the class name of the inviting model. If this is <tt>nil</tt>, polymorphic association is used.
132
-
133
- * <tt>invited_by_foreign_key</tt>: the foreign key to the inviting model (only used if <tt>invited_by_class_name</tt> is set, otherwise <tt>:invited_by_id</tt>)
134
-
135
- * <tt>invited_by_counter_cache</tt>: the column name used for counter_cache column. If this is <tt>nil</tt> (default value), the <tt>invited_by</tt> association is declared without <tt>counter_cache</tt>.
136
-
137
- * <tt>allow_insecure_sign_in_after_accept</tt>: automatically sign in the user after they set a password. Enabled by default.
138
-
139
- * <tt>require_password_on_accepting</tt>: require password when user accepts the invitation. Enabled by default. Disable if you don't want to ask or enforce to set password while accepting, because is set when user is invited or it will be set later.
140
-
141
- For more details, see <tt>config/initializers/devise.rb</tt> (after you invoked the <tt>devise_invitable:install</tt> generator described above).
142
-
143
- == Configuring views
144
-
145
- All the views are packaged inside the gem. If you'd like to customize the views, invoke the following generator and it will copy all the views to your application:
146
-
147
- rails generate devise_invitable:views
148
-
149
- You can also use the generator to generate scoped views:
150
-
151
- rails generate devise_invitable:views users
152
-
153
- Then turn scoped views on in <tt>config/initializers/devise.rb</tt>:
154
-
155
- config.scoped_views = true
156
-
157
- Please refer to {Devise's README}[https://github.com/plataformatec/devise] for more information about views.
158
-
159
- == Configuring controllers
160
-
161
- To change the controller's behavior, create a controller that inherits from <tt>Devise::InvitationsController</tt>. The available methods are: <tt>new</tt>, <tt>create</tt>, <tt>edit</tt>, and <tt>update</tt>. Refer to the {original controllers source}[https://github.com/scambra/devise_invitable/blob/master/app/controllers/devise/invitations_controller.rb] before editing any of these actions. Your controller might now look something like this:
162
-
163
- class Users::InvitationsController < Devise::InvitationsController
164
- def update
165
- if some_condition
166
- redirect_to root_path
167
- else
168
- super
169
- end
170
- end
171
- end
172
-
173
- Now just tell Devise that you want to use your controller, the controller above is <tt>'users/invitations'</tt>, so our routes.rb would have this line:
174
-
175
- devise_for :users, controllers: { invitations: 'users/invitations' }
176
-
177
- be sure that you generate the views and put them into the controller that you generated, so for this example it would be:
178
-
179
- rails generate devise_invitable:views users
180
-
181
- To change behaviour of inviting or accepting users, you can simply override two methods:
182
-
183
- class Users::InvitationsController < Devise::InvitationsController
184
- private
185
-
186
- # This is called when creating invitation.
187
- # It should return an instance of resource class.
188
- def invite_resource
189
- # skip sending emails on invite
190
- super { |user| user.skip_invitation = true }
191
- end
192
-
193
- # This is called when accepting invitation.
194
- # It should return an instance of resource class.
195
- def accept_resource
196
- resource = resource_class.accept_invitation!(update_resource_params)
197
- # Report accepting invitation to analytics
198
- Analytics.report('invite.accept', resource.id)
199
- resource
200
- end
201
- end
202
-
203
- == Strong Parameters
204
-
205
- When you customize your own views, you may end up adding new attributes to forms. Rails 4 moved the parameter sanitization from the model to the controller, causing DeviseInvitable to handle this concern at the controller as well. Read about it in {Devise README}[https://github.com/plataformatec/devise#strong-parameters]
206
-
207
- There are just two actions in DeviseInvitable that allows any set of parameters to be passed down to the model, therefore requiring sanitization. Their names and the permited parameters by default are:
208
-
209
- * <tt>invite</tt> (Devise::InvitationsController#create) - Permits only the authentication keys (like <tt>email</tt>)
210
- * <tt>accept_invitation</tt> (Devise::InvitationsController#update) - Permits <tt>invitation_token</tt> plus <tt>password</tt> and <tt>password_confirmation</tt>.
211
-
212
- Here is an example of what your application controller might need to include in order to add these parameters to the invitation view:
213
-
214
- before_action :configure_permitted_parameters, if: :devise_controller?
215
-
216
- protected
217
-
218
- def configure_permitted_parameters
219
- devise_parameter_sanitizer.permit(:accept_invitation, keys: [:first_name, :last_name, :phone])
220
- end
221
-
222
- Here is an example setting a User's first name, last name, and role for a custom invitation:
223
-
224
- #Configuring the InvitationsController to accept :first_name, :last_name, and :role
225
-
226
- class Users::InvitationsController < Devise::InvitationsController
227
- before_action :configure_permitted_parameters
228
-
229
- protected
230
-
231
- # Permit the new params here.
232
- def configure_permitted_parameters
233
- devise_parameter_sanitizer.permit(:invite, keys: [:first_name, :last_name, :role])
234
- end
235
- end
236
-
237
- #Define your roles in the User model
238
-
239
- class User < ApplicationRecord
240
- has_many :models
241
-
242
- enum role: {Role 1 Name: 0, Role 2 Name: 1, Role 3 Name: 2, etc...}
243
- end
244
-
245
- #In the Invitation view
246
-
247
- <h2><%= t "devise.invitations.new.header" %></h2>
248
-
249
- <%= form_for(resource, as: resource_name, url: invitation_path(resource_name), html: { method: :post }) do |f| %>
250
- <%= render "devise/shared/error_messages", resource: resource %>
251
- <% resource.class.invite_key_fields.each do |field| -%>
252
- <div class="field">
253
- <%= f.label field %><br />
254
- <%= f.text_field field %>
255
- </div>
256
- <% end %>
257
-
258
- <div class="field">
259
- <%= f.label :first_name %>
260
- <%= f.text_field :first_name %>
261
- </div>
262
-
263
- <div class="field">
264
- <%= f.label :last_name %>
265
- <%= f.text_field :last_name %>
266
- </div>
267
-
268
- <div class="field">
269
- <%= f.label :role %>
270
- <%= f.select :role, options_for_select(User.roles.map { |key, value| [key.humanize, key] }), {prompt: "Select Role"} %>
271
- </div>
272
-
273
- <div class="actions">
274
- <%= f.submit t("devise.invitations.new.submit_button") %>
275
- </div>
276
- <% end %>
277
-
278
- == Usage
279
-
280
- === Send an invitation
281
-
282
- To send an invitation to a user, use the <tt>invite!</tt> class method. <b>Note: This will create a user, and send an email for the invite.</b> <tt>:email</tt> must be present in the parameters hash. You can also include other attributes in the hash. The record will not be validated.
283
-
284
- User.invite!(email: 'new_user@example.com', name: 'John Doe')
285
- # => an invitation email will be sent to new_user@example.com
286
-
287
- If you want to create the invitation but not send it, you can set <tt>skip_invitation</tt> to <tt>true</tt>.
288
-
289
- user = User.invite!(email: 'new_user@example.com', name: 'John Doe') do |u|
290
- u.skip_invitation = true
291
- end
292
- # => the record will be created, but the invitation email will not be sent
293
-
294
- When generating the <tt>accept_user_invitation_url</tt> yourself, you must use the <tt>raw_invitation_token</tt>.
295
- This value is temporarily available when you invite a user and will be decrypted when received.
296
-
297
- accept_user_invitation_url(invitation_token: user.raw_invitation_token)
298
-
299
- When <tt>skip_invitation</tt> is used, you must also then set the <tt>invitation_sent_at</tt> field when the user is sent their token. Failure to do so will yield "Invalid invitation token" error when the user attempts to accept the invite.
300
- You can set the column, or call <tt>deliver_invitation</tt> to send the invitation and set the column:
301
-
302
- user.deliver_invitation
303
-
304
- You can add <tt>:skip_invitation</tt> to attributes hash if <tt>skip_invitation</tt> is added to <tt>attr_accessible</tt>.
305
-
306
- User.invite!(email: 'new_user@example.com', name: 'John Doe', skip_invitation: true)
307
- # => the record will be created, but the invitation email will not be sent
308
-
309
- <tt>skip_invitation</tt> skips sending the email, but sets <tt>invitation_token</tt>, so <tt>invited_to_sign_up?</tt> on the
310
- resulting user returns <tt>true</tt>.
311
-
312
- To check if a particular user is created by invitation, irrespective to state of invitation one can use <tt>created_by_invite?</tt>
313
-
314
- **Warning**
315
-
316
- When using <tt>skip_invitation</tt> you must send the email with the user object instance that generated the tokens, as <tt>user.raw_invitation_token</tt> is available only to the instance and is not persisted in the database.
317
-
318
- You can also set <tt>invited_by</tt> when using the <tt>invite!</tt> class method:
319
-
320
- User.invite!({ email: 'new_user@example.com' }, current_user) # current_user will be set as invited_by
321
-
322
- === Sending an invitation after user creation
323
-
324
- You can send an invitation to an existing user if your workflow creates them separately:
325
-
326
- user = User.find(42)
327
- user.invite!(current_user) # current user is optional to set the invited_by attribute
328
-
329
- === Find by invitation token
330
-
331
- To find by invitation token use the <tt>find_by_invitation_token</tt> class method.
332
-
333
- user = User.find_by_invitation_token(params[:invitation_token], true)
334
-
335
- === Accept an invitation
336
-
337
- To accept an invitation with a token use the <tt>accept_invitation!</tt> class method. <tt>:invitation_token</tt> must be present in the parameters hash. You can also include other attributes in the hash.
338
-
339
- User.accept_invitation!(invitation_token: params[:invitation_token], password: 'ad97nwj3o2', name: 'John Doe')
340
-
341
- === Callbacks
342
-
343
- A callback event is fired before and after an invitation is created (User#invite!) or accepted (User#accept_invitation!). For example, in your resource model you can add:
344
- # Note: callbacks should be placed after devise: :invitable is specified.
345
- before_invitation_created :email_admins
346
- after_invitation_accepted :email_invited_by
347
-
348
- def email_admins
349
- # ...
350
- end
351
-
352
- def email_invited_by
353
- # ...
354
- end
355
-
356
- The callbacks support all options and arguments available to the standard callbacks provided by ActiveRecord.
357
-
358
- === Scopes
359
-
360
- A pair of scopes to find those users that have accepted, and those that have not accepted, invitations are defined:
361
-
362
- User.invitation_accepted # => returns all Users for whom the invitation_accepted_at attribute is not nil
363
- User.invitation_not_accepted # => returns all Users for whom the invitation_accepted_at attribute is nil
364
- User.created_by_invite # => returns all Users who are created by invitations, irrespective to invitation status
365
-
366
- == Integration in a Rails application
367
-
368
- Since the invitations controller takes care of all the creation/acceptation of an invitation, in most cases you wouldn't call the <tt>invite!</tt> and <tt>accept_invitation!</tt> methods directly.
369
- Instead, in your views, put a link to <tt>new_user_invitation_path</tt> or <tt>new_invitation_path(:user)</tt> or even <tt>/users/invitation/new</tt> to prepare and send an invitation (to a user in this example).
370
-
371
- After an invitation is created and sent, the inviter will be redirected to <tt>after_invite_path_for(inviter, invitee)</tt>, which is the same path as <tt>signed_in_root_path</tt> by default.
372
-
373
- After an invitation is accepted, the invitee will be redirected to <tt>after_accept_path_for(resource)</tt>, which is the same path as <tt>signed_in_root_path</tt> by default. If you want to override the path, override invitations controller and define <tt>after_accept_path_for</tt> method. This is useful in the common case that a user is invited to a specific location in your application. More on {Devise's README}[https://github.com/plataformatec/devise], "Controller filters and helpers" section.
374
-
375
- The invitation email includes a link to accept the invitation that looks like this: <tt>/users/invitation/accept?invitation_token=abcd123</tt>. When clicked, the invited must set a password in order to accept its invitation. Note that if the <tt>invitation_token</tt> is not present or not valid, the invited is redirected to <tt>invalid_token_path_for(resource_name)</tt>, which by default is <tt>after_sign_out_path_for(resource_name)</tt>.
376
-
377
- The controller sets the <tt>invited_by_id</tt> attribute for the new user to the current user. This will let you easily keep track of who invited whom.
378
-
379
- == Controller filter
380
-
381
- InvitationsController uses <tt>authenticate_inviter!</tt> filter to restrict who can send invitations. You can override this method in your <tt>ApplicationController</tt>.
382
-
383
- Default behavior requires authentication of the same resource as the invited one. For example, if your model <tt>User</tt> is invitable, it will allow all authenticated users to send invitations to other users.
384
-
385
- You would have a <tt>User</tt> model which is configured as invitable and an <tt>Admin</tt> model which is not. If you want to allow only admins to send invitations, simply overwrite the <tt>authenticate_inviter!</tt> method as follow:
386
-
387
- class ApplicationController < ActionController::Base
388
- protected
389
-
390
- def authenticate_inviter!
391
- authenticate_admin!(force: true)
392
- end
393
- end
394
-
395
- And include <tt>DeviseInvitable::Inviter</tt> module into <tt>Admin</tt> model:
396
-
397
- class Admin < ActiveRecord::Base
398
- devise :database_authenticatable, :validatable
399
- include DeviseInvitable::Inviter
400
- end
401
-
402
- == Has many invitations
403
-
404
- If you want to get all records invited by a resource, you should define <tt>has_many</tt> association in the model allowed to send invitations.
405
-
406
- For the default behavior, define it like this:
407
-
408
- has_many :invitations, class_name: self.to_s, as: :invited_by
409
-
410
- For the previous example, where admins send invitations to users, define it like this:
411
-
412
- has_many :invitations, class_name: 'User', as: :invited_by
413
-
414
- == I18n
415
-
416
- DeviseInvitable uses flash messages with I18n with the flash keys <tt>:send_instructions</tt>, <tt>:invitation_token_invalid</tt> and <tt>:updated</tt>. To customize your app, you can modify the generated locale file:
417
-
418
- en:
419
- devise:
420
- invitations:
421
- send_instructions: 'An invitation email has been sent to %{email}.'
422
- invitation_token_invalid: 'The invitation token provided is not valid!'
423
- updated: 'Your password was set successfully. You are now signed in.'
424
- updated_not_active: 'Your password was set successfully.'
425
-
426
- You can also create distinct messages based on the resource you've configured using the singular name given in routes:
427
-
428
- en:
429
- devise:
430
- invitations:
431
- user:
432
- send_instructions: 'A new user invitation has been sent to %{email}.'
433
- invitation_token_invalid: 'Your invitation token is not valid!'
434
- updated: 'Welcome on board! You are now signed in.'
435
- updated_not_active: 'Welcome on board! Sign in to continue.'
436
-
437
- The DeviseInvitable mailer uses the same pattern as Devise to create mail subject messages:
438
-
439
- en:
440
- devise:
441
- mailer:
442
- invitation_instructions:
443
- subject: 'You got an invitation!'
444
- user_subject: 'You got a user invitation!'
445
-
446
- Take a look at the {generated locale file}[https://github.com/scambra/devise_invitable/blob/master/config/locales/en.yml] to check all available messages.
447
-
448
- Check out wiki[https://github.com/scambra/devise_invitable/wiki/I18n] for translations.
449
-
450
- === Use with sub schema
451
- If you are using sub schema in you application, you need to make sure that you are prioritizing your sub schema scheme over Warden in Rack.
452
- For instance, if you are using the Apartment gem go inside your <tt>config/application.rb</tt> file, add the following lines:
453
-
454
- module YourSite
455
- class Application < Rails::Application
456
- ...
457
- Rails.application.config.middleware.insert_before Warden::Manager, Apartment::Elevators::Subdomain
458
- end
459
- end
460
-
461
- == Other ORMs
462
-
463
- DeviseInvitable supports ActiveRecord and Mongoid, like Devise.
464
-
465
- == Wiki
466
-
467
- It's possible to find additional information about DeviseInvitable on the Wiki:
468
-
469
- https://github.com/scambra/devise_invitable/wiki
470
-
471
- == Testing
472
-
473
- To run tests:
474
-
475
- bundle install
476
- bundle exec rake test
477
-
478
- == Contributors
479
-
480
- Check them all at:
481
-
482
- https://github.com/scambra/devise_invitable/contributors
483
-
484
- Special thanks to rymai[https://github.com/rymai] for the Rails 3 support, his fork was a great help.
485
-
486
- == Note on Patches/Pull Requests
487
-
488
- * Fork the project.
489
- * Make your feature addition or bug fix.
490
- * Add tests for it. This is important so I don't break it in a future version unintentionally.
491
- * Commit, do not mess with rakefile, version, or history. (if you want to have your own version, that is fine but bump version in a commit by itself I can ignore when I pull)
492
- * Send me a pull request. Bonus points for topic branches.
493
-
494
- == Copyright
495
-
496
- Copyright (c) 2019 Sergio Cambra. See LICENSE for details.