# Copyright (C) 2012-2014 Zammad Foundation, http://zammad-foundation.org/ require 'digest/md5' # @model User # # @property id(required) [Integer] The identifier for the User. # @property login(required) [String] The login of the User used for authentication. # @property firstname [String] The firstname of the User. # @property lastname [String] The lastname of the User. # @property email [String] The email of the User. # @property image [String] The Image used as the User avatar (TODO: Image model?). # @property web [String] The website/URL of the User. # @property password [String] The password of the User. # @property phone [String] The phone number of the User. # @property fax [String] The fax number of the User. # @property mobile [String] The mobile number of the User. # @property department [String] The department the User is working at. # @property street [String] The street the User lives in. # @property zip [Integer] The zip postal code of the User city. # @property city [String] The city the User lives in. # @property country [String] The country the User lives in. # @property verified [Boolean] The flag that shows the verified state of the User. # @property active [Boolean] The flag that shows the active state of the User. # @property note [String] The note or comment stored to the User. class User < ApplicationModel include User::Permission load 'user/assets.rb' include User::Assets extend User::Search before_create :check_name, :check_email, :check_login, :check_password, :check_preferences_default before_update :check_password, :check_email, :check_login, :check_preferences_default after_create :avatar_for_email_check after_update :avatar_for_email_check after_destroy :avatar_destroy notify_clients_support has_and_belongs_to_many :groups, after_add: :cache_update, after_remove: :cache_update has_and_belongs_to_many :roles, after_add: [:cache_update, :check_notifications], after_remove: :cache_update has_and_belongs_to_many :organizations, after_add: :cache_update, after_remove: :cache_update has_many :tokens, after_add: :cache_update, after_remove: :cache_update has_many :authorizations, after_add: :cache_update, after_remove: :cache_update belongs_to :organization, class_name: 'Organization' store :preferences activity_stream_support( role: Z_ROLENAME_ADMIN, ignore_attributes: { last_login: true, image: true, image_source: true, preferences: true, } ) history_support( ignore_attributes: { password: true, image: true, image_source: true, preferences: true, } ) search_index_support( ignore_attributes: { password: true, image: true, image_source: true, source: true, login_failed: true, preferences: true, } ) =begin fullname of user user = User.find(123) result = user.fulename returns result = "Bob Smith" =end def fullname fullname = '' if firstname && !firstname.empty? fullname = fullname + firstname end if lastname && !lastname.empty? if fullname != '' fullname = fullname + ' ' end fullname = fullname + lastname end if fullname == '' && email fullname = email end fullname end =begin check if user is in role user = User.find(123) result = user.role?('Customer') result = user.role?(['Agent', 'Admin']) returns result = true|false =end def role?(role_name) result = false roles.each { |role| if role_name.class == Array next if !role_name.include?(role.name) elsif role.name != role_name next end result = true break } result end =begin get users activity stream user = User.find(123) result = user.activity_stream(20) returns result = [ { :id => 2, :o_id => 2, :created_by_id => 3, :created_at => '2013-09-28 00:57:21', :object => "User", :type => "created", }, { :id => 2, :o_id => 2, :created_by_id => 3, :created_at => '2013-09-28 00:59:21', :object => "User", :type => "updated", }, ] =end def activity_stream(limit, fulldata = false) activity_stream = ActivityStream.list(self, limit) return activity_stream if !fulldata # get related objects assets = ApplicationModel.assets_of_object_list(activity_stream) { activity_stream: activity_stream, assets: assets, } end =begin authenticate user result = User.authenticate(username, password) returns result = user_model # user model if authentication was successfully =end def self.authenticate(username, password) # do not authenticate with nothing return if !username || username == '' return if !password || password == '' # try to find user based on login user = User.find_by( login: username.downcase, active: true ) # try second lookup with email if !user user = User.find_by( email: username.downcase, active: true ) end # check failed logins max_login_failed = Setting.get('password_max_login_failed').to_i || 10 if user && user.login_failed > max_login_failed logger.info "Max login faild reached for user #{user.login}." return false end user_auth = Auth.check(username, password, user) # set login failed +1 if !user_auth && user sleep 1 user.login_failed = user.login_failed + 1 user.save end # auth ok user_auth end =begin authenticate user agains sso result = User.sso(sso_params) returns result = user_model # user model if authentication was successfully =end def self.sso(params) # try to login against configure auth backends user_auth = Sso.check(params) return if !user_auth user_auth end =begin create user from from omni auth hash result = User.create_from_hash!(hash) returns result = user_model # user model if create was successfully =end def self.create_from_hash!(hash) roles = Role.where(name: 'Customer') url = '' if hash['info']['urls'] hash['info']['urls'].each {|_name, local_url| next if !local_url next if local_url.empty? url = local_url } end create( login: hash['info']['nickname'] || hash['uid'], firstname: hash['info']['name'], email: hash['info']['email'], image_source: hash['info']['image'], web: url, address: hash['info']['location'], note: hash['info']['description'], source: hash['provider'], roles: roles, updated_by_id: 1, created_by_id: 1, ) end =begin send reset password email with token to user result = User.password_reset_send(username) returns result = token =end def self.password_reset_send(username) return if !username || username == '' # try to find user based on login user = User.find_by(login: username.downcase, active: true) # try second lookup with email if !user user = User.find_by(email: username.downcase, active: true) end # check if email address exists return if !user return if !user.email # generate token token = Token.create(action: 'PasswordReset', user_id: user.id) # send mail data = {} data[:subject] = 'Reset your #{config.product_name} password' data[:body] = 'Forgot your password? We received a request to reset the password for your #{config.product_name} account (#{user.login}). If you want to reset your password, click on the link below (or copy and paste the URL into your browser): #{config.http_type}://#{config.fqdn}/#password_reset_verify/#{token.name} This link takes you to a page where you can change your password. If you don\'t want to reset your password, please ignore this message. Your password will not be reset. Your #{config.product_name} Team' # prepare subject & body [:subject, :body].each { |key| data[key.to_sym] = NotificationFactory.build( locale: user.preferences[:locale], string: data[key.to_sym], objects: { token: token, user: user, } ) } # send notification NotificationFactory.send( recipient: user, subject: data[:subject], body: data[:body] ) token end =begin check reset password token result = User.password_reset_check(token) returns result = user_model # user_model if token was verified =end def self.password_reset_check(token) user = Token.check(action: 'PasswordReset', name: token) # reset login failed if token is valid if user user.login_failed = 0 user.save end user end =begin reset reset password with token and set new password result = User.password_reset_via_token(token,password) returns result = user_model # user_model if token was verified =end def self.password_reset_via_token(token, password) # check token user = Token.check(action: 'PasswordReset', name: token) return if !user # reset password user.update_attributes(password: password) # delete token Token.find_by(action: 'PasswordReset', name: token).destroy user end =begin update last login date and reset login_failed (is automatically done by auth and sso backend) user = User.find(123) result = user.update_last_login returns result = new_user_model =end def update_last_login self.last_login = Time.zone.now # reset login failed self.login_failed = 0 # set updated by user self.updated_by_id = id save end =begin merge two users to one user = User.find(123) result = user.merge(user_id_of_duplicate_user) returns result = new_user_model =end def merge(user_id_of_duplicate_user) # find email addresses and move them to primary user duplicate_user = User.find(user_id_of_duplicate_user) # merge missing attibutes Models.merge('User', id, user_id_of_duplicate_user) true end =begin list of active users in role result = User.of_role('Agent', group_ids) returns result = [user1, user2] =end def self.of_role(role, group_ids = nil) roles_ids = Role.where(active: true, name: role).map(&:id) if !group_ids return User.where(active: true).joins(:users_roles).where('roles_users.role_id IN (?)', roles_ids) end User.where(active: true) .joins(:users_roles) .joins(:users_groups) .where('roles_users.role_id IN (?) AND users_groups.group_ids IN (?)', roles_ids, group_ids) end =begin update/sync default preferences of users in a dedecated role result = User.update_default_preferences('Agent') returns result = true # false =end def self.update_default_preferences(role_name) role = Role.lookup(name: role_name) User.of_role(role_name).each {|user| user.check_notifications(role) user.check_preferences_default user.save } true end def check_notifications(o) default = Rails.configuration.preferences_default_by_role return if !default default.deep_stringify_keys! return if !default[o.name] if !@preferences_default @preferences_default = {} end default[o.name].each {|key, value| next if @preferences_default[key] @preferences_default[key] = value } end def check_preferences_default return if !@preferences_default return if @preferences_default.empty? preferences_tmp = @preferences_default.merge(preferences) self.preferences = preferences_tmp end private def cache_delete super # delete asset caches key = "User::authorizations::#{id}" Cache.delete(key) key = "User::role_ids::#{id}" Cache.delete(key) key = "User::group_ids::#{id}" Cache.delete(key) key = "User::organization_ids::#{id}" Cache.delete(key) end def check_name if (firstname && !firstname.empty?) && (!lastname || lastname.empty?) # Lastname, Firstname scan = firstname.scan(/, /) if scan[0] name = firstname.split(', ', 2) if !name[0].nil? self.lastname = name[0] end if !name[1].nil? self.firstname = name[1] end return end # Firstname Lastname name = firstname.split(' ', 2) if !name[0].nil? self.firstname = name[0] end if !name[1].nil? self.lastname = name[1] end return # -no name- firstname.lastname@example.com elsif (!firstname || firstname.empty?) && (!lastname || lastname.empty?) && (email && !email.empty?) scan = email.scan(/^(.+?)\.(.+?)\@.+?$/) if scan[0] if !scan[0][0].nil? self.firstname = scan[0][0].capitalize end if !scan[0][1].nil? self.lastname = scan[0][1].capitalize end end end end def check_email return if !email self.email = email.downcase end def check_login # use email as login if not given if !login && email self.login = email end # if email has changed, login is old email, change also login if changes && changes['email'] if changes['email'][0] == login self.login = email end end # check if login already exists return if !login self.login = login.downcase check = true while check exists = User.find_by(login: login) if exists && exists.id != id self.login = login + rand(999).to_s else check = false end end end def avatar_for_email_check return if !email return if email.empty? return if email !~ /@/ # save/update avatar avatar = Avatar.auto_detection( object: 'User', o_id: id, url: email, source: 'app', updated_by_id: updated_by_id, created_by_id: updated_by_id, ) # update user link return if !avatar update_column(:image, avatar.store_hash) cache_delete end def avatar_destroy Avatar.remove('User', id) end def check_password # set old password again if not given if password == '' || !password # get current record if id #current = User.find(self.id) #self.password = current.password self.password = password_was end end # crypt password if not already crypted return if !password return if password =~ /^\{sha2\}/ crypted = Digest::SHA2.hexdigest(password) self.password = "{sha2}#{crypted}" end end