summaryrefslogtreecommitdiff
path: root/app/services/users/set_status_service.rb
blob: d0bb40cbcfb9f19b3d4ed6308588f8388bb6dec7 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
# frozen_string_literal: true

module Users
  class SetStatusService
    include Gitlab::Allowable

    attr_reader :current_user, :target_user, :params

    def initialize(current_user, params)
      @current_user = current_user
      @params = params.dup
      @target_user = params.delete(:user) || current_user
    end

    def execute
      return false unless can?(current_user, :update_user_status, target_user)

      if status_cleared?
        remove_status
      else
        set_status
      end
    end

    private

    def set_status
      params[:emoji] = UserStatus::DEFAULT_EMOJI if params[:emoji].blank?
      params[:availability] = UserStatus.availabilities[:not_set] unless new_user_availability

      bump_user if user_status.update(params)
    end

    def remove_status
      bump_user if UserStatus.delete(target_user.id).nonzero?
      true
    end

    def user_status
      target_user.status || target_user.build_status
    end

    def status_cleared?
      params[:emoji].blank? &&
        params[:message].blank? &&
        (new_user_availability.blank? || new_user_availability == UserStatus.availabilities[:not_set])
    end

    def new_user_availability
      UserStatus.availabilities[params[:availability]]
    end

    def bump_user
      # Intentionally not calling `touch` as that will trigger other callbacks
      # on target_user (e.g. after_touch, after_commit, after_rollback) and we
      # don't need them to happen here.
      target_user.update_column(:updated_at, Time.current)
    end
  end
end