devise_invitable 0.4.rc → 0.5.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 +33 -6
- data/app/controllers/devise/invitations_controller.rb +24 -5
- data/config/locales/en.yml +4 -3
- data/lib/devise_invitable/controllers/helpers.rb +1 -1
- data/lib/devise_invitable/inviter.rb +35 -0
- data/lib/devise_invitable/model.rb +43 -6
- data/lib/devise_invitable/rails.rb +4 -1
- data/lib/devise_invitable/schema.rb +3 -0
- data/lib/devise_invitable/version.rb +1 -1
- data/lib/devise_invitable.rb +34 -1
- data/lib/generators/active_record/templates/migration.rb +10 -5
- data/lib/generators/devise_invitable/install_generator.rb +14 -0
- data/lib/generators/mongoid/devise_invitable_generator.rb +8 -0
- metadata +37 -65
- data/app/controllers/devise/invitations_controller.rb~ +0 -48
- data/app/views/devise/mailer/invitation.html.erb +0 -8
- data/lib/devise_invitable/model.rb~ +0 -129
data/README.rdoc
CHANGED
|
@@ -4,7 +4,7 @@ It adds support to devise[http://github.com/plataformatec/devise] for send invit
|
|
|
4
4
|
|
|
5
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 and Devise ~> 1.
|
|
7
|
+
== Installation for Rails ~> 3.0 and Devise ~> 1.2
|
|
8
8
|
|
|
9
9
|
Install DeviseInvitable gem, it will also install dependencies (such as devise and warden):
|
|
10
10
|
|
|
@@ -12,8 +12,8 @@ Install DeviseInvitable gem, it will also install dependencies (such as devise a
|
|
|
12
12
|
|
|
13
13
|
Add DeviseInvitable to your Gemfile (and Devise if you weren't using them):
|
|
14
14
|
|
|
15
|
-
gem 'devise', '~> 1.
|
|
16
|
-
gem 'devise_invitable', '~> 0.
|
|
15
|
+
gem 'devise', '~> 1.2.0'
|
|
16
|
+
gem 'devise_invitable', '~> 0.4.0'
|
|
17
17
|
|
|
18
18
|
=== Automatic installation
|
|
19
19
|
|
|
@@ -61,7 +61,7 @@ or for a model that already exists, define a migration to add DeviseInvitable to
|
|
|
61
61
|
|
|
62
62
|
== Model configuration
|
|
63
63
|
|
|
64
|
-
DeviseInvitable adds
|
|
64
|
+
DeviseInvitable adds four new configuration options:
|
|
65
65
|
|
|
66
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.
|
|
67
67
|
|
|
@@ -77,6 +77,12 @@ or directly as parameters to the <tt>devise</tt> method:
|
|
|
77
77
|
|
|
78
78
|
devise :database_authenticatable, :confirmable, :invitable, :invite_for => 2.weeks
|
|
79
79
|
|
|
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
|
+
* validate_on_invite: force a record to be valid before being actually invited.
|
|
85
|
+
|
|
80
86
|
For more details, see <tt>config/initializers/devise.rb</tt> (after you invoked the "devise_invitable:install" generator described above).
|
|
81
87
|
|
|
82
88
|
== Configuring views
|
|
@@ -100,6 +106,18 @@ To send an invitation to a user, use the <tt>invite!</tt> class method. <tt>:ema
|
|
|
100
106
|
User.invite!(:email => "new_user@example.com", :name => "John Doe")
|
|
101
107
|
# => an invitation email will be sent to new_user@example.com
|
|
102
108
|
|
|
109
|
+
If you want to create the invitation but not send it, you can set <tt>skip_invitation</tt> to true.
|
|
110
|
+
|
|
111
|
+
User.invite!(:email => "new_user@example.com", :name => "John Doe") do |u|
|
|
112
|
+
u.skip_invitation = true
|
|
113
|
+
end
|
|
114
|
+
# => the record will be created, but the invitation email will not be sent
|
|
115
|
+
|
|
116
|
+
You can add :skip_invitation to attributes hash if skip_invitation is added to attr_accessible.
|
|
117
|
+
|
|
118
|
+
User.invite!(:email => "new_user@example.com", :name => "John Doe", :skip_invitation => true)
|
|
119
|
+
# => the record will be created, but the invitation email will not be sent
|
|
120
|
+
|
|
103
121
|
=== Accept an invitation
|
|
104
122
|
|
|
105
123
|
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.
|
|
@@ -110,10 +128,12 @@ To accept an invitation with a token use the <tt>accept_invitation!</tt> class m
|
|
|
110
128
|
|
|
111
129
|
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.
|
|
112
130
|
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).
|
|
113
|
-
|
|
131
|
+
|
|
132
|
+
After an invitation is created and sent, the inviter will be redirected to after_accept_path_for(resource), which is the same path as after_sign_in_path_for by default. If you want to override the path, override invitations controller and define after_accept_path_for method. More on {Devise's README}[http://github.com/plataformatec/devise], "Controller filters and helpers" section.
|
|
114
133
|
|
|
115
134
|
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).
|
|
116
|
-
|
|
135
|
+
|
|
136
|
+
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.
|
|
117
137
|
|
|
118
138
|
== Controller filter
|
|
119
139
|
|
|
@@ -130,6 +150,13 @@ You would have a User model which is configured as invitable and an Admin model
|
|
|
130
150
|
end
|
|
131
151
|
end
|
|
132
152
|
|
|
153
|
+
And include DeviseInvitable::Inviter module into Admin model:
|
|
154
|
+
|
|
155
|
+
class Admin < ActiveRecord::Base
|
|
156
|
+
devise :database_authenticatable, :validatable
|
|
157
|
+
include DeviseInvitable::Inviter
|
|
158
|
+
end
|
|
159
|
+
|
|
133
160
|
== I18n
|
|
134
161
|
|
|
135
162
|
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:
|
|
@@ -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,13 +14,13 @@ 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
20
|
set_flash_message :notice, :send_instructions, :email => self.resource.email
|
|
20
|
-
|
|
21
|
+
respond_with resource, :location => redirect_location(resource_name, resource)
|
|
21
22
|
else
|
|
22
|
-
render_with_scope :new
|
|
23
|
+
respond_with_navigational(resource) { render_with_scope :new }
|
|
23
24
|
end
|
|
24
25
|
end
|
|
25
26
|
|
|
@@ -39,9 +40,27 @@ class Devise::InvitationsController < ApplicationController
|
|
|
39
40
|
|
|
40
41
|
if resource.errors.empty?
|
|
41
42
|
set_flash_message :notice, :updated
|
|
42
|
-
|
|
43
|
+
sign_in(resource_name, resource)
|
|
44
|
+
respond_with resource, :location => after_accept_path_for(resource)
|
|
43
45
|
else
|
|
44
|
-
render_with_scope :edit
|
|
46
|
+
respond_with_navigational(resource){ render_with_scope :edit }
|
|
47
|
+
end
|
|
48
|
+
end
|
|
49
|
+
|
|
50
|
+
protected
|
|
51
|
+
def current_inviter
|
|
52
|
+
@current_inviter ||= authenticate_inviter!
|
|
53
|
+
end
|
|
54
|
+
|
|
55
|
+
def has_invitations_left?
|
|
56
|
+
unless current_inviter.nil? || current_inviter.has_invitations_left?
|
|
57
|
+
build_resource
|
|
58
|
+
set_flash_message :alert, :no_invitations_remaining
|
|
59
|
+
respond_with_navigational(resource) { render_with_scope :new }
|
|
45
60
|
end
|
|
46
61
|
end
|
|
62
|
+
|
|
63
|
+
def after_accept_path_for(resource)
|
|
64
|
+
after_sign_in_path_for(resource)
|
|
65
|
+
end
|
|
47
66
|
end
|
data/config/locales/en.yml
CHANGED
|
@@ -4,6 +4,7 @@ en:
|
|
|
4
4
|
send_instructions: 'An invitation email has been sent to %{email}.'
|
|
5
5
|
invitation_token_invalid: 'The invitation token provided is not valid!'
|
|
6
6
|
updated: 'Your password was set successfully. You are now signed in.'
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
7
|
+
no_invitations_remaining: "No invitations remaining"
|
|
8
|
+
mailer:
|
|
9
|
+
invitation_instructions:
|
|
10
|
+
subject: 'Invitation instructions'
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
module DeviseInvitable
|
|
2
|
+
module Inviter
|
|
3
|
+
extend ActiveSupport::Concern
|
|
4
|
+
|
|
5
|
+
included do
|
|
6
|
+
extend ClassMethods
|
|
7
|
+
attr_accessor :invitation_limit unless respond_to? :invitation_limit
|
|
8
|
+
end
|
|
9
|
+
|
|
10
|
+
# Return true if this user has invitations left to send
|
|
11
|
+
def has_invitations_left?
|
|
12
|
+
if self.class.invitation_limit.present?
|
|
13
|
+
if invitation_limit
|
|
14
|
+
return invitation_limit > 0
|
|
15
|
+
else
|
|
16
|
+
return self.class.invitation_limit > 0
|
|
17
|
+
end
|
|
18
|
+
else
|
|
19
|
+
return true
|
|
20
|
+
end
|
|
21
|
+
end
|
|
22
|
+
|
|
23
|
+
protected
|
|
24
|
+
def decrement_invitation_limit!
|
|
25
|
+
if self.class.invitation_limit.present?
|
|
26
|
+
self.invitation_limit ||= self.class.invitation_limit
|
|
27
|
+
self.update_attribute(:invitation_limit, invitation_limit - 1)
|
|
28
|
+
end
|
|
29
|
+
end
|
|
30
|
+
|
|
31
|
+
module ClassMethods
|
|
32
|
+
Devise::Models.config(self, :invitation_limit)
|
|
33
|
+
end
|
|
34
|
+
end
|
|
35
|
+
end
|
|
@@ -21,6 +21,13 @@ module Devise
|
|
|
21
21
|
module Invitable
|
|
22
22
|
extend ActiveSupport::Concern
|
|
23
23
|
|
|
24
|
+
attr_accessor :skip_invitation
|
|
25
|
+
|
|
26
|
+
included do
|
|
27
|
+
include ::DeviseInvitable::Inviter
|
|
28
|
+
belongs_to :invited_by, :polymorphic => true
|
|
29
|
+
end
|
|
30
|
+
|
|
24
31
|
# Accept an invitation by clearing invitation token and confirming it if model
|
|
25
32
|
# is confirmable
|
|
26
33
|
def accept_invitation!
|
|
@@ -38,11 +45,14 @@ module Devise
|
|
|
38
45
|
# Reset invitation token and send invitation again
|
|
39
46
|
def invite!
|
|
40
47
|
if new_record? || invited?
|
|
48
|
+
@skip_password = true
|
|
41
49
|
self.skip_confirmation! if self.new_record? && self.respond_to?(:skip_confirmation!)
|
|
42
50
|
generate_invitation_token if self.invitation_token.nil?
|
|
43
51
|
self.invitation_sent_at = Time.now.utc
|
|
44
|
-
save(:validate =>
|
|
45
|
-
|
|
52
|
+
if save(:validate => self.class.validate_on_invite)
|
|
53
|
+
self.invited_by.decrement_invitation_limit! if self.invited_by
|
|
54
|
+
!!deliver_invitation unless @skip_invitation
|
|
55
|
+
end
|
|
46
56
|
end
|
|
47
57
|
end
|
|
48
58
|
|
|
@@ -53,7 +63,27 @@ module Devise
|
|
|
53
63
|
invited? && invitation_period_valid?
|
|
54
64
|
end
|
|
55
65
|
|
|
66
|
+
# Only verify password when is not invited
|
|
67
|
+
def valid_password?(password)
|
|
68
|
+
super unless invited?
|
|
69
|
+
end
|
|
70
|
+
|
|
56
71
|
protected
|
|
72
|
+
# Overriding the method in Devise's :validatable module so password is not required on inviting
|
|
73
|
+
def password_required?
|
|
74
|
+
!@skip_password && super
|
|
75
|
+
end
|
|
76
|
+
|
|
77
|
+
# Deliver the invitation email
|
|
78
|
+
def deliver_invitation
|
|
79
|
+
::Devise.mailer.invitation_instructions(self).deliver
|
|
80
|
+
end
|
|
81
|
+
|
|
82
|
+
# Clear invitation token when reset password token is cleared too
|
|
83
|
+
def clear_reset_password_token
|
|
84
|
+
self.invitation_token = nil if invited?
|
|
85
|
+
super
|
|
86
|
+
end
|
|
57
87
|
|
|
58
88
|
# Checks if the invitation for the user is within the limit time.
|
|
59
89
|
# We do this by calculating if the difference between today and the
|
|
@@ -89,17 +119,21 @@ module Devise
|
|
|
89
119
|
# user and send invitation to it. If user is found, returns the user with an
|
|
90
120
|
# email already exists error.
|
|
91
121
|
# Attributes must contain the user email, other attributes will be set in the record
|
|
92
|
-
def invite!(attributes={})
|
|
93
|
-
invitable = find_or_initialize_with_error_by(
|
|
122
|
+
def invite!(attributes={}, invited_by=nil, &block)
|
|
123
|
+
invitable = find_or_initialize_with_error_by(invite_key, attributes.delete(invite_key))
|
|
94
124
|
invitable.attributes = attributes
|
|
125
|
+
invitable.invited_by = invited_by
|
|
95
126
|
|
|
96
127
|
if invitable.new_record?
|
|
97
128
|
invitable.errors.clear if invitable.email.try(:match, Devise.email_regexp)
|
|
98
129
|
else
|
|
99
|
-
invitable.errors.add(
|
|
130
|
+
invitable.errors.add(invite_key, :taken) unless invitable.invited?
|
|
100
131
|
end
|
|
101
132
|
|
|
102
|
-
|
|
133
|
+
if invitable.errors.empty?
|
|
134
|
+
yield invitable if block_given?
|
|
135
|
+
invitable.invite!
|
|
136
|
+
end
|
|
103
137
|
invitable
|
|
104
138
|
end
|
|
105
139
|
|
|
@@ -124,6 +158,9 @@ module Devise
|
|
|
124
158
|
end
|
|
125
159
|
|
|
126
160
|
Devise::Models.config(self, :invite_for)
|
|
161
|
+
Devise::Models.config(self, :validate_on_invite)
|
|
162
|
+
Devise::Models.config(self, :invitation_limit)
|
|
163
|
+
Devise::Models.config(self, :invite_key)
|
|
127
164
|
end
|
|
128
165
|
end
|
|
129
166
|
end
|
|
@@ -4,7 +4,10 @@ module DeviseInvitable
|
|
|
4
4
|
ActiveSupport.on_load(:action_controller) { include DeviseInvitable::Controllers::UrlHelpers }
|
|
5
5
|
ActiveSupport.on_load(:action_view) { include DeviseInvitable::Controllers::UrlHelpers }
|
|
6
6
|
|
|
7
|
-
|
|
7
|
+
# We use to_prepare instead of after_initialize here because Devise is a Rails engine; its
|
|
8
|
+
# mailer is reloaded like the rest of the user's app. Got to make sure that our mailer methods
|
|
9
|
+
# are included each time Devise::Mailer is (re)loaded.
|
|
10
|
+
config.to_prepare do
|
|
8
11
|
require 'devise/mailer'
|
|
9
12
|
Devise::Mailer.send :include, DeviseInvitable::Mailer
|
|
10
13
|
end
|
|
@@ -26,6 +26,9 @@ module DeviseInvitable
|
|
|
26
26
|
def invitable
|
|
27
27
|
apply_devise_schema :invitation_token, String, :limit => 60
|
|
28
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
|
|
29
32
|
end
|
|
30
33
|
end
|
|
31
34
|
end
|
data/lib/devise_invitable.rb
CHANGED
|
@@ -1,5 +1,9 @@
|
|
|
1
1
|
require 'devise'
|
|
2
2
|
|
|
3
|
+
module DeviseInvitable
|
|
4
|
+
autoload :Inviter, 'devise_invitable/inviter'
|
|
5
|
+
end
|
|
6
|
+
|
|
3
7
|
require 'devise_invitable/mailer'
|
|
4
8
|
require 'devise_invitable/routes'
|
|
5
9
|
require 'devise_invitable/schema'
|
|
@@ -8,9 +12,38 @@ require 'devise_invitable/controllers/helpers'
|
|
|
8
12
|
require 'devise_invitable/rails'
|
|
9
13
|
|
|
10
14
|
module Devise
|
|
11
|
-
#
|
|
15
|
+
# Public: Validity period of the invitation token (default: 0). If
|
|
16
|
+
# invite_for is 0 or nil, the invitation will never expire.
|
|
17
|
+
# Set invite_for in the Devise configuration file (in config/initializers/devise.rb).
|
|
18
|
+
#
|
|
19
|
+
# config.invite_for = 2.weeks # => The invitation token will be valid 2 weeks
|
|
12
20
|
mattr_accessor :invite_for
|
|
13
21
|
@@invite_for = 0
|
|
22
|
+
|
|
23
|
+
# Public: Flag that force a record to be valid before being actually invited
|
|
24
|
+
# (default: false).
|
|
25
|
+
#
|
|
26
|
+
# Examples (in config/initializers/devise.rb)
|
|
27
|
+
#
|
|
28
|
+
# config.validate_on_invite = true
|
|
29
|
+
mattr_accessor :validate_on_invite
|
|
30
|
+
@@validate_on_invite = false
|
|
31
|
+
|
|
32
|
+
# Public: number of invitations the user is allowed to send
|
|
33
|
+
#
|
|
34
|
+
# Examples (in config/initializers/devise.rb)
|
|
35
|
+
#
|
|
36
|
+
# config.invitation_limit = nil
|
|
37
|
+
mattr_accessor :invitation_limit
|
|
38
|
+
@@invitation_limit = nil
|
|
39
|
+
|
|
40
|
+
# Public: The key to be used to check existing users when sending an invitation
|
|
41
|
+
#
|
|
42
|
+
# Examples (in config/initializers/devise.rb)
|
|
43
|
+
#
|
|
44
|
+
# config.invite_key = :email
|
|
45
|
+
mattr_accessor :invite_key
|
|
46
|
+
@@invite_key = :email
|
|
14
47
|
end
|
|
15
48
|
|
|
16
49
|
Devise.add_module :invitable, :controller => :invitations, :model => 'devise_invitable/model', :route => :invitation
|
|
@@ -1,9 +1,12 @@
|
|
|
1
1
|
class DeviseInvitableAddTo<%= table_name.camelize %> < ActiveRecord::Migration
|
|
2
2
|
def self.up
|
|
3
3
|
change_table :<%= table_name %> do |t|
|
|
4
|
-
t.string
|
|
5
|
-
t.datetime
|
|
6
|
-
t.
|
|
4
|
+
t.string :invitation_token, :limit => 60
|
|
5
|
+
t.datetime :invitation_sent_at
|
|
6
|
+
t.integer :invitation_limit
|
|
7
|
+
t.references :invited_by, :polymorphic => true
|
|
8
|
+
t.index :invitation_token # for invitable
|
|
9
|
+
t.index :invited_by_id
|
|
7
10
|
end
|
|
8
11
|
|
|
9
12
|
# And allow null encrypted_password and password_salt:
|
|
@@ -14,7 +17,9 @@ class DeviseInvitableAddTo<%= table_name.camelize %> < ActiveRecord::Migration
|
|
|
14
17
|
end
|
|
15
18
|
|
|
16
19
|
def self.down
|
|
17
|
-
|
|
18
|
-
|
|
20
|
+
change_table :<%= table_name %> do |t|
|
|
21
|
+
t.remove_references :invited_by, :polymorphic => true
|
|
22
|
+
t.remove :invitation_limit, :invitation_sent_at, :invitation_token
|
|
23
|
+
end
|
|
19
24
|
end
|
|
20
25
|
end
|
|
@@ -24,6 +24,20 @@ module DeviseInvitable
|
|
|
24
24
|
# When invite_for is 0 (the default), the invitation won't expire.
|
|
25
25
|
# config.invite_for = 2.weeks
|
|
26
26
|
|
|
27
|
+
# Number of invitations users can send.
|
|
28
|
+
# If invitation_limit is nil, users can send unlimited invitations.
|
|
29
|
+
# If invitation_limit is 0, users can't send invitations.
|
|
30
|
+
# If invitation_limit n > 0, users can send n invitations.
|
|
31
|
+
# Default: nil
|
|
32
|
+
# config.invitation_limit = 5
|
|
33
|
+
|
|
34
|
+
# The key to be used to check existing users when sending an invitation
|
|
35
|
+
# config.invite_key = :email
|
|
36
|
+
|
|
37
|
+
# Flag that force a record to be valid before being actually invited
|
|
38
|
+
# Default: false
|
|
39
|
+
# config.validate_on_invite = true
|
|
40
|
+
|
|
27
41
|
CONTENT
|
|
28
42
|
end
|
|
29
43
|
end
|
metadata
CHANGED
|
@@ -1,13 +1,13 @@
|
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
|
2
2
|
name: devise_invitable
|
|
3
3
|
version: !ruby/object:Gem::Version
|
|
4
|
-
hash:
|
|
5
|
-
prerelease:
|
|
4
|
+
hash: 11
|
|
5
|
+
prerelease:
|
|
6
6
|
segments:
|
|
7
7
|
- 0
|
|
8
|
-
-
|
|
9
|
-
-
|
|
10
|
-
version: 0.
|
|
8
|
+
- 5
|
|
9
|
+
- 0
|
|
10
|
+
version: 0.5.0
|
|
11
11
|
platform: ruby
|
|
12
12
|
authors:
|
|
13
13
|
- Sergio Cambra
|
|
@@ -15,47 +15,13 @@ autorequire:
|
|
|
15
15
|
bindir: bin
|
|
16
16
|
cert_chain: []
|
|
17
17
|
|
|
18
|
-
date: 2011-
|
|
18
|
+
date: 2011-05-09 00:00:00 +02:00
|
|
19
19
|
default_executable:
|
|
20
20
|
dependencies:
|
|
21
|
-
- !ruby/object:Gem::Dependency
|
|
22
|
-
name: rspec-rails
|
|
23
|
-
prerelease: false
|
|
24
|
-
requirement: &id001 !ruby/object:Gem::Requirement
|
|
25
|
-
none: false
|
|
26
|
-
requirements:
|
|
27
|
-
- - ~>
|
|
28
|
-
- !ruby/object:Gem::Version
|
|
29
|
-
hash: 11
|
|
30
|
-
segments:
|
|
31
|
-
- 2
|
|
32
|
-
- 1
|
|
33
|
-
- 0
|
|
34
|
-
version: 2.1.0
|
|
35
|
-
type: :development
|
|
36
|
-
version_requirements: *id001
|
|
37
|
-
- !ruby/object:Gem::Dependency
|
|
38
|
-
name: steak
|
|
39
|
-
prerelease: false
|
|
40
|
-
requirement: &id002 !ruby/object:Gem::Requirement
|
|
41
|
-
none: false
|
|
42
|
-
requirements:
|
|
43
|
-
- - ~>
|
|
44
|
-
- !ruby/object:Gem::Version
|
|
45
|
-
hash: 15424051
|
|
46
|
-
segments:
|
|
47
|
-
- 1
|
|
48
|
-
- 0
|
|
49
|
-
- 0
|
|
50
|
-
- rc
|
|
51
|
-
- 3
|
|
52
|
-
version: 1.0.0.rc.3
|
|
53
|
-
type: :development
|
|
54
|
-
version_requirements: *id002
|
|
55
21
|
- !ruby/object:Gem::Dependency
|
|
56
22
|
name: bundler
|
|
57
23
|
prerelease: false
|
|
58
|
-
requirement: &
|
|
24
|
+
requirement: &id001 !ruby/object:Gem::Requirement
|
|
59
25
|
none: false
|
|
60
26
|
requirements:
|
|
61
27
|
- - ~>
|
|
@@ -67,14 +33,14 @@ dependencies:
|
|
|
67
33
|
- 7
|
|
68
34
|
version: 1.0.7
|
|
69
35
|
type: :development
|
|
70
|
-
version_requirements: *
|
|
36
|
+
version_requirements: *id001
|
|
71
37
|
- !ruby/object:Gem::Dependency
|
|
72
38
|
name: rails
|
|
73
39
|
prerelease: false
|
|
74
|
-
requirement: &
|
|
40
|
+
requirement: &id002 !ruby/object:Gem::Requirement
|
|
75
41
|
none: false
|
|
76
42
|
requirements:
|
|
77
|
-
- -
|
|
43
|
+
- - ">="
|
|
78
44
|
- !ruby/object:Gem::Version
|
|
79
45
|
hash: 7
|
|
80
46
|
segments:
|
|
@@ -82,24 +48,31 @@ dependencies:
|
|
|
82
48
|
- 0
|
|
83
49
|
- 0
|
|
84
50
|
version: 3.0.0
|
|
51
|
+
- - <=
|
|
52
|
+
- !ruby/object:Gem::Version
|
|
53
|
+
hash: 3
|
|
54
|
+
segments:
|
|
55
|
+
- 3
|
|
56
|
+
- 2
|
|
57
|
+
version: "3.2"
|
|
85
58
|
type: :runtime
|
|
86
|
-
version_requirements: *
|
|
59
|
+
version_requirements: *id002
|
|
87
60
|
- !ruby/object:Gem::Dependency
|
|
88
61
|
name: devise
|
|
89
62
|
prerelease: false
|
|
90
|
-
requirement: &
|
|
63
|
+
requirement: &id003 !ruby/object:Gem::Requirement
|
|
91
64
|
none: false
|
|
92
65
|
requirements:
|
|
93
66
|
- - ~>
|
|
94
67
|
- !ruby/object:Gem::Version
|
|
95
|
-
hash:
|
|
68
|
+
hash: 25
|
|
96
69
|
segments:
|
|
97
70
|
- 1
|
|
98
|
-
-
|
|
99
|
-
-
|
|
100
|
-
version: 1.
|
|
71
|
+
- 3
|
|
72
|
+
- 1
|
|
73
|
+
version: 1.3.1
|
|
101
74
|
type: :runtime
|
|
102
|
-
version_requirements: *
|
|
75
|
+
version_requirements: *id003
|
|
103
76
|
description: It adds support for send invitations by email (it requires to be authenticated) and accept the invitation by setting a password.
|
|
104
77
|
email:
|
|
105
78
|
- sergio@entrecables.com
|
|
@@ -110,32 +83,31 @@ extensions: []
|
|
|
110
83
|
extra_rdoc_files: []
|
|
111
84
|
|
|
112
85
|
files:
|
|
86
|
+
- app/controllers/devise/invitations_controller.rb
|
|
113
87
|
- app/views/devise/invitations/edit.html.erb
|
|
114
88
|
- app/views/devise/invitations/new.html.erb
|
|
115
89
|
- app/views/devise/mailer/invitation_instructions.html.erb
|
|
116
|
-
- app/views/devise/mailer/invitation.html.erb
|
|
117
|
-
- app/controllers/devise/invitations_controller.rb~
|
|
118
|
-
- app/controllers/devise/invitations_controller.rb
|
|
119
90
|
- config/locales/en.yml
|
|
120
|
-
- lib/devise_invitable
|
|
121
|
-
- lib/devise_invitable/schema.rb
|
|
91
|
+
- lib/devise_invitable.rb
|
|
122
92
|
- lib/devise_invitable/mailer.rb
|
|
123
|
-
- lib/devise_invitable/routes.rb
|
|
124
|
-
- lib/devise_invitable/version.rb
|
|
125
93
|
- lib/devise_invitable/model.rb
|
|
126
|
-
- lib/devise_invitable/
|
|
94
|
+
- lib/devise_invitable/rails.rb
|
|
95
|
+
- lib/devise_invitable/routes.rb
|
|
96
|
+
- lib/devise_invitable/schema.rb
|
|
127
97
|
- lib/devise_invitable/controllers/helpers.rb
|
|
128
98
|
- lib/devise_invitable/controllers/url_helpers.rb
|
|
129
|
-
- lib/devise_invitable.rb
|
|
99
|
+
- lib/devise_invitable/version.rb
|
|
100
|
+
- lib/devise_invitable/inviter.rb
|
|
101
|
+
- lib/generators/active_record/devise_invitable_generator.rb
|
|
102
|
+
- lib/generators/active_record/templates/migration.rb
|
|
130
103
|
- lib/generators/devise_invitable/views_generator.rb
|
|
131
104
|
- lib/generators/devise_invitable/devise_invitable_generator.rb
|
|
132
105
|
- lib/generators/devise_invitable/install_generator.rb
|
|
133
|
-
- lib/generators/
|
|
134
|
-
- lib/generators/active_record/templates/migration.rb
|
|
106
|
+
- lib/generators/mongoid/devise_invitable_generator.rb
|
|
135
107
|
- LICENSE
|
|
136
108
|
- README.rdoc
|
|
137
109
|
has_rdoc: true
|
|
138
|
-
homepage:
|
|
110
|
+
homepage: https://github.com/scambra/devise_invitable
|
|
139
111
|
licenses: []
|
|
140
112
|
|
|
141
113
|
post_install_message:
|
|
@@ -159,7 +131,7 @@ required_ruby_version: !ruby/object:Gem::Requirement
|
|
|
159
131
|
required_rubygems_version: !ruby/object:Gem::Requirement
|
|
160
132
|
none: false
|
|
161
133
|
requirements:
|
|
162
|
-
- -
|
|
134
|
+
- - ">="
|
|
163
135
|
- !ruby/object:Gem::Version
|
|
164
136
|
hash: 23
|
|
165
137
|
segments:
|
|
@@ -170,7 +142,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
|
170
142
|
requirements: []
|
|
171
143
|
|
|
172
144
|
rubyforge_project:
|
|
173
|
-
rubygems_version: 1.
|
|
145
|
+
rubygems_version: 1.5.2
|
|
174
146
|
signing_key:
|
|
175
147
|
specification_version: 3
|
|
176
148
|
summary: An invitation strategy for Devise
|
|
@@ -1,48 +0,0 @@
|
|
|
1
|
-
class Devise::InvitationsController < ApplicationController
|
|
2
|
-
include Devise::Controllers::InternalHelpers
|
|
3
|
-
|
|
4
|
-
before_filter :authenticate_inviter!, :only => [:new, :create]
|
|
5
|
-
before_filter :require_no_authentication, :only => [:edit, :update]
|
|
6
|
-
helper_method :after_sign_in_path_for
|
|
7
|
-
|
|
8
|
-
# GET /resource/invitation/new
|
|
9
|
-
def new
|
|
10
|
-
build_resource
|
|
11
|
-
render_with_scope :new
|
|
12
|
-
end
|
|
13
|
-
|
|
14
|
-
# POST /resource/invitation
|
|
15
|
-
def create
|
|
16
|
-
self.resource = resource_class.invite!(params[resource_name])
|
|
17
|
-
|
|
18
|
-
if resource.errors.empty?
|
|
19
|
-
puts params.inspect
|
|
20
|
-
set_flash_message :notice, :send_instructions, :email => self.resource.email
|
|
21
|
-
redirect_to after_sign_in_path_for(resource_name)
|
|
22
|
-
else
|
|
23
|
-
render_with_scope :new
|
|
24
|
-
end
|
|
25
|
-
end
|
|
26
|
-
|
|
27
|
-
# GET /resource/invitation/accept?invitation_token=abcdef
|
|
28
|
-
def edit
|
|
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
|
|
35
|
-
end
|
|
36
|
-
|
|
37
|
-
# PUT /resource/invitation
|
|
38
|
-
def update
|
|
39
|
-
self.resource = resource_class.accept_invitation!(params[resource_name])
|
|
40
|
-
|
|
41
|
-
if resource.errors.empty?
|
|
42
|
-
set_flash_message :notice, :updated
|
|
43
|
-
sign_in_and_redirect(resource_name, resource)
|
|
44
|
-
else
|
|
45
|
-
render_with_scope :edit
|
|
46
|
-
end
|
|
47
|
-
end
|
|
48
|
-
end
|
|
@@ -1,8 +0,0 @@
|
|
|
1
|
-
<p>Hello <%= @resource.email %>!</p>
|
|
2
|
-
|
|
3
|
-
<p>Someone has invited you to <%= root_url %>, you can accept it through the link below.</p>
|
|
4
|
-
|
|
5
|
-
<p><%= link_to 'Accept invitation', accept_invitation_url(@resource, :invitation_token => @resource.invitation_token) %></p>
|
|
6
|
-
|
|
7
|
-
<p>If you don't want to accept the invitation, please ignore this email.<br />
|
|
8
|
-
Your account won't be created until you access the link above and set your password.</p>
|
|
@@ -1,129 +0,0 @@
|
|
|
1
|
-
module Devise
|
|
2
|
-
module Models
|
|
3
|
-
# Invitable is responsible to send emails with invitations.
|
|
4
|
-
# When an invitation is sent to an email, an account is created for it.
|
|
5
|
-
# An invitation has a link to set the password, as reset password from recoverable.
|
|
6
|
-
#
|
|
7
|
-
# Configuration:
|
|
8
|
-
#
|
|
9
|
-
# invite_for: the time you want the user will have to confirm the account after
|
|
10
|
-
# is invited. When invite_for is zero, the invitation won't expire.
|
|
11
|
-
# By default invite_for is 0.
|
|
12
|
-
#
|
|
13
|
-
# Examples:
|
|
14
|
-
#
|
|
15
|
-
# User.find(1).invited? # true/false
|
|
16
|
-
# User.invite!(:email => 'someone@example.com') # send invitation
|
|
17
|
-
# User.accept_invitation!(:invitation_token => '...') # accept invitation with a token
|
|
18
|
-
# User.find(1).accept_invitation! # accept invitation
|
|
19
|
-
# User.find(1).invite! # reset invitation status and send invitation again
|
|
20
|
-
module Invitable
|
|
21
|
-
extend ActiveSupport::Concern
|
|
22
|
-
|
|
23
|
-
# Accept an invitation by clearing invitation token and confirming it if model
|
|
24
|
-
# is confirmable
|
|
25
|
-
def accept_invitation!
|
|
26
|
-
if self.invited?
|
|
27
|
-
self.invitation_token = nil
|
|
28
|
-
self.save
|
|
29
|
-
end
|
|
30
|
-
end
|
|
31
|
-
|
|
32
|
-
# Verifies whether a user has been invited or not
|
|
33
|
-
def invited?
|
|
34
|
-
persisted? && invitation_token.present?
|
|
35
|
-
end
|
|
36
|
-
|
|
37
|
-
# Send invitation by email
|
|
38
|
-
def send_invitation
|
|
39
|
-
::Devise.mailer.invitation(self).deliver
|
|
40
|
-
end
|
|
41
|
-
|
|
42
|
-
# Reset invitation token and send invitation again
|
|
43
|
-
def invite!
|
|
44
|
-
if new_record? || invited?
|
|
45
|
-
self.skip_confirmation! if self.new_record? and self.respond_to? :skip_confirmation!
|
|
46
|
-
generate_invitation_token
|
|
47
|
-
save(:validate=>false)
|
|
48
|
-
send_invitation
|
|
49
|
-
end
|
|
50
|
-
end
|
|
51
|
-
|
|
52
|
-
# Verify whether a invitation is active or not. If the user has been
|
|
53
|
-
# invited, we need to calculate if the invitation time has not expired
|
|
54
|
-
# for this user, in other words, if the invitation is still valid.
|
|
55
|
-
def valid_invitation?
|
|
56
|
-
invited? && invitation_period_valid?
|
|
57
|
-
end
|
|
58
|
-
|
|
59
|
-
protected
|
|
60
|
-
|
|
61
|
-
# Checks if the invitation for the user is within the limit time.
|
|
62
|
-
# We do this by calculating if the difference between today and the
|
|
63
|
-
# invitation sent date does not exceed the invite for time configured.
|
|
64
|
-
# Invite_for is a model configuration, must always be an integer value.
|
|
65
|
-
#
|
|
66
|
-
# Example:
|
|
67
|
-
#
|
|
68
|
-
# # invite_for = 1.day and invitation_sent_at = today
|
|
69
|
-
# invitation_period_valid? # returns true
|
|
70
|
-
#
|
|
71
|
-
# # invite_for = 5.days and invitation_sent_at = 4.days.ago
|
|
72
|
-
# invitation_period_valid? # returns true
|
|
73
|
-
#
|
|
74
|
-
# # invite_for = 5.days and invitation_sent_at = 5.days.ago
|
|
75
|
-
# invitation_period_valid? # returns false
|
|
76
|
-
#
|
|
77
|
-
# # invite_for = nil
|
|
78
|
-
# invitation_period_valid? # will always return true
|
|
79
|
-
#
|
|
80
|
-
def invitation_period_valid?
|
|
81
|
-
invitation_sent_at && (self.class.invite_for.to_i.zero? || invitation_sent_at.utc >= self.class.invite_for.ago)
|
|
82
|
-
end
|
|
83
|
-
|
|
84
|
-
# Generates a new random token for invitation, and stores the time
|
|
85
|
-
# this token is being generated
|
|
86
|
-
def generate_invitation_token
|
|
87
|
-
self.invitation_token = Devise.friendly_token
|
|
88
|
-
self.invitation_sent_at = Time.now.utc
|
|
89
|
-
end
|
|
90
|
-
|
|
91
|
-
module ClassMethods
|
|
92
|
-
# Attempt to find a user by it's email. If a record is not found, create a new
|
|
93
|
-
# user and send invitation to it. If user is found, returns the user with an
|
|
94
|
-
# email already exists error.
|
|
95
|
-
# Attributes must contain the user email, other attributes will be set in the record
|
|
96
|
-
def invite!(attributes={})
|
|
97
|
-
invitable = find_or_initialize_with_error_by(:email, attributes.delete(:email))
|
|
98
|
-
invitable.attributes = attributes
|
|
99
|
-
|
|
100
|
-
if invitable.new_record?
|
|
101
|
-
invitable.errors.clear if invitable.email.try(:match, Devise.email_regexp)
|
|
102
|
-
else
|
|
103
|
-
invitable.errors.add(:email, :taken) unless invitable.invited?
|
|
104
|
-
end
|
|
105
|
-
|
|
106
|
-
invitable.invite! if invitable.errors.empty?
|
|
107
|
-
invitable
|
|
108
|
-
end
|
|
109
|
-
|
|
110
|
-
# Attempt to find a user by it's invitation_token to set it's password.
|
|
111
|
-
# If a user is found, reset it's password and automatically try saving
|
|
112
|
-
# the record. If not user is found, returns a new user containing an
|
|
113
|
-
# error in invitation_token attribute.
|
|
114
|
-
# Attributes must contain invitation_token, password and confirmation
|
|
115
|
-
def accept_invitation!(attributes={})
|
|
116
|
-
invitable = find_or_initialize_with_error_by(:invitation_token, attributes.delete(:invitation_token))
|
|
117
|
-
invitable.errors.add(:invitation_token, :invalid) if attributes[:invitation_token] && !invitable.new_record? && !invitable.valid_invitation?
|
|
118
|
-
if invitable.errors.empty?
|
|
119
|
-
invitable.attributes = attributes
|
|
120
|
-
invitable.accept_invitation!
|
|
121
|
-
end
|
|
122
|
-
invitable
|
|
123
|
-
end
|
|
124
|
-
|
|
125
|
-
Devise::Models.config(self, :invite_for)
|
|
126
|
-
end
|
|
127
|
-
end
|
|
128
|
-
end
|
|
129
|
-
end
|