1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- module ApplicationController::HandlesDevices
- extend ActiveSupport::Concern
- included do
- before_action :user_device_log
- end
- def user_device_log(user = current_user, type = 'session')
- switched_from_user_id = ENV['SWITCHED_FROM_USER_ID'] || session[:switched_from_user_id]
- return true if %w[init mobile].include?(params[:controller]) # do no device logging on static initial page
- return true if switched_from_user_id
- return true if current_user_on_behalf # do no device logging for the user on behalf feature
- return true if !user
- return true if !policy(UserDevice).log?
- return true if type == 'SSO'
- time_to_check = true
- user_device_updated_at = session[:user_device_updated_at]
- if ENV['USER_DEVICE_UPDATED_AT']
- user_device_updated_at = Time.zone.parse(ENV['USER_DEVICE_UPDATED_AT'])
- end
- if user_device_updated_at
- # check if entry exists / only if write action
- diff = 10.minutes.ago
- if %w[GET OPTIONS HEAD].include?(request.method)
- diff = 30.minutes.ago
- end
- # only update if needed
- if user_device_updated_at > diff
- time_to_check = false
- end
- end
- # if ip has not changed and ttl in still valid
- remote_ip = ENV['TEST_REMOTE_IP'] || request.remote_ip
- return true if time_to_check == false && session[:user_device_remote_ip] == remote_ip
- session[:user_device_remote_ip] = remote_ip
- # for sessions we need the fingperprint
- if type == 'session'
- fingerprint = params[:fingerprint] || request.headers['X-Browser-Fingerprint']
- if !session[:user_device_updated_at] && !fingerprint && !session[:user_device_fingerprint]
- raise Exceptions::UnprocessableEntity, __('Need fingerprint param!')
- end
- if fingerprint
- UserDevice.fingerprint_validation(fingerprint)
- session[:user_device_fingerprint] = fingerprint
- end
- end
- session[:user_device_updated_at] = Time.zone.now
- # add device if needed
- http_user_agent = ENV['HTTP_USER_AGENT'] || request.env['HTTP_USER_AGENT']
- UserDeviceLogJob.perform_later(
- http_user_agent,
- remote_ip,
- user.id,
- session[:user_device_fingerprint],
- type,
- )
- end
- end
|