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 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.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.1.3'
16
- gem 'devise_invitable', '~> 0.3.4'
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 a new configuration option:
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
- After an invitation is created and sent, the inviter will be redirected to after_sign_in_path_for(resource_name).
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
- 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.
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
- redirect_to after_sign_in_path_for(resource_name)
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
- sign_in_and_redirect(resource_name, resource)
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
@@ -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
- mailer:
8
- invitation_instructions:
9
- subject: 'Invitation instructions'
7
+ no_invitations_remaining: "No invitations remaining"
8
+ mailer:
9
+ invitation_instructions:
10
+ subject: 'Invitation instructions'
@@ -1,7 +1,7 @@
1
1
  module DeviseInvitable::Controllers::Helpers
2
2
  protected
3
3
  def authenticate_inviter!
4
- send(:"authenticate_#{resource_name}!")
4
+ send(:"authenticate_#{resource_name}!", true)
5
5
  end
6
6
  end
7
7
  ActionController::Base.send :include, DeviseInvitable::Controllers::Helpers
@@ -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 => false)
45
- ::Devise.mailer.invitation_instructions(self).deliver
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(:email, attributes.delete(:email))
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(:email, :taken) unless invitable.invited?
130
+ invitable.errors.add(invite_key, :taken) unless invitable.invited?
100
131
  end
101
132
 
102
- invitable.invite! if invitable.errors.empty?
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
- config.after_initialize do
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
@@ -1,3 +1,3 @@
1
1
  module DeviseInvitable
2
- VERSION = '0.4.rc'
2
+ VERSION = '0.5.0'
3
3
  end
@@ -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
- # The period the generated invitation token is valid.
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 :invitation_token, :limit => 60
5
- t.datetime :invitation_sent_at
6
- t.index :invitation_token # for invitable
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
- remove_column :<%= table_name %>, :invitation_sent_at
18
- remove_column :<%= table_name %>, :invitation_token
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
@@ -0,0 +1,8 @@
1
+ require 'generators/devise/orm_helpers'
2
+
3
+ module Mongoid
4
+ module Generators
5
+ class DeviseInvitableGenerator < Rails::Generators::NamedBase
6
+ end
7
+ end
8
+ 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: 7712090
5
- prerelease: true
4
+ hash: 11
5
+ prerelease:
6
6
  segments:
7
7
  - 0
8
- - 4
9
- - rc
10
- version: 0.4.rc
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-01-06 00:00:00 +01:00
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: &id003 !ruby/object:Gem::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: *id003
36
+ version_requirements: *id001
71
37
  - !ruby/object:Gem::Dependency
72
38
  name: rails
73
39
  prerelease: false
74
- requirement: &id004 !ruby/object:Gem::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: *id004
59
+ version_requirements: *id002
87
60
  - !ruby/object:Gem::Dependency
88
61
  name: devise
89
62
  prerelease: false
90
- requirement: &id005 !ruby/object:Gem::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: 7712074
68
+ hash: 25
96
69
  segments:
97
70
  - 1
98
- - 2
99
- - rc
100
- version: 1.2.rc
71
+ - 3
72
+ - 1
73
+ version: 1.3.1
101
74
  type: :runtime
102
- version_requirements: *id005
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/rails.rb
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/model.rb~
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/active_record/devise_invitable_generator.rb
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: http://github.com/rymai/devise_invitable
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.3.7
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