Merge pull request #2854 from techAPJ/patch-1

Feature: resend invites
This commit is contained in:
Robin Ward 2014-10-06 17:25:36 -04:00
commit e383a8ab01
8 changed files with 79 additions and 4 deletions

View File

@ -72,6 +72,17 @@ export default Ember.ObjectController.extend({
return false; return false;
}, },
/**
Resend a given invite
@method reinvite
@param {Discourse.Invite} invite the invite to resend.
**/
reinvite: function(invite) {
invite.reinvite();
return false;
},
loadMore: function() { loadMore: function() {
var self = this; var self = this;
var model = self.get('model'); var model = self.get('model');

View File

@ -15,6 +15,14 @@ Discourse.Invite = Discourse.Model.extend({
data: { email: this.get('email') } data: { email: this.get('email') }
}); });
this.set('rescinded', true); this.set('rescinded', true);
},
reinvite: function() {
Discourse.ajax('/invites/reinvite', {
type: 'POST',
data: { email: this.get('email') }
});
this.set('reinvited', true);
} }
}); });
@ -46,5 +54,3 @@ Discourse.Invite.reopenClass({
} }
}); });

View File

@ -51,12 +51,19 @@
<td colspan='6'> <td colspan='6'>
{{#if invite.expired}} {{#if invite.expired}}
{{i18n user.invited.expired}} {{i18n user.invited.expired}}
&nbsp;&nbsp;&nbsp;&nbsp;
{{/if}} {{/if}}
{{#if invite.rescinded}} {{#if invite.rescinded}}
{{i18n user.invited.rescinded}} {{i18n user.invited.rescinded}}
{{else}} {{else}}
<button class='btn' {{action "rescind" invite}}><i class="fa fa-times"></i>{{i18n user.invited.rescind}}</button> <button class='btn' {{action "rescind" invite}}><i class="fa fa-times"></i>{{i18n user.invited.rescind}}</button>
{{/if}} {{/if}}
&nbsp;&nbsp;&nbsp;&nbsp;
{{#if invite.reinvited}}
{{i18n user.invited.reinvited}}
{{else}}
<button class='btn' {{action "reinvite" invite}}><i class="fa fa-envelope"></i>{{i18n user.invited.reinvite}}</button>
{{/if}}
</td> </td>
{{/if}} {{/if}}
</tr> </tr>

View File

@ -3,7 +3,7 @@ class InvitesController < ApplicationController
skip_before_filter :check_xhr skip_before_filter :check_xhr
skip_before_filter :redirect_to_login_if_required skip_before_filter :redirect_to_login_if_required
before_filter :ensure_logged_in, only: [:destroy, :create, :check_csv_chunk, :upload_csv_chunk] before_filter :ensure_logged_in, only: [:destroy, :create, :resend_invite, :check_csv_chunk, :upload_csv_chunk]
before_filter :ensure_new_registrations_allowed, only: [:show, :redeem_disposable_invite] before_filter :ensure_new_registrations_allowed, only: [:show, :redeem_disposable_invite]
def show def show
@ -96,6 +96,16 @@ class InvitesController < ApplicationController
render nothing: true render nothing: true
end end
def resend_invite
params.require(:email)
invite = Invite.find_by(invited_by_id: current_user.id, email: params[:email])
raise Discourse::InvalidParameters.new(:email) if invite.blank?
invite.resend_invite
render nothing: true
end
def check_csv_chunk def check_csv_chunk
guardian.ensure_can_bulk_invite_to_forum!(current_user) guardian.ensure_can_bulk_invite_to_forum!(current_user)

View File

@ -95,7 +95,6 @@ class Invite < ActiveRecord::Base
invite invite
end end
# generate invite tokens without email # generate invite tokens without email
def self.generate_disposable_tokens(invited_by, quantity=nil, group_names=nil) def self.generate_disposable_tokens(invited_by, quantity=nil, group_names=nil)
invite_tokens = [] invite_tokens = []
@ -179,6 +178,11 @@ class Invite < ActiveRecord::Base
user user
end end
def resend_invite
self.update_columns(created_at: Time.zone.now, updated_at: Time.zone.now)
Jobs.enqueue(:invite_email, invite_id: self.id)
end
def self.base_directory def self.base_directory
File.join(Rails.root, "public", "csv", RailsMultisite::ConnectionManagement.current_db) File.join(Rails.root, "public", "csv", RailsMultisite::ConnectionManagement.current_db)
end end

View File

@ -474,6 +474,8 @@ en:
expired: "This invite has expired." expired: "This invite has expired."
rescind: "Remove" rescind: "Remove"
rescinded: "Invite removed" rescinded: "Invite removed"
reinvite: "Resend Invite"
reinvited: "Invite re-sent"
time_read: "Read Time" time_read: "Read Time"
days_visited: "Days Visited" days_visited: "Days Visited"
account_age_days: "Account age in days" account_age_days: "Account age in days"

View File

@ -413,6 +413,7 @@ Discourse::Application.routes.draw do
post "upload" => "invites#upload_csv_chunk" post "upload" => "invites#upload_csv_chunk"
end end
end end
post "invites/reinvite" => "invites#resend_invite"
post "invites/disposable" => "invites#create_disposable_invite" post "invites/disposable" => "invites#create_disposable_invite"
get "invites/redeem/:token" => "invites#redeem_disposable_invite" get "invites/redeem/:token" => "invites#redeem_disposable_invite"
delete "invites" => "invites#destroy" delete "invites" => "invites#destroy"

View File

@ -280,6 +280,40 @@ describe InvitesController do
end end
context '.resend_invite' do
it 'requires you to be logged in' do
lambda {
delete :resend_invite, email: 'first_name@example.com'
}.should raise_error(Discourse::NotLoggedIn)
end
context 'while logged in' do
let!(:user) { log_in }
let!(:invite) { Fabricate(:invite, invited_by: user) }
let(:another_invite) { Fabricate(:invite, email: 'last_name@example.com') }
it 'raises an error when the email is missing' do
lambda { post :resend_invite }.should raise_error(ActionController::ParameterMissing)
end
it "raises an error when the email cannot be found" do
lambda { post :resend_invite, email: 'first_name@example.com' }.should raise_error(Discourse::InvalidParameters)
end
it 'raises an error when the invite is not yours' do
lambda { post :resend_invite, email: another_invite.email }.should raise_error(Discourse::InvalidParameters)
end
it "resends the invite" do
Invite.any_instance.expects(:resend_invite)
post :resend_invite, email: invite.email
end
end
end
context '.check_csv_chunk' do context '.check_csv_chunk' do
it 'requires you to be logged in' do it 'requires you to be logged in' do
lambda { lambda {