2019-05-03 06:17:27 +08:00
# frozen_string_literal: true
2022-11-14 10:02:06 +08:00
# NOTE: There are a _lot_ of complicated rules and conditions for our
# invite system, and the code is spread out through a lot of places.
# Tread lightly and read carefully when modifying this code. You may
# also want to look at:
#
# * InvitesController
# * SessionController
# * Invite model
# * User model
#
# Invites that are scoped to a specific email (email IS NOT NULL on the Invite
# model) have different rules to invites that are considered an "invite link",
# (email IS NULL) on the Invite model.
2022-11-02 00:33:32 +08:00
class InviteRedeemer
attr_reader :invite ,
:email ,
:username ,
:name ,
:password ,
:user_custom_fields ,
:ip_address ,
:session ,
:email_token ,
:redeeming_user
def initialize (
2022-11-14 10:02:06 +08:00
invite : ,
2022-11-02 00:33:32 +08:00
email : nil ,
username : nil ,
name : nil ,
password : nil ,
user_custom_fields : nil ,
ip_address : nil ,
session : nil ,
email_token : nil ,
redeeming_user : nil
)
@invite = invite
@username = username
@name = name
@password = password
@user_custom_fields = user_custom_fields
@ip_address = ip_address
@session = session
@email_token = email_token
@redeeming_user = redeeming_user
2022-11-14 10:02:06 +08:00
ensure_email_is_present! ( email )
2022-11-02 00:33:32 +08:00
end
2013-05-30 23:33:11 +08:00
def redeem
Invite . transaction do
2022-06-21 11:56:50 +08:00
if can_redeem_invite? && mark_invite_redeemed
2014-07-02 00:52:52 +08:00
process_invitation
2021-04-21 17:36:32 +08:00
invited_user
2014-07-02 00:52:52 +08:00
end
2013-05-30 23:33:11 +08:00
end
end
2022-11-14 10:02:06 +08:00
# The email must be present in some form since many of the methods
# for processing + redemption rely on it. If it's still nil after
# these checks then we have hit an edge case and should not proceed!
def ensure_email_is_present! ( email )
if email . blank?
Rails . logger . warn (
" email param was blank in InviteRedeemer for invite ID #{ @invite . id } . The `redeeming_user` was #{ @redeeming_user . present? ? " (ID: #{ @redeeming_user . id } ) " : " not " } present. " ,
)
end
if email . blank? && @invite . is_email_invite?
@email = @invite . email
elsif @redeeming_user . present?
@email = @redeeming_user . email
else
@email = email
end
raise Discourse :: InvalidParameters if @email . blank?
end
# This will _never_ be called if there is a redeeming_user being passed
# in to InviteRedeemer -- see invited_user below.
2021-04-14 17:15:56 +08:00
def self . create_user_from_invite (
email : ,
invite : ,
username : nil ,
name : nil ,
password : nil ,
user_custom_fields : nil ,
ip_address : nil ,
session : nil ,
email_token : nil
2021-07-12 05:57:38 +08:00
)
if username && UsernameValidator . new ( username ) . valid_format? &&
User . username_available? ( username , email )
2014-07-14 23:56:26 +08:00
available_username = username
else
2020-06-09 23:19:32 +08:00
available_username = UserNameSuggester . suggest ( email )
2014-07-14 23:56:26 +08:00
end
2013-08-28 15:18:31 +08:00
2021-07-12 05:57:38 +08:00
user = User . where ( staged : true ) . with_email ( email . strip . downcase ) . first
user . unstage! if user
user || = User . new
2020-03-17 23:48:24 +08:00
user . attributes = {
2020-06-09 23:19:32 +08:00
email : email ,
2018-01-19 22:29:15 +08:00
username : available_username ,
2020-03-17 23:48:24 +08:00
name : name || available_username ,
2018-12-11 06:24:02 +08:00
active : false ,
2019-04-13 15:34:25 +08:00
trust_level : SiteSetting . default_invitee_trust_level ,
ip_address : ip_address ,
registration_ip_address : ip_address ,
2018-01-19 22:29:15 +08:00
}
2022-06-03 11:43:52 +08:00
if ( ! SiteSetting . must_approve_users && SiteSetting . invite_only ) ||
( SiteSetting . must_approve_users? && EmailValidator . can_auto_approve_user? ( user . email ) )
2022-06-02 22:10:48 +08:00
ReviewableUser . set_approved_fields! ( user , Discourse . system_user )
2017-03-05 21:55:21 +08:00
end
2017-06-09 03:10:43 +08:00
user_fields = UserField . all
if user_custom_fields . present? && user_fields . present?
field_params = user_custom_fields || { }
fields = user . custom_fields
user_fields . each do | f |
2024-03-27 21:12:14 +08:00
field_params [ f . id . to_s ] = nil if field_params [ f . id . to_s ] === " false "
2017-06-09 03:10:43 +08:00
field_val = field_params [ f . id . to_s ]
2018-09-07 06:02:47 +08:00
fields [ " #{ User :: USER_FIELD_PREFIX } #{ f . id } " ] = field_val [
0 ... UserField . max_length
2024-05-27 18:27:13 +08:00
] if field_val . present?
2017-06-09 03:10:43 +08:00
end
user . custom_fields = fields
end
2016-09-21 01:12:00 +08:00
user . moderator = true if invite . moderator? && invite . invited_by . staff?
2013-08-28 15:18:31 +08:00
2017-11-24 00:39:24 +08:00
if password
user . password = password
user . password_required!
end
2021-03-02 15:13:04 +08:00
authenticator = UserAuthenticator . new ( user , session , require_password : false )
if ! authenticator . has_authenticator? && ! SiteSetting . enable_local_logins
raise ActiveRecord :: RecordNotSaved . new ( I18n . t ( " login.incorrect_username_email_or_password " ) )
end
authenticator . start
if authenticator . email_valid? && ! authenticator . authenticated?
raise ActiveRecord :: RecordNotSaved . new ( I18n . t ( " login.incorrect_username_email_or_password " ) )
end
2017-11-24 00:39:24 +08:00
user . save!
2021-03-02 15:13:04 +08:00
authenticator . finish
2018-12-11 06:24:02 +08:00
2022-11-14 10:02:06 +08:00
if invite . emailed_status != Invite . emailed_status_types [ :not_required ] &&
email == invite . email && invite . email_token . present? && email_token == invite . email_token
2018-12-11 06:24:02 +08:00
user . activate
end
2017-11-24 00:39:24 +08:00
User . find ( user . id )
2013-08-28 15:18:31 +08:00
end
2013-05-30 23:33:11 +08:00
private
2022-06-21 11:56:50 +08:00
def can_redeem_invite?
2022-11-02 00:33:32 +08:00
return false if ! invite . redeemable?
2022-11-14 10:02:06 +08:00
return false if email . blank?
2022-08-05 10:20:48 +08:00
2022-11-14 10:02:06 +08:00
# Invite scoped to email has already been redeemed by anyone.
return false if invite . is_email_invite? && InvitedUser . exists? ( invite_id : invite . id )
2022-06-21 11:56:50 +08:00
2022-11-14 10:02:06 +08:00
# The email will be present for either an invite link (where the user provides
# us the email manually) or for an invite scoped to an email, where we
# prefill the email and do not let the user modify it.
#
# Note that an invite link can also have a domain scope which must be checked.
email_to_check = redeeming_user & . email || email
2022-06-21 11:56:50 +08:00
2022-11-14 10:02:06 +08:00
if invite . email . present? && ! invite . email_matches? ( email_to_check )
raise ActiveRecord :: RecordNotSaved . new ( I18n . t ( " invite.not_matching_email " ) )
end
2022-06-21 11:56:50 +08:00
2022-11-14 10:02:06 +08:00
if invite . domain . present? && ! invite . domain_matches? ( email_to_check )
raise ActiveRecord :: RecordNotSaved . new ( I18n . t ( " invite.domain_not_allowed " ) )
2022-11-02 00:33:32 +08:00
end
# Anon user is trying to redeem an invitation, if an existing user already
# redeemed it then we cannot redeem now.
redeeming_user || = User . where ( admin : false , staged : false ) . find_by_email ( email )
if redeeming_user . present? &&
InvitedUser . exists? ( user_id : redeeming_user . id , invite_id : invite . id )
2024-02-27 18:24:20 +08:00
raise Invite :: UserExists . new ( I18n . t ( " invite.existing_user_already_redemeed " ) )
2022-06-21 11:56:50 +08:00
end
true
end
2022-11-14 10:02:06 +08:00
# Note that the invited_user is returned by #redeemed, so other places
# (e.g. the InvitesController) can perform further actions on it, this
# is why things like send_welcome_message are set without being saved
# on the model.
2022-11-02 00:33:32 +08:00
def invited_user
return @invited_user if defined? ( @invited_user )
# The redeeming user is an already logged in user or a user who is
# activating their account who is redeeming the invite,
# which is valid for existing users to be invited to topics or groups.
if redeeming_user . present?
@invited_user = redeeming_user
return @invited_user
2022-06-21 11:56:50 +08:00
end
2022-11-02 00:33:32 +08:00
# If there was no logged in user then we must attempt to create
# one based on the provided params.
invited_user || =
InviteRedeemer . create_user_from_invite (
email : email ,
invite : invite ,
username : username ,
name : name ,
password : password ,
user_custom_fields : user_custom_fields ,
ip_address : ip_address ,
session : session ,
email_token : email_token ,
)
invited_user . send_welcome_message = false
@invited_user = invited_user
@invited_user
2013-05-30 23:33:11 +08:00
end
def process_invitation
add_to_private_topics_if_invited
2014-05-08 14:45:49 +08:00
add_user_to_groups
2013-05-30 23:33:11 +08:00
send_welcome_message
notify_invitee
end
def mark_invite_redeemed
2020-06-09 23:19:32 +08:00
@invited_user_record = InvitedUser . create! ( invite_id : invite . id , redeemed_at : Time . zone . now )
2022-06-21 11:56:50 +08:00
2021-01-20 16:50:02 +08:00
if @invited_user_record . present?
2023-07-28 19:53:48 +08:00
invite . with_lock ( " FOR UPDATE NOWAIT " ) do
Invite . increment_counter ( :redemption_count , invite . id )
invite . save!
end
2019-05-13 15:42:13 +08:00
delete_duplicate_invites
end
2020-06-09 23:19:32 +08:00
@invited_user_record . present?
2013-05-30 23:33:11 +08:00
end
def add_to_private_topics_if_invited
2022-11-14 10:02:06 +08:00
# Should not happen because of ensure_email_is_present!, but better to cover bases.
return if email . blank?
topic_ids =
TopicInvite
. joins ( :invite )
. joins ( :topic )
. where ( " topics.archetype = ? " , Archetype . private_message )
. where ( " invites.email = ? " , email )
. pluck ( :topic_id )
2019-03-30 00:03:33 +08:00
topic_ids . each do | id |
2022-11-14 10:02:06 +08:00
if ! TopicAllowedUser . exists? ( user_id : invited_user . id , topic_id : id )
TopicAllowedUser . create! ( user_id : invited_user . id , topic_id : id )
end
2013-05-30 23:33:11 +08:00
end
end
2014-05-08 14:45:49 +08:00
def add_user_to_groups
2020-06-15 17:13:56 +08:00
guardian = Guardian . new ( invite . invited_by )
2015-02-26 11:05:44 +08:00
new_group_ids = invite . groups . pluck ( :id ) - invited_user . group_users . pluck ( :group_id )
new_group_ids . each do | id |
2020-06-15 17:13:56 +08:00
group = Group . find_by ( id : id )
if guardian . can_edit_group? ( group )
invited_user . group_users . create! ( group_id : group . id )
2022-11-03 06:58:12 +08:00
GroupActionLogger . new ( invite . invited_by , group ) . log_add_user_to_group ( invited_user )
2020-06-15 17:13:56 +08:00
DiscourseEvent . trigger ( :user_added_to_group , invited_user , group , automatic : false )
end
2014-05-08 14:45:49 +08:00
end
end
2013-05-30 23:33:11 +08:00
def send_welcome_message
2020-06-09 23:19:32 +08:00
@invited_user_record . update! ( user_id : invited_user . id )
invited_user . send_welcome_message = true
2013-05-30 23:33:11 +08:00
end
def notify_invitee
2022-11-14 10:02:06 +08:00
return if invite . invited_by . blank?
invite . invited_by . notifications . create! (
notification_type : Notification . types [ :invitee_accepted ] ,
data : { display_username : invited_user . username } . to_json ,
)
2013-05-30 23:33:11 +08:00
end
2015-03-26 00:55:18 +08:00
def delete_duplicate_invites
2022-11-14 10:02:06 +08:00
# Should not happen because of ensure_email_is_present!, but better to cover bases.
return if email . blank?
2021-03-03 17:45:29 +08:00
Invite
. where ( " invites.max_redemptions_allowed = 1 " )
2020-06-09 23:19:32 +08:00
. joins ( " LEFT JOIN invited_users ON invites.id = invited_users.invite_id " )
. where ( " invited_users.user_id IS NULL " )
. where ( " invites.email = ? AND invites.id != ? " , email , invite . id )
. delete_all
2015-03-26 00:55:18 +08:00
end
2013-07-11 09:21:39 +08:00
end