2019-05-02 18:17:27 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2014-05-27 16:14:37 -04:00
|
|
|
module Jobs
|
2019-10-02 00:01:53 -04:00
|
|
|
class BulkInvite < ::Jobs::Base
|
2014-05-27 16:14:37 -04:00
|
|
|
sidekiq_options retry: false
|
|
|
|
|
|
|
|
def initialize
|
2018-08-30 03:34:58 -04:00
|
|
|
super
|
2021-03-29 07:03:19 -04:00
|
|
|
|
|
|
|
@logs = []
|
|
|
|
@sent = 0
|
2022-10-04 11:41:06 -04:00
|
|
|
@skipped = 0
|
2024-05-17 14:21:21 -04:00
|
|
|
@skipped_emails = []
|
2022-10-04 11:41:06 -04:00
|
|
|
@warnings = 0
|
2021-03-29 07:03:19 -04:00
|
|
|
@failed = 0
|
2024-05-17 14:21:21 -04:00
|
|
|
@failed_emails = []
|
2021-03-29 07:03:19 -04:00
|
|
|
@groups = {}
|
|
|
|
@user_fields = {}
|
2019-04-15 21:31:51 -04:00
|
|
|
@valid_groups = {}
|
2014-05-27 16:14:37 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def execute(args)
|
2021-03-29 07:03:19 -04:00
|
|
|
@invites = args[:invites]
|
|
|
|
raise Discourse::InvalidParameters.new(:invites) if @invites.blank?
|
2019-04-15 09:13:53 -04:00
|
|
|
|
|
|
|
@current_user = User.find_by(id: args[:current_user_id])
|
|
|
|
raise Discourse::InvalidParameters.new(:current_user_id) unless @current_user
|
2021-03-29 07:03:19 -04:00
|
|
|
|
2024-03-29 13:22:00 -04:00
|
|
|
@skip_email = SiteSetting.skip_email_bulk_invites
|
|
|
|
|
2019-04-15 21:31:51 -04:00
|
|
|
@guardian = Guardian.new(@current_user)
|
2019-04-15 09:13:53 -04:00
|
|
|
|
2021-03-29 07:03:19 -04:00
|
|
|
process_invites(@invites)
|
2019-07-19 01:59:12 -04:00
|
|
|
|
2021-03-29 07:03:19 -04:00
|
|
|
if @invites.length > Invite::BULK_INVITE_EMAIL_LIMIT
|
2019-10-21 13:25:35 -04:00
|
|
|
::Jobs.enqueue(:process_bulk_invite_emails)
|
2019-07-19 01:59:12 -04:00
|
|
|
end
|
2016-10-17 15:00:29 -04:00
|
|
|
ensure
|
2014-06-25 16:04:26 -04:00
|
|
|
notify_user
|
2014-06-25 14:35:11 -04:00
|
|
|
end
|
|
|
|
|
2019-04-15 09:13:53 -04:00
|
|
|
private
|
|
|
|
|
2019-06-04 10:49:46 -04:00
|
|
|
def process_invites(invites)
|
|
|
|
invites.each do |invite|
|
2022-02-17 20:12:51 -05:00
|
|
|
if EmailAddressValidator.valid_value?(invite[:email])
|
2019-06-04 10:49:46 -04:00
|
|
|
# email is valid
|
2022-10-04 11:41:06 -04:00
|
|
|
result = send_invite(invite)
|
|
|
|
if Invite === result
|
|
|
|
@sent += 1
|
|
|
|
elsif User === result
|
|
|
|
@skipped += 1
|
2024-05-17 14:21:21 -04:00
|
|
|
@skipped_emails << invite[:email]
|
2022-10-04 11:41:06 -04:00
|
|
|
else
|
|
|
|
@failed += 1
|
2024-05-17 14:21:21 -04:00
|
|
|
@failed_emails << invite[:email]
|
2022-10-04 11:41:06 -04:00
|
|
|
end
|
2019-06-04 10:49:46 -04:00
|
|
|
else
|
|
|
|
# invalid email
|
|
|
|
save_log "Invalid Email '#{invite[:email]}"
|
|
|
|
@failed += 1
|
2024-05-17 14:21:21 -04:00
|
|
|
@failed_emails << invite[:email]
|
2014-05-27 16:14:37 -04:00
|
|
|
end
|
|
|
|
end
|
2016-10-17 15:00:29 -04:00
|
|
|
rescue Exception => e
|
2018-08-30 03:34:58 -04:00
|
|
|
save_log "Bulk Invite Process Failed -- '#{e.message}'"
|
2016-10-17 15:00:29 -04:00
|
|
|
@failed += 1
|
2024-05-17 14:21:21 -04:00
|
|
|
@failed_emails << invite[:email]
|
2014-05-27 16:14:37 -04:00
|
|
|
end
|
|
|
|
|
2024-05-17 14:21:21 -04:00
|
|
|
def get_groups(group_names, email)
|
2019-04-15 09:26:03 -04:00
|
|
|
groups = []
|
2019-04-15 09:13:53 -04:00
|
|
|
|
2014-06-26 13:19:34 -04:00
|
|
|
if group_names
|
2014-06-26 11:16:53 -04:00
|
|
|
group_names = group_names.split(";")
|
2019-04-15 09:13:53 -04:00
|
|
|
|
2022-10-04 11:41:06 -04:00
|
|
|
group_names.each do |group_name|
|
2019-04-15 21:31:51 -04:00
|
|
|
group = fetch_group(group_name)
|
2019-04-15 09:13:53 -04:00
|
|
|
|
2019-04-15 21:31:51 -04:00
|
|
|
if group && can_edit_group?(group)
|
2014-06-26 11:16:53 -04:00
|
|
|
# valid group
|
2019-04-15 21:31:51 -04:00
|
|
|
groups.push(group)
|
2014-06-26 11:16:53 -04:00
|
|
|
else
|
|
|
|
# invalid group
|
2024-05-17 14:21:21 -04:00
|
|
|
save_log "Invalid Group '#{group_name}' for '#{email}'"
|
2022-10-04 11:41:06 -04:00
|
|
|
@warnings += 1
|
2014-06-26 11:16:53 -04:00
|
|
|
end
|
2022-10-04 11:41:06 -04:00
|
|
|
end
|
2014-06-26 11:16:53 -04:00
|
|
|
end
|
2019-04-15 09:26:03 -04:00
|
|
|
|
|
|
|
groups
|
2014-05-27 16:14:37 -04:00
|
|
|
end
|
|
|
|
|
2024-05-17 14:21:21 -04:00
|
|
|
def get_topic(topic_id, email)
|
2014-06-26 11:16:53 -04:00
|
|
|
topic = nil
|
2019-04-15 09:26:03 -04:00
|
|
|
|
2014-06-26 13:19:34 -04:00
|
|
|
if topic_id
|
2014-06-26 11:16:53 -04:00
|
|
|
topic = Topic.find_by_id(topic_id)
|
|
|
|
if topic.nil?
|
2024-05-17 14:21:21 -04:00
|
|
|
save_log "Invalid Topic ID '#{topic_id}' for '#{email}'"
|
2022-10-04 11:41:06 -04:00
|
|
|
@warnings += 1
|
2014-06-25 14:35:11 -04:00
|
|
|
end
|
2014-06-26 11:16:53 -04:00
|
|
|
end
|
2019-04-15 09:26:03 -04:00
|
|
|
|
2019-11-14 15:10:51 -05:00
|
|
|
topic
|
2014-06-25 14:35:11 -04:00
|
|
|
end
|
|
|
|
|
2024-05-17 14:21:21 -04:00
|
|
|
def get_user_fields(fields, email)
|
2021-03-29 07:03:19 -04:00
|
|
|
user_fields = {}
|
|
|
|
|
|
|
|
fields.each do |key, value|
|
2021-03-31 12:19:57 -04:00
|
|
|
@user_fields[key] ||= UserField
|
|
|
|
.includes(:user_field_options)
|
|
|
|
.where("name ILIKE ?", key)
|
|
|
|
.first || :nil
|
2022-10-04 11:41:06 -04:00
|
|
|
if @user_fields[key] == :nil
|
2024-05-17 14:21:21 -04:00
|
|
|
save_log "Invalid User Field '#{key}' for '#{email}'"
|
2022-10-04 11:41:06 -04:00
|
|
|
@warnings += 1
|
|
|
|
next
|
|
|
|
end
|
2021-03-31 12:19:57 -04:00
|
|
|
|
|
|
|
# Automatically correct user field value
|
|
|
|
if @user_fields[key].field_type == "dropdown"
|
|
|
|
value =
|
|
|
|
@user_fields[key].user_field_options.find { |ufo| ufo.value.casecmp?(value) }&.value
|
|
|
|
end
|
|
|
|
|
|
|
|
user_fields[@user_fields[key].id] = value
|
2021-03-29 07:03:19 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
user_fields
|
|
|
|
end
|
|
|
|
|
2019-06-04 10:49:46 -04:00
|
|
|
def send_invite(invite)
|
|
|
|
email = invite[:email]
|
2024-05-17 14:21:21 -04:00
|
|
|
groups = get_groups(invite[:groups], email)
|
|
|
|
topic = get_topic(invite[:topic_id], email)
|
2021-12-05 20:08:21 -05:00
|
|
|
locale = invite[:locale]
|
2024-05-17 14:21:21 -04:00
|
|
|
user_fields = get_user_fields(invite.except(:email, :groups, :topic_id, :locale), email)
|
2019-04-15 09:26:03 -04:00
|
|
|
|
2014-10-09 10:44:15 -04:00
|
|
|
begin
|
2019-10-30 02:08:47 -04:00
|
|
|
if user = Invite.find_user_by_email(email)
|
2019-04-15 09:26:03 -04:00
|
|
|
if groups.present?
|
|
|
|
Group.transaction do
|
|
|
|
groups.each do |group|
|
|
|
|
group.add(user)
|
|
|
|
|
|
|
|
GroupActionLogger.new(@current_user, group).log_add_user_to_group(user)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2021-03-29 07:03:19 -04:00
|
|
|
|
|
|
|
if user_fields.present?
|
|
|
|
user_fields.each { |user_field, value| user.set_user_field(user_field, value) }
|
|
|
|
user.save_custom_fields
|
|
|
|
end
|
2021-12-05 20:08:21 -05:00
|
|
|
|
|
|
|
if locale.present?
|
|
|
|
user.locale = locale
|
|
|
|
user.save!
|
|
|
|
end
|
2022-10-04 11:41:06 -04:00
|
|
|
|
|
|
|
user
|
2019-04-15 09:26:03 -04:00
|
|
|
else
|
2021-12-05 20:08:21 -05:00
|
|
|
if user_fields.present? || locale.present?
|
2021-03-29 07:03:19 -04:00
|
|
|
user = User.where(staged: true).find_by_email(email)
|
|
|
|
user ||=
|
|
|
|
User.new(username: UserNameSuggester.suggest(email), email: email, staged: true)
|
2021-12-05 20:08:21 -05:00
|
|
|
|
|
|
|
if user_fields.present?
|
|
|
|
user_fields.each { |user_field, value| user.set_user_field(user_field, value) }
|
2021-03-29 07:03:19 -04:00
|
|
|
end
|
2021-12-05 20:08:21 -05:00
|
|
|
|
|
|
|
user.locale = locale if locale.present?
|
|
|
|
|
2021-03-29 07:03:19 -04:00
|
|
|
user.save!
|
2019-07-19 01:59:12 -04:00
|
|
|
end
|
2021-03-29 07:03:19 -04:00
|
|
|
|
2024-03-29 13:22:00 -04:00
|
|
|
invite_opts = {
|
|
|
|
email: email,
|
|
|
|
topic: topic,
|
|
|
|
group_ids: groups.map(&:id),
|
|
|
|
skip_email: @skip_email,
|
|
|
|
}
|
2021-03-29 07:03:19 -04:00
|
|
|
|
|
|
|
if @invites.length > Invite::BULK_INVITE_EMAIL_LIMIT
|
|
|
|
invite_opts[:emailed_status] = Invite.emailed_status_types[:bulk_pending]
|
|
|
|
end
|
|
|
|
|
|
|
|
Invite.generate(@current_user, invite_opts)
|
2019-04-15 09:26:03 -04:00
|
|
|
end
|
2014-10-09 10:44:15 -04:00
|
|
|
rescue => e
|
2018-08-30 03:34:58 -04:00
|
|
|
save_log "Error inviting '#{email}' -- #{Rails::Html::FullSanitizer.new.sanitize(e.message)}"
|
2022-10-04 11:41:06 -04:00
|
|
|
|
|
|
|
nil
|
2014-10-09 10:44:15 -04:00
|
|
|
end
|
2014-06-25 14:35:11 -04:00
|
|
|
end
|
|
|
|
|
2014-05-27 16:14:37 -04:00
|
|
|
def save_log(message)
|
|
|
|
@logs << "[#{Time.now}] #{message}"
|
|
|
|
end
|
|
|
|
|
2014-06-25 16:04:26 -04:00
|
|
|
def notify_user
|
|
|
|
if @current_user
|
2022-10-04 11:41:06 -04:00
|
|
|
if @sent > 0 && @failed == 0
|
2018-08-30 03:34:58 -04:00
|
|
|
SystemMessage.create_from_system_user(
|
|
|
|
@current_user,
|
|
|
|
:bulk_invite_succeeded,
|
2022-10-04 11:41:06 -04:00
|
|
|
sent: @sent,
|
|
|
|
skipped: @skipped,
|
2024-05-17 14:21:21 -04:00
|
|
|
skipped_emails: @skipped_emails.join("\n"),
|
2022-10-04 11:41:06 -04:00
|
|
|
warnings: @warnings,
|
|
|
|
logs: @logs.join("\n"),
|
2018-08-30 03:34:58 -04:00
|
|
|
)
|
2014-05-27 16:14:37 -04:00
|
|
|
else
|
2018-08-30 03:34:58 -04:00
|
|
|
SystemMessage.create_from_system_user(
|
|
|
|
@current_user,
|
|
|
|
:bulk_invite_failed,
|
|
|
|
sent: @sent,
|
2022-10-04 11:41:06 -04:00
|
|
|
skipped: @skipped,
|
2024-05-17 14:21:21 -04:00
|
|
|
skipped_emails: @skipped_emails.join("\n"),
|
2022-10-04 11:41:06 -04:00
|
|
|
warnings: @warnings,
|
2018-08-30 03:34:58 -04:00
|
|
|
failed: @failed,
|
2024-05-17 14:21:21 -04:00
|
|
|
failed_emails: @failed_emails.join("\n"),
|
2018-08-30 03:34:58 -04:00
|
|
|
logs: @logs.join("\n"),
|
|
|
|
)
|
2014-05-27 16:14:37 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-04-15 21:31:51 -04:00
|
|
|
def fetch_group(group_name)
|
|
|
|
group_name = group_name.downcase
|
|
|
|
group = @groups[group_name]
|
|
|
|
|
|
|
|
unless group
|
2021-03-29 07:03:19 -04:00
|
|
|
group = Group.find_by("lower(name) = ?", group_name)
|
2019-04-15 21:31:51 -04:00
|
|
|
@groups[group_name] = group
|
|
|
|
end
|
|
|
|
|
|
|
|
group
|
|
|
|
end
|
|
|
|
|
|
|
|
def can_edit_group?(group)
|
|
|
|
group_name = group.name.downcase
|
|
|
|
result = @valid_groups[group_name]
|
|
|
|
|
|
|
|
unless result
|
|
|
|
result = @guardian.can_edit_group?(group)
|
|
|
|
@valid_groups[group_name] = result
|
|
|
|
end
|
|
|
|
|
|
|
|
result
|
|
|
|
end
|
2014-05-27 16:14:37 -04:00
|
|
|
end
|
|
|
|
end
|