devise_invitable 0.3.4 → 0.4.0
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 +63 -42
- data/app/controllers/devise/invitations_controller.rb +23 -6
- data/config/locales/en.yml +5 -3
- data/lib/devise_invitable/controllers/helpers.rb +1 -6
- data/lib/devise_invitable/mailer.rb +8 -2
- data/lib/devise_invitable/model.rb +85 -38
- data/lib/devise_invitable/rails.rb +1 -1
- data/lib/devise_invitable/routes.rb +8 -5
- data/lib/devise_invitable/schema.rb +27 -2
- data/lib/devise_invitable/version.rb +3 -0
- data/lib/devise_invitable.rb +37 -8
- data/lib/generators/active_record/templates/migration.rb +14 -7
- data/lib/generators/devise_invitable/devise_invitable_generator.rb +4 -0
- data/lib/generators/devise_invitable/install_generator.rb +17 -2
- data/lib/generators/mongoid/devise_invitable_generator.rb +8 -0
- metadata +49 -142
- data/.document +0 -5
- data/.gitignore +0 -22
- data/Gemfile +0 -3
- data/Gemfile.lock +0 -108
- data/Rakefile +0 -55
- data/VERSION +0 -1
- data/devise_invitable.gemspec +0 -146
- data/test/generators_test.rb +0 -45
- data/test/integration/invitable_test.rb +0 -109
- data/test/integration_tests_helper.rb +0 -58
- data/test/mailers/invitation_test.rb +0 -62
- data/test/model_tests_helper.rb +0 -41
- data/test/models/invitable_test.rb +0 -188
- data/test/models_test.rb +0 -31
- data/test/rails_app/app/controllers/admins_controller.rb +0 -6
- data/test/rails_app/app/controllers/application_controller.rb +0 -3
- data/test/rails_app/app/controllers/home_controller.rb +0 -4
- data/test/rails_app/app/controllers/users_controller.rb +0 -12
- data/test/rails_app/app/helpers/application_helper.rb +0 -2
- data/test/rails_app/app/models/octopussy.rb +0 -11
- data/test/rails_app/app/models/user.rb +0 -4
- data/test/rails_app/app/views/home/index.html.erb +0 -0
- data/test/rails_app/app/views/layouts/application.html.erb +0 -15
- data/test/rails_app/app/views/users/invitations/new.html.erb +0 -15
- data/test/rails_app/config/application.rb +0 -46
- data/test/rails_app/config/boot.rb +0 -13
- data/test/rails_app/config/database.yml +0 -22
- data/test/rails_app/config/environment.rb +0 -5
- data/test/rails_app/config/environments/development.rb +0 -26
- data/test/rails_app/config/environments/production.rb +0 -49
- data/test/rails_app/config/environments/test.rb +0 -35
- data/test/rails_app/config/initializers/backtrace_silencers.rb +0 -7
- data/test/rails_app/config/initializers/devise.rb +0 -144
- data/test/rails_app/config/initializers/inflections.rb +0 -10
- data/test/rails_app/config/initializers/mime_types.rb +0 -5
- data/test/rails_app/config/initializers/secret_token.rb +0 -7
- data/test/rails_app/config/initializers/session_store.rb +0 -8
- data/test/rails_app/config/locales/en.yml +0 -5
- data/test/rails_app/config/routes.rb +0 -4
- data/test/rails_app/config.ru +0 -4
- data/test/rails_app/script/rails +0 -6
- data/test/routes_test.rb +0 -20
- data/test/test_helper.rb +0 -30
- /data/app/views/devise/mailer/{invitation.html.erb → invitation_instructions.html.erb} +0 -0
data/README.rdoc
CHANGED
|
@@ -2,26 +2,26 @@
|
|
|
2
2
|
|
|
3
3
|
It adds support to devise[http://github.com/plataformatec/devise] for send invitations by email (it requires to be authenticated) and accept the invitation setting the password.
|
|
4
4
|
|
|
5
|
-
DeviseInvitable currently only support
|
|
5
|
+
DeviseInvitable currently only support Rails 3, if you want to use it with Rails 2.3 you must install version {0.2.3}[http://rubygems.org/gems/devise_invitable/versions/0.2.3]
|
|
6
6
|
|
|
7
|
-
== Installation for Rails ~> 3.0
|
|
7
|
+
== Installation for Rails ~> 3.0 and Devise ~> 1.1
|
|
8
8
|
|
|
9
|
-
Install
|
|
9
|
+
Install DeviseInvitable gem, it will also install dependencies (such as devise and warden):
|
|
10
10
|
|
|
11
|
-
|
|
11
|
+
gem install devise_invitable
|
|
12
12
|
|
|
13
|
-
|
|
13
|
+
Add DeviseInvitable to your Gemfile (and Devise if you weren't using them):
|
|
14
14
|
|
|
15
|
-
gem 'devise'
|
|
16
|
-
gem 'devise_invitable'
|
|
15
|
+
gem 'devise', '~> 1.1.3'
|
|
16
|
+
gem 'devise_invitable', '~> 0.3.4'
|
|
17
17
|
|
|
18
18
|
=== Automatic installation
|
|
19
19
|
|
|
20
|
-
|
|
20
|
+
Run the following generator to add DeviseInvitable’s configuration option in the Devise configuration file (config/initializers/devise.rb):
|
|
21
21
|
|
|
22
22
|
rails generate devise_invitable:install
|
|
23
23
|
|
|
24
|
-
|
|
24
|
+
When you are done, you are ready to add DeviseInvitable to any of your Devise models using the following generator:
|
|
25
25
|
|
|
26
26
|
rails generate devise_invitable MODEL
|
|
27
27
|
|
|
@@ -31,7 +31,7 @@ Replace MODEL by the class name you want to add DeviseInvitable, like User, Admi
|
|
|
31
31
|
|
|
32
32
|
Follow the walkthrough for Devise and after it's done, follow this walkthrough.
|
|
33
33
|
|
|
34
|
-
Add :invitable to the
|
|
34
|
+
Add :invitable to the <tt>devise</tt> call in your model (we’re assuming here you already have a User model with some Devise modules):
|
|
35
35
|
|
|
36
36
|
class User < ActiveRecord::Base
|
|
37
37
|
devise :database_authenticatable, :confirmable, :invitable
|
|
@@ -46,39 +46,42 @@ Add t.invitable to your Devise model migration:
|
|
|
46
46
|
end
|
|
47
47
|
add_index :users, :invitation_token
|
|
48
48
|
|
|
49
|
-
or for a model that
|
|
49
|
+
or for a model that already exists, define a migration to add DeviseInvitable to your model:
|
|
50
50
|
|
|
51
51
|
change_table :users do |t|
|
|
52
|
-
t.string :invitation_token, :limit =>
|
|
52
|
+
t.string :invitation_token, :limit => 60
|
|
53
53
|
t.datetime :invitation_sent_at
|
|
54
54
|
t.index :invitation_token
|
|
55
55
|
end
|
|
56
56
|
|
|
57
|
-
# Allow null encrypted_password
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
DeviseInvitable doesn't use _attr_accessible_ or _attr_protected_, so be sure to define attributes as accessible or protected in your model.
|
|
57
|
+
# Allow null encrypted_password
|
|
58
|
+
change_column_null :users, :encrypted_password, true
|
|
59
|
+
# Allow null password_salt (add it if you are using Devise's encryptable module)
|
|
60
|
+
change_column_null :users, :password_salt, true
|
|
62
61
|
|
|
63
62
|
== Model configuration
|
|
64
63
|
|
|
65
|
-
DeviseInvitable adds
|
|
64
|
+
DeviseInvitable adds three new configuration options:
|
|
66
65
|
|
|
67
|
-
|
|
66
|
+
* invite_for: The period the generated invitation token is valid, after this period, the invited resource won't be able to accept the invitation. When invite_for is 0 (the default), the invitation won't expire.
|
|
68
67
|
|
|
69
|
-
You can set
|
|
68
|
+
You can set this configuration option in the Devise initializer as follow:
|
|
70
69
|
|
|
71
70
|
# ==> Configuration for :invitable
|
|
72
|
-
#
|
|
73
|
-
#
|
|
74
|
-
#
|
|
71
|
+
# The period the generated invitation token is valid, after
|
|
72
|
+
# this period, the invited resource won't be able to accept the invitation.
|
|
73
|
+
# When invite_for is 0 (the default), the invitation won't expire.
|
|
75
74
|
# config.invite_for = 2.weeks
|
|
76
75
|
|
|
77
|
-
or directly as parameters to the <tt>devise</tt> method
|
|
76
|
+
or directly as parameters to the <tt>devise</tt> method:
|
|
78
77
|
|
|
79
78
|
devise :database_authenticatable, :confirmable, :invitable, :invite_for => 2.weeks
|
|
80
79
|
|
|
81
|
-
|
|
80
|
+
* invitation_limit: The number of invitations users can send. The default value of nil means users can send as many invites as they want. A setting of 0 means they can't send invitations. A setting n > 0 means they can send n invitations.
|
|
81
|
+
|
|
82
|
+
* invite_key: The key to be used to check existing users when sending an invitation. The key must be an unique field. The default value is looking for users by email.
|
|
83
|
+
|
|
84
|
+
For more details, see <tt>config/initializers/devise.rb</tt> (after you invoked the "devise_invitable:install" generator described above).
|
|
82
85
|
|
|
83
86
|
== Configuring views
|
|
84
87
|
|
|
@@ -96,31 +99,47 @@ Please refer to {Devise's README}[http://github.com/plataformatec/devise] for mo
|
|
|
96
99
|
|
|
97
100
|
=== Send an invitation
|
|
98
101
|
|
|
99
|
-
To send an invitation to a user, use the <tt>invite!</tt> class method.
|
|
100
|
-
You can also include other attributes in the hash. The record will not be validated.
|
|
102
|
+
To send an invitation to a user, use the <tt>invite!</tt> class method. <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.
|
|
101
103
|
|
|
102
|
-
User.invite(:email => "new_user@example.com", :name => "John Doe")
|
|
104
|
+
User.invite!(:email => "new_user@example.com", :name => "John Doe")
|
|
103
105
|
# => an invitation email will be sent to new_user@example.com
|
|
104
106
|
|
|
107
|
+
If you want to create the invitation but not send it, you can set <tt>skip_invitation</tt> to true.
|
|
108
|
+
|
|
109
|
+
User.invite!(:email => "new_user@example.com", :name => "John Doe") do |u|
|
|
110
|
+
u.skip_invitation = true
|
|
111
|
+
end
|
|
112
|
+
# => the record will be created, but the invitation email will not be sent
|
|
113
|
+
|
|
114
|
+
You can add :skip_invitation to attributes hash if skip_invitation is added to attr_accessible.
|
|
115
|
+
|
|
116
|
+
User.invite!(:email => "new_user@example.com", :name => "John Doe", :skip_invitation => true)
|
|
117
|
+
# => the record will be created, but the invitation email will not be sent
|
|
118
|
+
|
|
105
119
|
=== Accept an invitation
|
|
106
120
|
|
|
107
|
-
To accept an invitation with a token use the <tt>accept_invitation
|
|
121
|
+
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.
|
|
108
122
|
|
|
109
|
-
User.accept_invitation(:invitation_token => params[:invitation_token], :password => "ad97nwj3o2", :name => "John Doe")
|
|
123
|
+
User.accept_invitation!(:invitation_token => params[:invitation_token], :password => "ad97nwj3o2", :name => "John Doe")
|
|
110
124
|
|
|
111
125
|
== Integration in a Rails application
|
|
112
126
|
|
|
113
|
-
Since the invitations controller take care of all the
|
|
114
|
-
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.
|
|
115
|
-
|
|
127
|
+
Since the invitations controller take 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.
|
|
128
|
+
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).
|
|
129
|
+
After an invitation is created and sent, the inviter will be redirected to after_sign_in_path_for(resource_name).
|
|
130
|
+
|
|
131
|
+
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 invitation_token is not present or not valid, the invited is redirected to after_sign_out_path_for(resource_name).
|
|
132
|
+
You can also overwrite after_sign_in_path_for and after_sign_out_path_for to customize your redirect hooks. More on {Devise's README}[http://github.com/plataformatec/devise], "Controller filters and helpers" section.
|
|
133
|
+
|
|
134
|
+
The controller sets the invited_by_id attribute for the new user to the current user. This will let you easily keep track of who invited who.
|
|
116
135
|
|
|
117
136
|
== Controller filter
|
|
118
137
|
|
|
119
138
|
InvitationsController uses authenticate_inviter! filter to restrict who can send invitations. You can override this method in your ApplicationController.
|
|
120
139
|
|
|
121
|
-
Default behavior requires authentication of the same resource. For example, if your model User is
|
|
140
|
+
Default behavior requires authentication of the same resource as the invited one. For example, if your model User is invitable, it will allow all authenticated users to send invitations to other users.
|
|
122
141
|
|
|
123
|
-
You would have a User model which is configured as invitable and an Admin model which is not. If you
|
|
142
|
+
You would have a User model which is configured as invitable and an Admin model which is not. If you want to allow only admins to send invitations, simply overwrite the authenticate_inviter! method as follow:
|
|
124
143
|
|
|
125
144
|
class ApplicationController < ActionController::Base
|
|
126
145
|
protected
|
|
@@ -131,12 +150,13 @@ You would have a User model which is configured as invitable and an Admin model
|
|
|
131
150
|
|
|
132
151
|
== I18n
|
|
133
152
|
|
|
134
|
-
DeviseInvitable uses flash messages with I18n with the flash keys <tt>:send_instructions</tt> and <tt>:updated</tt>. To customize your app, you can modify the generated locale file:
|
|
153
|
+
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:
|
|
135
154
|
|
|
136
155
|
en:
|
|
137
156
|
devise:
|
|
138
157
|
invitations:
|
|
139
|
-
send_instructions: 'An email
|
|
158
|
+
send_instructions: 'An invitation email has been sent to %{email}.'
|
|
159
|
+
invitation_token_invalid: 'The invitation token provided is not valid!'
|
|
140
160
|
updated: 'Your password was set successfully. You are now signed in.'
|
|
141
161
|
|
|
142
162
|
You can also create distinct messages based on the resource you've configured using the singular name given in routes:
|
|
@@ -145,17 +165,18 @@ You can also create distinct messages based on the resource you've configured us
|
|
|
145
165
|
devise:
|
|
146
166
|
invitations:
|
|
147
167
|
user:
|
|
148
|
-
send_instructions: 'A new user invitation has been sent.'
|
|
168
|
+
send_instructions: 'A new user invitation has been sent to %{email}.'
|
|
169
|
+
invitation_token_invalid: 'Your invitation token is not valid!'
|
|
149
170
|
updated: 'Welcome on board! You are now signed in.'
|
|
150
171
|
|
|
151
|
-
The DeviseInvitable mailer uses the
|
|
172
|
+
The DeviseInvitable mailer uses the same pattern as Devise to create mail subject messages:
|
|
152
173
|
|
|
153
174
|
en:
|
|
154
175
|
devise:
|
|
155
176
|
mailer:
|
|
156
|
-
|
|
177
|
+
invitation_instructions:
|
|
157
178
|
subject: 'You got an invitation!'
|
|
158
|
-
user_subject: 'You got
|
|
179
|
+
user_subject: 'You got a user invitation!'
|
|
159
180
|
|
|
160
181
|
Take a look at the generated locale file (in <tt>config/locales/devise_invitable.en.yml</tt>) to check all available messages.
|
|
161
182
|
|
|
@@ -169,7 +190,7 @@ Check them all at:
|
|
|
169
190
|
|
|
170
191
|
http://github.com/scambra/devise_invitable/contributors
|
|
171
192
|
|
|
172
|
-
Special thanks to rymai[http://github.com/rymai] for
|
|
193
|
+
Special thanks to rymai[http://github.com/rymai] for the Rails 3 support, his fork was a great help.
|
|
173
194
|
|
|
174
195
|
== Note on Patches/Pull Requests
|
|
175
196
|
|
|
@@ -2,6 +2,7 @@ class Devise::InvitationsController < ApplicationController
|
|
|
2
2
|
include Devise::Controllers::InternalHelpers
|
|
3
3
|
|
|
4
4
|
before_filter :authenticate_inviter!, :only => [:new, :create]
|
|
5
|
+
before_filter :has_invitations_left?, :only => [:create]
|
|
5
6
|
before_filter :require_no_authentication, :only => [:edit, :update]
|
|
6
7
|
helper_method :after_sign_in_path_for
|
|
7
8
|
|
|
@@ -13,11 +14,11 @@ class Devise::InvitationsController < ApplicationController
|
|
|
13
14
|
|
|
14
15
|
# POST /resource/invitation
|
|
15
16
|
def create
|
|
16
|
-
self.resource = resource_class.invite!(params[resource_name])
|
|
17
|
+
self.resource = resource_class.invite!(params[resource_name], current_inviter)
|
|
17
18
|
|
|
18
19
|
if resource.errors.empty?
|
|
19
|
-
set_flash_message :notice, :send_instructions
|
|
20
|
-
redirect_to
|
|
20
|
+
set_flash_message :notice, :send_instructions, :email => self.resource.email
|
|
21
|
+
redirect_to after_sign_in_path_for(resource_name)
|
|
21
22
|
else
|
|
22
23
|
render_with_scope :new
|
|
23
24
|
end
|
|
@@ -25,9 +26,12 @@ class Devise::InvitationsController < ApplicationController
|
|
|
25
26
|
|
|
26
27
|
# GET /resource/invitation/accept?invitation_token=abcdef
|
|
27
28
|
def edit
|
|
28
|
-
self.resource = resource_class.
|
|
29
|
-
|
|
30
|
-
|
|
29
|
+
if params[:invitation_token] && self.resource = resource_class.first(:conditions => { :invitation_token => params[:invitation_token] })
|
|
30
|
+
render_with_scope :edit
|
|
31
|
+
else
|
|
32
|
+
set_flash_message(:alert, :invitation_token_invalid)
|
|
33
|
+
redirect_to after_sign_out_path_for(resource_name)
|
|
34
|
+
end
|
|
31
35
|
end
|
|
32
36
|
|
|
33
37
|
# PUT /resource/invitation
|
|
@@ -41,4 +45,17 @@ class Devise::InvitationsController < ApplicationController
|
|
|
41
45
|
render_with_scope :edit
|
|
42
46
|
end
|
|
43
47
|
end
|
|
48
|
+
|
|
49
|
+
protected
|
|
50
|
+
def current_inviter
|
|
51
|
+
@current_inviter ||= authenticate_inviter!
|
|
52
|
+
end
|
|
53
|
+
|
|
54
|
+
def has_invitations_left?
|
|
55
|
+
unless current_inviter.nil? || current_inviter.has_invitations_left?
|
|
56
|
+
build_resource
|
|
57
|
+
set_flash_message :alert, :no_invitations_remaining
|
|
58
|
+
render_with_scope :new
|
|
59
|
+
end
|
|
60
|
+
end
|
|
44
61
|
end
|
data/config/locales/en.yml
CHANGED
|
@@ -1,8 +1,10 @@
|
|
|
1
1
|
en:
|
|
2
2
|
devise:
|
|
3
3
|
invitations:
|
|
4
|
-
send_instructions: 'An email
|
|
4
|
+
send_instructions: 'An invitation email has been sent to %{email}.'
|
|
5
|
+
invitation_token_invalid: 'The invitation token provided is not valid!'
|
|
5
6
|
updated: 'Your password was set successfully. You are now signed in.'
|
|
7
|
+
no_invitations_remaining: "No invitations remaining"
|
|
6
8
|
mailer:
|
|
7
|
-
|
|
8
|
-
subject: 'Invitation'
|
|
9
|
+
invitation_instructions:
|
|
10
|
+
subject: 'Invitation instructions'
|
|
@@ -1,12 +1,7 @@
|
|
|
1
1
|
module DeviseInvitable::Controllers::Helpers
|
|
2
2
|
protected
|
|
3
|
-
def authenticate_resource!
|
|
4
|
-
ActiveSupport::Deprecation.warn('authenticate_resource! has been renamed to authenticate_inviter!')
|
|
5
|
-
authenticate_inviter!
|
|
6
|
-
end
|
|
7
|
-
|
|
8
3
|
def authenticate_inviter!
|
|
9
|
-
send(:"authenticate_#{resource_name}!")
|
|
4
|
+
send(:"authenticate_#{resource_name}!", true)
|
|
10
5
|
end
|
|
11
6
|
end
|
|
12
7
|
ActionController::Base.send :include, DeviseInvitable::Controllers::Helpers
|
|
@@ -1,8 +1,14 @@
|
|
|
1
1
|
module DeviseInvitable
|
|
2
2
|
module Mailer
|
|
3
|
-
|
|
3
|
+
|
|
4
|
+
# Deliver an invitation email
|
|
5
|
+
def invitation_instructions(record)
|
|
6
|
+
setup_mail(record, :invitation_instructions)
|
|
7
|
+
end
|
|
8
|
+
|
|
4
9
|
def invitation(record)
|
|
5
|
-
|
|
10
|
+
ActiveSupport::Deprecation.warn('invitation has been renamed to invitation_instructions')
|
|
11
|
+
invitation_instructions(record)
|
|
6
12
|
end
|
|
7
13
|
end
|
|
8
14
|
end
|
|
@@ -1,29 +1,36 @@
|
|
|
1
1
|
module Devise
|
|
2
2
|
module Models
|
|
3
|
-
# Invitable is responsible
|
|
4
|
-
# When an invitation is sent to an email, an account is created for it.
|
|
5
|
-
#
|
|
3
|
+
# Invitable is responsible for sending invitation emails.
|
|
4
|
+
# When an invitation is sent to an email address, an account is created for it.
|
|
5
|
+
# Invitation email contains a link allowing the user to accept the invitation
|
|
6
|
+
# by setting a password (as reset password from Devise's recoverable module).
|
|
6
7
|
#
|
|
7
8
|
# Configuration:
|
|
8
9
|
#
|
|
9
|
-
# invite_for:
|
|
10
|
-
#
|
|
11
|
-
#
|
|
10
|
+
# invite_for: The period the generated invitation token is valid, after
|
|
11
|
+
# this period, the invited resource won't be able to accept the invitation.
|
|
12
|
+
# When invite_for is 0 (the default), the invitation won't expire.
|
|
12
13
|
#
|
|
13
14
|
# Examples:
|
|
14
15
|
#
|
|
15
|
-
# User.find(1).invited?
|
|
16
|
-
# User.invite!(:email => 'someone@example.com')
|
|
17
|
-
# User.accept_invitation!(:invitation_token => '...')
|
|
18
|
-
# User.find(1).accept_invitation!
|
|
19
|
-
# User.find(1).invite!
|
|
16
|
+
# User.find(1).invited? # => true/false
|
|
17
|
+
# User.invite!(:email => 'someone@example.com') # => send invitation
|
|
18
|
+
# User.accept_invitation!(:invitation_token => '...') # => accept invitation with a token
|
|
19
|
+
# User.find(1).accept_invitation! # => accept invitation
|
|
20
|
+
# User.find(1).invite! # => reset invitation status and send invitation again
|
|
20
21
|
module Invitable
|
|
21
22
|
extend ActiveSupport::Concern
|
|
22
23
|
|
|
24
|
+
attr_accessor :skip_invitation
|
|
25
|
+
|
|
26
|
+
included do
|
|
27
|
+
belongs_to :invited_by, :polymorphic => true
|
|
28
|
+
end
|
|
29
|
+
|
|
23
30
|
# Accept an invitation by clearing invitation token and confirming it if model
|
|
24
31
|
# is confirmable
|
|
25
32
|
def accept_invitation!
|
|
26
|
-
if self.invited?
|
|
33
|
+
if self.invited? && self.valid?
|
|
27
34
|
self.invitation_token = nil
|
|
28
35
|
self.save
|
|
29
36
|
end
|
|
@@ -34,26 +41,33 @@ module Devise
|
|
|
34
41
|
persisted? && invitation_token.present?
|
|
35
42
|
end
|
|
36
43
|
|
|
37
|
-
#
|
|
38
|
-
def
|
|
39
|
-
|
|
44
|
+
# Return true if this user has invitations left to send
|
|
45
|
+
def has_invitations_left?
|
|
46
|
+
if self.class.invitation_limit.present?
|
|
47
|
+
if invitation_limit
|
|
48
|
+
return invitation_limit > 0
|
|
49
|
+
else
|
|
50
|
+
return self.class.invitation_limit > 0
|
|
51
|
+
end
|
|
52
|
+
else
|
|
53
|
+
return true
|
|
54
|
+
end
|
|
40
55
|
end
|
|
41
56
|
|
|
42
57
|
# Reset invitation token and send invitation again
|
|
43
58
|
def invite!
|
|
44
59
|
if new_record? || invited?
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
60
|
+
@skip_password = true
|
|
61
|
+
self.skip_confirmation! if self.new_record? && self.respond_to?(:skip_confirmation!)
|
|
62
|
+
generate_invitation_token if self.invitation_token.nil?
|
|
63
|
+
self.invitation_sent_at = Time.now.utc
|
|
64
|
+
if save(:validate => self.class.validate_on_invite)
|
|
65
|
+
self.invited_by.decrement_invitation_limit! if self.invited_by
|
|
66
|
+
!!deliver_invitation unless @skip_invitation
|
|
67
|
+
end
|
|
49
68
|
end
|
|
50
69
|
end
|
|
51
70
|
|
|
52
|
-
def resend_invitation!
|
|
53
|
-
ActiveSupport::Deprecation.warn('resend_invitation! has been renamed to invite!')
|
|
54
|
-
self.invite!
|
|
55
|
-
end
|
|
56
|
-
|
|
57
71
|
# Verify whether a invitation is active or not. If the user has been
|
|
58
72
|
# invited, we need to calculate if the invitation time has not expired
|
|
59
73
|
# for this user, in other words, if the invitation is still valid.
|
|
@@ -61,7 +75,34 @@ module Devise
|
|
|
61
75
|
invited? && invitation_period_valid?
|
|
62
76
|
end
|
|
63
77
|
|
|
78
|
+
# Only verify password when is not invited
|
|
79
|
+
def valid_password?(password)
|
|
80
|
+
super unless invited?
|
|
81
|
+
end
|
|
82
|
+
|
|
64
83
|
protected
|
|
84
|
+
def decrement_invitation_limit!
|
|
85
|
+
if self.class.invitation_limit.present?
|
|
86
|
+
self.invitation_limit ||= self.class.invitation_limit
|
|
87
|
+
self.decrement!(:invitation_limit)
|
|
88
|
+
end
|
|
89
|
+
end
|
|
90
|
+
|
|
91
|
+
# Overriding the method in Devise's :validatable module so password is not required on inviting
|
|
92
|
+
def password_required?
|
|
93
|
+
!@skip_password && super
|
|
94
|
+
end
|
|
95
|
+
|
|
96
|
+
# Deliver the invitation email
|
|
97
|
+
def deliver_invitation
|
|
98
|
+
::Devise.mailer.invitation_instructions(self).deliver
|
|
99
|
+
end
|
|
100
|
+
|
|
101
|
+
# Clear invitation token when reset password token is cleared too
|
|
102
|
+
def clear_reset_password_token
|
|
103
|
+
self.invitation_token = nil if invited?
|
|
104
|
+
super
|
|
105
|
+
end
|
|
65
106
|
|
|
66
107
|
# Checks if the invitation for the user is within the limit time.
|
|
67
108
|
# We do this by calculating if the difference between today and the
|
|
@@ -89,8 +130,7 @@ module Devise
|
|
|
89
130
|
# Generates a new random token for invitation, and stores the time
|
|
90
131
|
# this token is being generated
|
|
91
132
|
def generate_invitation_token
|
|
92
|
-
self.invitation_token
|
|
93
|
-
self.invitation_sent_at = Time.now.utc
|
|
133
|
+
self.invitation_token = self.class.invitation_token
|
|
94
134
|
end
|
|
95
135
|
|
|
96
136
|
module ClassMethods
|
|
@@ -98,33 +138,32 @@ module Devise
|
|
|
98
138
|
# user and send invitation to it. If user is found, returns the user with an
|
|
99
139
|
# email already exists error.
|
|
100
140
|
# Attributes must contain the user email, other attributes will be set in the record
|
|
101
|
-
def invite!(attributes={})
|
|
102
|
-
invitable = find_or_initialize_with_error_by(
|
|
141
|
+
def invite!(attributes={}, invited_by=nil, &block)
|
|
142
|
+
invitable = find_or_initialize_with_error_by(invite_key, attributes.delete(invite_key))
|
|
103
143
|
invitable.attributes = attributes
|
|
144
|
+
invitable.invited_by = invited_by
|
|
104
145
|
|
|
105
146
|
if invitable.new_record?
|
|
106
|
-
invitable.errors.clear if invitable.email.match Devise.email_regexp
|
|
147
|
+
invitable.errors.clear if invitable.email.try(:match, Devise.email_regexp)
|
|
107
148
|
else
|
|
108
|
-
invitable.errors.add(
|
|
149
|
+
invitable.errors.add(invite_key, :taken) unless invitable.invited?
|
|
109
150
|
end
|
|
110
151
|
|
|
111
|
-
|
|
152
|
+
if invitable.errors.empty?
|
|
153
|
+
yield invitable if block_given?
|
|
154
|
+
invitable.invite!
|
|
155
|
+
end
|
|
112
156
|
invitable
|
|
113
157
|
end
|
|
114
158
|
|
|
115
|
-
def send_invitation(attributes = {})
|
|
116
|
-
ActiveSupport::Deprecation.warn('send_invitation has been renamed to invite!')
|
|
117
|
-
self.invite!(attributes)
|
|
118
|
-
end
|
|
119
|
-
|
|
120
159
|
# Attempt to find a user by it's invitation_token to set it's password.
|
|
121
160
|
# If a user is found, reset it's password and automatically try saving
|
|
122
161
|
# the record. If not user is found, returns a new user containing an
|
|
123
162
|
# error in invitation_token attribute.
|
|
124
163
|
# Attributes must contain invitation_token, password and confirmation
|
|
125
164
|
def accept_invitation!(attributes={})
|
|
126
|
-
invitable = find_or_initialize_with_error_by(:invitation_token, attributes
|
|
127
|
-
invitable.errors.add(:invitation_token, :invalid) if
|
|
165
|
+
invitable = find_or_initialize_with_error_by(:invitation_token, attributes.delete(:invitation_token))
|
|
166
|
+
invitable.errors.add(:invitation_token, :invalid) if invitable.invitation_token && invitable.persisted? && !invitable.valid_invitation?
|
|
128
167
|
if invitable.errors.empty?
|
|
129
168
|
invitable.attributes = attributes
|
|
130
169
|
invitable.accept_invitation!
|
|
@@ -132,7 +171,15 @@ module Devise
|
|
|
132
171
|
invitable
|
|
133
172
|
end
|
|
134
173
|
|
|
174
|
+
# Generate a token checking if one does not already exist in the database.
|
|
175
|
+
def invitation_token
|
|
176
|
+
generate_token(:invitation_token)
|
|
177
|
+
end
|
|
178
|
+
|
|
135
179
|
Devise::Models.config(self, :invite_for)
|
|
180
|
+
Devise::Models.config(self, :validate_on_invite)
|
|
181
|
+
Devise::Models.config(self, :invitation_limit)
|
|
182
|
+
Devise::Models.config(self, :invite_key)
|
|
136
183
|
end
|
|
137
184
|
end
|
|
138
185
|
end
|
|
@@ -2,7 +2,7 @@ module DeviseInvitable
|
|
|
2
2
|
class Engine < ::Rails::Engine
|
|
3
3
|
|
|
4
4
|
ActiveSupport.on_load(:action_controller) { include DeviseInvitable::Controllers::UrlHelpers }
|
|
5
|
-
ActiveSupport.on_load(:action_view)
|
|
5
|
+
ActiveSupport.on_load(:action_view) { include DeviseInvitable::Controllers::UrlHelpers }
|
|
6
6
|
|
|
7
7
|
config.after_initialize do
|
|
8
8
|
require 'devise/mailer'
|
|
@@ -1,10 +1,13 @@
|
|
|
1
1
|
module ActionDispatch::Routing
|
|
2
2
|
class Mapper
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
3
|
+
|
|
4
|
+
protected
|
|
5
|
+
def devise_invitation(mapping, controllers)
|
|
6
|
+
resource :invitation, :only => [:new, :create, :update],
|
|
7
|
+
:path => mapping.path_names[:invitation], :controller => controllers[:invitations] do
|
|
8
|
+
get :edit, :path => mapping.path_names[:accept], :as => :accept
|
|
8
9
|
end
|
|
10
|
+
end
|
|
11
|
+
|
|
9
12
|
end
|
|
10
13
|
end
|
|
@@ -1,9 +1,34 @@
|
|
|
1
1
|
module DeviseInvitable
|
|
2
2
|
module Schema
|
|
3
|
-
#
|
|
3
|
+
# Add invitation_token and invitation_sent_at columns in the resource's database table.
|
|
4
|
+
#
|
|
5
|
+
# Examples
|
|
6
|
+
#
|
|
7
|
+
# # For a new resource migration:
|
|
8
|
+
# create_table :the_resources do
|
|
9
|
+
# t.database_authenticatable :null => false # you need at least this
|
|
10
|
+
# t.invitable
|
|
11
|
+
# ...
|
|
12
|
+
# end
|
|
13
|
+
# add_index :the_resources, :invitation_token # for invitable
|
|
14
|
+
#
|
|
15
|
+
# # or if the resource's table already exists, define a migration and put this in:
|
|
16
|
+
# change_table :the_resources do |t|
|
|
17
|
+
# t.string :invitation_token, :limit => 60
|
|
18
|
+
# t.datetime :invitation_sent_at
|
|
19
|
+
# t.index :invitation_token # for invitable
|
|
20
|
+
# end
|
|
21
|
+
#
|
|
22
|
+
# # And allow encrypted_password to be null:
|
|
23
|
+
# change_column :the_resources, :encrypted_password, :string, :null => true
|
|
24
|
+
# # the following line is only if you use Devise's encryptable module!
|
|
25
|
+
# change_column :the_resources, :password_salt, :string, :null => true
|
|
4
26
|
def invitable
|
|
5
|
-
apply_devise_schema :invitation_token, String, :limit =>
|
|
27
|
+
apply_devise_schema :invitation_token, String, :limit => 60
|
|
6
28
|
apply_devise_schema :invitation_sent_at, DateTime
|
|
29
|
+
apply_devise_schema :invitation_limit, Integer
|
|
30
|
+
apply_devise_schema :invited_by_id, Integer
|
|
31
|
+
apply_devise_schema :invited_by_type, String
|
|
7
32
|
end
|
|
8
33
|
end
|
|
9
34
|
end
|
data/lib/devise_invitable.rb
CHANGED
|
@@ -1,16 +1,45 @@
|
|
|
1
1
|
require 'devise'
|
|
2
2
|
|
|
3
|
-
module Devise
|
|
4
|
-
# Time interval where the invitation token is valid.
|
|
5
|
-
mattr_accessor :invite_for
|
|
6
|
-
@@invite_for = 0
|
|
7
|
-
end
|
|
8
|
-
|
|
9
|
-
Devise.add_module :invitable, :controller => :invitations, :model => 'devise_invitable/model', :route => :invitation
|
|
10
|
-
|
|
11
3
|
require 'devise_invitable/mailer'
|
|
12
4
|
require 'devise_invitable/routes'
|
|
13
5
|
require 'devise_invitable/schema'
|
|
14
6
|
require 'devise_invitable/controllers/url_helpers'
|
|
15
7
|
require 'devise_invitable/controllers/helpers'
|
|
16
8
|
require 'devise_invitable/rails'
|
|
9
|
+
|
|
10
|
+
module Devise
|
|
11
|
+
# Public: Validity period of the invitation token (default: 0). If
|
|
12
|
+
# invite_for is 0 or nil, the invitation will never expire.
|
|
13
|
+
# Set invite_for in the Devise configuration file (in config/initializers/devise.rb).
|
|
14
|
+
#
|
|
15
|
+
# config.invite_for = 2.weeks # => The invitation token will be valid 2 weeks
|
|
16
|
+
mattr_accessor :invite_for
|
|
17
|
+
@@invite_for = 0
|
|
18
|
+
|
|
19
|
+
# Public: Flag that force a record to be valid before being actually invited
|
|
20
|
+
# (default: false).
|
|
21
|
+
#
|
|
22
|
+
# Examples (in config/initializers/devise.rb)
|
|
23
|
+
#
|
|
24
|
+
# config.validate_on_invite = true
|
|
25
|
+
mattr_accessor :validate_on_invite
|
|
26
|
+
@@validate_on_invite = false
|
|
27
|
+
|
|
28
|
+
# Public: number of invitations the user is allowed to send
|
|
29
|
+
#
|
|
30
|
+
# Examples (in config/initializers/devise.rb)
|
|
31
|
+
#
|
|
32
|
+
# config.invitation_limit = nil
|
|
33
|
+
mattr_accessor :invitation_limit
|
|
34
|
+
@@invitation_limit = nil
|
|
35
|
+
|
|
36
|
+
# Public: The key to be used to check existing users when sending an invitation
|
|
37
|
+
#
|
|
38
|
+
# Examples (in config/initializers/devise.rb)
|
|
39
|
+
#
|
|
40
|
+
# config.invite_key = :email
|
|
41
|
+
mattr_accessor :invite_key
|
|
42
|
+
@@invite_key = :email
|
|
43
|
+
end
|
|
44
|
+
|
|
45
|
+
Devise.add_module :invitable, :controller => :invitations, :model => 'devise_invitable/model', :route => :invitation
|