member.rb 8.3 KB
Newer Older
D
Dmitriy Zaporozhets 已提交
1
class Member < ActiveRecord::Base
2
  include Sortable
3
  include Importable
4
  include Expirable
D
Dmitriy Zaporozhets 已提交
5 6
  include Gitlab::Access

7 8
  attr_accessor :raw_invite_token

9
  belongs_to :created_by, class_name: "User"
D
Dmitriy Zaporozhets 已提交
10 11 12
  belongs_to :user
  belongs_to :source, polymorphic: true

D
Douwe Maan 已提交
13
  validates :user, presence: true, unless: :invite?
D
Dmitriy Zaporozhets 已提交
14
  validates :source, presence: true
15
  validates :user_id, uniqueness: { scope: [:source_type, :source_id],
D
Douwe Maan 已提交
16 17
                                    message: "already exists in source",
                                    allow_nil: true }
18
  validates :access_level, inclusion: { in: Gitlab::Access.all_values }, presence: true
D
Douwe Maan 已提交
19 20 21 22
  validates :invite_email,
    presence: {
      if: :invite?
    },
23
    email: {
D
Douwe Maan 已提交
24 25 26 27 28 29
      allow_nil: true
    },
    uniqueness: {
      scope: [:source_type, :source_id],
      allow_nil: true
    }
D
Dmitriy Zaporozhets 已提交
30

31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46
  # This scope encapsulates (most of) the conditions a row in the member table
  # must satisfy if it is a valid permission. Of particular note:
  #
  #   * Access requests must be excluded
  #   * Blocked users must be excluded
  #   * Invitations take effect immediately
  #   * expires_at is not implemented. A background worker purges expired rows
  scope :active, -> do
    is_external_invite = arel_table[:user_id].eq(nil).and(arel_table[:invite_token].not_eq(nil))
    user_is_active = User.arel_table[:state].eq(:active)

    includes(:user).references(:users)
      .where(is_external_invite.or(user_is_active))
      .where(requested_at: nil)
  end

R
Rémy Coutable 已提交
47
  scope :invite, -> { where.not(invite_token: nil) }
R
Rémy Coutable 已提交
48
  scope :non_invite, -> { where(invite_token: nil) }
R
Rémy Coutable 已提交
49
  scope :request, -> { where.not(requested_at: nil) }
50 51 52 53 54 55 56 57 58

  scope :has_access, -> { active.where('access_level > 0') }

  scope :guests, -> { active.where(access_level: GUEST) }
  scope :reporters, -> { active.where(access_level: REPORTER) }
  scope :developers, -> { active.where(access_level: DEVELOPER) }
  scope :masters,  -> { active.where(access_level: MASTER) }
  scope :owners,  -> { active.where(access_level: OWNER) }
  scope :owners_and_masters,  -> { active.where(access_level: [OWNER, MASTER]) }
59

60 61 62 63 64
  scope :order_name_asc, -> { joins(:user).merge(User.order_name_asc) }
  scope :order_name_desc, -> { joins(:user).merge(User.order_name_desc) }
  scope :order_recent_sign_in, -> { joins(:user).merge(User.order_recent_sign_in) }
  scope :order_oldest_sign_in, -> { joins(:user).merge(User.order_oldest_sign_in) }

D
Douwe Maan 已提交
65
  before_validation :generate_invite_token, on: :create, if: -> (member) { member.invite_email.present? }
66

67
  after_create :send_invite, if: :invite?, unless: :importing?
J
James Lopez 已提交
68 69 70
  after_create :send_request, if: :request?, unless: :importing?
  after_create :create_notification_setting, unless: [:pending?, :importing?]
  after_create :post_create_hook, unless: [:pending?, :importing?]
71
  after_create :refresh_member_authorized_projects, if: :importing?
J
James Lopez 已提交
72
  after_update :post_update_hook, unless: [:pending?, :importing?]
73
  after_destroy :post_destroy_hook, unless: :pending?
D
Douwe Maan 已提交
74

75
  delegate :name, :username, :email, to: :user, prefix: true
D
Douwe Maan 已提交
76

77 78
  default_value_for :notification_level, NotificationSetting.levels[:global]

79
  class << self
80 81 82 83 84 85 86 87 88 89 90 91 92 93 94
    def search(query)
      joins(:user).merge(User.search(query))
    end

    def sort(method)
      case method.to_s
      when 'recent_sign_in' then order_recent_sign_in
      when 'oldest_sign_in' then order_oldest_sign_in
      when 'last_joined' then order_created_desc
      when 'oldest_joined' then order_created_asc
      else
        order_by(method)
      end
    end

S
Stan Hu 已提交
95
    def access_for_user_ids(user_ids)
A
Adam Niedzielski 已提交
96
      where(user_id: user_ids).has_access.pluck(:user_id, :access_level).to_h
S
Stan Hu 已提交
97 98
    end

99 100 101 102 103
    def find_by_invite_token(invite_token)
      invite_token = Devise.token_generator.digest(self, :invite_token, invite_token)
      find_by(invite_token: invite_token)
    end

104 105 106
    def add_user(source, user, access_level, current_user: nil, expires_at: nil)
      user = retrieve_user(user)
      access_level = retrieve_access_level(access_level)
107

108
      # `user` can be either a User object or an email to be invited
109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126
      member =
        if user.is_a?(User)
          source.members.find_by(user_id: user.id) ||
          source.requesters.find_by(user_id: user.id) ||
          source.members.build(user_id: user.id)
        else
          source.members.build(invite_email: user)
        end

      return member unless can_update_member?(current_user, member)

      member.attributes = {
        created_by: member.created_by || current_user,
        access_level: access_level,
        expires_at: expires_at
      }

      if member.request?
127 128 129 130 131 132
        ::Members::ApproveAccessRequestService.new(
          source,
          current_user,
          id: member.id,
          access_level: access_level
        ).execute
133
      else
134
        member.save
135
      end
136

137 138
      UserProjectAccessChangedService.new(user.id).execute if user.is_a?(User)

139 140
      member
    end
141

142 143
    def access_levels
      Gitlab::Access.sym_options
144
    end
145 146 147

    private

148 149 150 151 152 153 154 155 156 157 158 159
    # This method is used to find users that have been entered into the "Add members" field.
    # These can be the User objects directly, their IDs, their emails, or new emails to be invited.
    def retrieve_user(user)
      return user if user.is_a?(User)

      User.find_by(id: user) || User.find_by(email: user) || user
    end

    def retrieve_access_level(access_level)
      access_levels.fetch(access_level) { access_level.to_i }
    end

160
    def can_update_member?(current_user, member)
D
Douwe Maan 已提交
161
      # There is no current user for bulk actions, in which case anything is allowed
162
      !current_user || current_user.can?(:"update_#{member.type.underscore}", member)
163
    end
164

165 166 167 168 169 170 171 172 173 174
    def add_users_to_source(source, users, access_level, current_user: nil, expires_at: nil)
      users.each do |user|
        add_user(
          source,
          user,
          access_level,
          current_user: current_user,
          expires_at: expires_at
        )
      end
175
    end
D
Douwe Maan 已提交
176 177
  end

R
Rémy Coutable 已提交
178 179 180 181
  def real_source_type
    source_type
  end

D
Douwe Maan 已提交
182 183 184 185
  def invite?
    self.invite_token.present?
  end

186
  def request?
R
Rémy Coutable 已提交
187
    requested_at.present?
188 189
  end

R
Rémy Coutable 已提交
190 191
  def pending?
    invite? || request?
D
Douwe Maan 已提交
192 193
  end

R
Rémy Coutable 已提交
194
  def accept_request
195 196
    return false unless request?

R
Rémy Coutable 已提交
197
    updated = self.update(requested_at: nil)
R
Rémy Coutable 已提交
198
    after_accept_request if updated
199

R
Rémy Coutable 已提交
200
    updated
201 202
  end

D
Douwe Maan 已提交
203
  def accept_invite!(new_user)
D
Douwe Maan 已提交
204
    return false unless invite?
205

D
Douwe Maan 已提交
206 207 208 209 210 211 212 213 214 215 216 217
    self.invite_token = nil
    self.invite_accepted_at = Time.now.utc

    self.user = new_user

    saved = self.save

    after_accept_invite if saved

    saved
  end

D
Douwe Maan 已提交
218 219 220 221 222 223 224 225 226 227
  def decline_invite!
    return false unless invite?

    destroyed = self.destroy

    after_decline_invite if destroyed

    destroyed
  end

D
Douwe Maan 已提交
228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245
  def generate_invite_token
    raw, enc = Devise.token_generator.generate(self.class, :invite_token)
    @raw_invite_token = raw
    self.invite_token = enc
  end

  def generate_invite_token!
    generate_invite_token && save(validate: false)
  end

  def resend_invite
    return unless invite?

    generate_invite_token! unless @raw_invite_token

    send_invite
  end

246
  def create_notification_setting
247
    user.notification_settings.find_or_create_for(source)
248 249
  end

250
  def notification_setting
251
    @notification_setting ||= user.notification_settings_for(source)
252 253
  end

D
Douwe Maan 已提交
254 255 256 257 258 259
  private

  def send_invite
    # override in subclass
  end

R
Rémy Coutable 已提交
260
  def send_request
R
Rémy Coutable 已提交
261
    notification_service.new_access_request(self)
D
Douwe Maan 已提交
262 263 264
  end

  def post_create_hook
265
    UserProjectAccessChangedService.new(user.id).execute
D
Douwe Maan 已提交
266 267 268 269
    system_hook_service.execute_hooks_for(self, :create)
  end

  def post_update_hook
270
    UserProjectAccessChangedService.new(user.id).execute if access_level_changed?
D
Douwe Maan 已提交
271 272 273
  end

  def post_destroy_hook
274
    refresh_member_authorized_projects
D
Douwe Maan 已提交
275 276 277
    system_hook_service.execute_hooks_for(self, :destroy)
  end

278 279 280 281 282 283 284 285 286
  def refresh_member_authorized_projects
    # If user/source is being destroyed, project access are gonna be destroyed eventually
    # because of DB foreign keys, so we shouldn't bother with refreshing after each
    # member is destroyed through association
    return if destroyed_by_association.present?

    UserProjectAccessChangedService.new(user_id).execute
  end

D
Douwe Maan 已提交
287 288 289 290
  def after_accept_invite
    post_create_hook
  end

D
Douwe Maan 已提交
291 292 293 294
  def after_decline_invite
    # override in subclass
  end

R
Rémy Coutable 已提交
295
  def after_accept_request
D
Douwe Maan 已提交
296 297 298 299 300 301 302 303 304 305
    post_create_hook
  end

  def system_hook_service
    SystemHooksService.new
  end

  def notification_service
    NotificationService.new
  end
D
Dmitriy Zaporozhets 已提交
306
end