OpenProject is the leading open source project management software.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 
openproject/app/controllers/concerns/user_invitation.rb

167 lines
4.9 KiB

#-- copyright
# OpenProject is an open source project management software.
# Copyright (C) 2012-2021 the OpenProject GmbH
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License version 3.
#
# OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows:
# Copyright (C) 2006-2013 Jean-Philippe Lang
# Copyright (C) 2010-2013 the ChiliProject Team
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
#
# See docs/COPYRIGHT.rdoc for more details.
#++
module UserInvitation
module Events
class << self
def user_invited
'user_invited'
end
def user_reinvited
'user_reinvited'
end
end
end
module_function
##
# Creates an invited user with the given email address.
#
# @param email E-Mail address the invitation is sent to.
# @param login User's login (optional)
# @param first_name The user's first name (optional)
# @param last_name The user's last name (optional)
#
# @yield [user] Allows modifying the created user before saving it.
#
# @return The invited user. If the invitation failed, calling `#registered?`
# on the returned user will yield `false`. Check for validation errors
# in that case.
def invite_new_user(email:, login: nil, first_name: nil, last_name: nil)
user = User.new mail: email,
login: login,
firstname: first_name,
lastname: last_name,
status: Principal.statuses[:invited]
assign_user_attributes(user)
yield user if block_given?
invite_user! user
end
##
# For the given user with at least the mail attribute set,
# derives login and first name
#
# The default login is the email address.
def assign_user_attributes(user)
placeholder = placeholder_name(user.mail)
user.login = user.login.presence || user.mail
user.firstname = user.firstname.presence || placeholder.first
user.lastname = user.lastname.presence || placeholder.last
user.language = user.language.presence || Setting.default_language
end
##
# Sends a new invitation to the user with a new token.
#
# @param user_id [Integer] ID of the user to be re-invited.
# @return [Token] The new token used for the invitation.
def reinvite_user(user_id)
User.transaction do
clear_tokens user_id
reset_login user_id
Token::Invitation.create!(user_id: user_id).tap do |token|
OpenProject::Notifications.send Events.user_reinvited, token
end
end
end
def clear_tokens(user_id)
Token::Invitation.where(user_id: user_id).delete_all
end
def reset_login(user_id)
User.where(id: user_id).update_all identity_url: nil
UserPassword.where(user_id: user_id).destroy_all
end
##
# Creates a placeholder name for the user based on their email address.
# For the unlikely case that the local or domain part of the email address
# are longer than 30 characters they will be trimmed to 27 characters and an
# ellipsis will be appended.
def placeholder_name(email)
first, last = email.split('@').map { |name| trim_name(name) }
[first, '@' + last]
end
def trim_name(name)
if name.size > 30
name[0..26] + '...'
else
name
end
end
##
# Invites the given user. An email will be sent to their email address
# containing the token necessary for the user to register.
#
# Validates and saves the given user. The invitation will fail if the user is invalid.
#
# @return The invited user or nil if the invitation failed.
def invite_user!(user)
user, token = user_invitation user
if token
OpenProject::Notifications.send(Events.user_invited, token)
user
end
end
##
# Creates an invited user with the given email address.
# If no first and last is given it will default to 'OpenProject User'
# for the first name and 'To-be' for the last name.
# The default login is the email address.
#
# @return Returns the user and the invitation token required to register.
def user_invitation(user)
User.transaction do
user.invite
if user.valid?
token = Token::Invitation.create! user: user
user.save!
return [user, token]
end
end
[user, nil]
end
end