notification_service.rb 15.1 KB
Newer Older
1 2
# NotificationService class
#
J
Johannes Schleifenbaum 已提交
3
# Used for notifying users with emails about different events
4 5 6 7 8 9 10
#
# Ex.
#   NotificationService.new.new_issue(issue, current_user)
#
class NotificationService
  # Always notify user about ssh key added
  # only if ssh key is not deploy key
11 12 13
  #
  # This is security email so it will be sent
  # even if user disabled notifications
14 15
  def new_key(key)
    if key.user
V
Valery Sizov 已提交
16
      mailer.new_ssh_key_email(key.id).deliver_later
17 18 19
    end
  end

20 21 22
  # Always notify user about email added to profile
  def new_email(email)
    if email.user
V
Valery Sizov 已提交
23
      mailer.new_email_email(email.id).deliver_later
24 25 26
    end
  end

27 28
  # When create an issue we should send next emails:
  #
29
  #  * issue assignee if their notification level is not Disabled
30 31 32
  #  * project team members with notification level higher then Participating
  #
  def new_issue(issue, current_user)
I
Izaak Alpert 已提交
33
    new_resource_email(issue, issue.project, 'new_issue_email')
34 35 36
  end

  # When we close an issue we should send next emails:
37
  #
38 39
  #  * issue author if their notification level is not Disabled
  #  * issue assignee if their notification level is not Disabled
40 41 42
  #  * project team members with notification level higher then Participating
  #
  def close_issue(issue, current_user)
I
Izaak Alpert 已提交
43
    close_resource_email(issue, issue.project, current_user, 'closed_issue_email')
44 45 46 47
  end

  # When we reassign an issue we should send next emails:
  #
48 49
  #  * issue old assignee if their notification level is not Disabled
  #  * issue new assignee if their notification level is not Disabled
50 51
  #
  def reassigned_issue(issue, current_user)
I
Izaak Alpert 已提交
52
    reassign_resource_email(issue, issue.project, current_user, 'reassigned_issue_email')
53 54
  end

55 56 57 58 59 60 61 62
  # When we change labels on an issue we should send emails.
  #
  # We pass in the labels, here, because we only want the labels that
  # have been *added* during this relabel, not all of them.
  def relabeled_issue(issue, labels, current_user)
    relabel_resource_email(issue, issue.project, labels, current_user, 'relabeled_issue_email')
  end

63 64 65

  # When create a merge request we should send next emails:
  #
66
  #  * mr assignee if their notification level is not Disabled
67 68
  #
  def new_merge_request(merge_request, current_user)
I
Izaak Alpert 已提交
69
    new_resource_email(merge_request, merge_request.target_project, 'new_merge_request_email')
70
  end
71 72 73

  # When we reassign a merge_request we should send next emails:
  #
74 75
  #  * merge_request old assignee if their notification level is not Disabled
  #  * merge_request assignee if their notification level is not Disabled
76 77
  #
  def reassigned_merge_request(merge_request, current_user)
I
Izaak Alpert 已提交
78
    reassign_resource_email(merge_request, merge_request.target_project, current_user, 'reassigned_merge_request_email')
79
  end
80

81 82 83 84 85 86 87 88
  # When we change labels on a merge request we should send emails.
  #
  # We pass in the labels, here, because we only want the labels that
  # have been *added* during this relabel, not all of them.
  def relabeled_merge_request(merge_request, labels, current_user)
    relabel_resource_email(merge_request, merge_request.project, labels, current_user, 'relabeled_merge_request_email')
  end

89
  def close_mr(merge_request, current_user)
I
Izaak Alpert 已提交
90
    close_resource_email(merge_request, merge_request.target_project, current_user, 'closed_merge_request_email')
91 92
  end

93 94 95 96
  def reopen_issue(issue, current_user)
    reopen_resource_email(issue, issue.project, current_user, 'issue_status_changed_email', 'reopened')
  end

97
  def merge_mr(merge_request, current_user)
V
Valery Sizov 已提交
98 99 100 101 102 103
    close_resource_email(
      merge_request,
      merge_request.target_project,
      current_user,
      'merged_merge_request_email'
    )
104 105
  end

M
Marin Jankovski 已提交
106
  def reopen_mr(merge_request, current_user)
V
Valery Sizov 已提交
107 108 109 110 111 112
    reopen_resource_email(
      merge_request,
      merge_request.target_project,
      current_user, 'merge_request_status_email',
      'reopened'
    )
M
Marin Jankovski 已提交
113 114
  end

115
  # Notify new user with email after creation
116
  def new_user(user, token = nil)
J
Johannes Schleifenbaum 已提交
117
    # Don't email omniauth created users
V
Valery Sizov 已提交
118
    mailer.new_user_email(user.id, token).deliver_later unless user.identities.any?
119 120 121 122 123 124 125
  end

  # Notify users on new note in system
  #
  # TODO: split on methods and refactor
  #
  def new_note(note)
126 127
    return true unless note.noteable_type.present?

128
    # ignore gitlab service messages
129
    return true if note.note.start_with?('Status changed to closed')
130
    return true if note.cross_reference? && note.system == true
V
Valery Sizov 已提交
131
    return true if note.is_award
132

133
    target = note.noteable
134

135
    recipients = []
136

137 138 139 140 141
    mentioned_users = note.mentioned_users
    mentioned_users.select! do |user|
      user.can?(:read_project, note.project)
    end

D
Douwe Maan 已提交
142
    # Add all users participating in the thread (author, assignee, comment authors)
143
    participants =
144
      if target.respond_to?(:participants)
145
        target.participants(note.author)
D
Douwe Maan 已提交
146
      else
147
        mentioned_users
D
Douwe Maan 已提交
148 149
      end
    recipients = recipients.concat(participants)
150 151

    # Merge project watchers
152
    recipients = add_project_watchers(recipients, note.project)
153

D
Douwe Maan 已提交
154
    # Reject users with Mention notification level, except those mentioned in _this_ note.
155 156
    recipients = reject_mention_users(recipients - mentioned_users, note.project)
    recipients = recipients + mentioned_users
157

158
    recipients = reject_muted_users(recipients, note.project)
159

V
Valery Sizov 已提交
160
    recipients = add_subscribed_users(recipients, note.noteable)
161
    recipients = add_label_subscriptions(recipients, note.noteable)
V
Valery Sizov 已提交
162 163
    recipients = reject_unsubscribed_users(recipients, note.noteable)

164
    recipients.delete(note.author)
165
    recipients = recipients.uniq
166 167 168 169

    # build notify method like 'note_commit_email'
    notify_method = "note_#{note.noteable_type.underscore}_email".to_sym
    recipients.each do |recipient|
V
Valery Sizov 已提交
170
      mailer.send(notify_method, recipient.id, note.id).deliver_later
171 172
    end
  end
173

D
Douwe Maan 已提交
174
  def invite_project_member(project_member, token)
V
Valery Sizov 已提交
175
    mailer.project_member_invited_email(project_member.id, token).deliver_later
D
Douwe Maan 已提交
176 177 178
  end

  def accept_project_invite(project_member)
V
Valery Sizov 已提交
179
    mailer.project_invite_accepted_email(project_member.id).deliver_later
D
Douwe Maan 已提交
180 181
  end

D
Douwe Maan 已提交
182
  def decline_project_invite(project_member)
V
Valery Sizov 已提交
183 184 185 186 187 188
    mailer.project_invite_declined_email(
      project_member.project.id,
      project_member.invite_email,
      project_member.access_level,
      project_member.created_by_id
    ).deliver_later
D
Douwe Maan 已提交
189 190
  end

191
  def new_project_member(project_member)
V
Valery Sizov 已提交
192
    mailer.project_access_granted_email(project_member.id).deliver_later
193 194
  end

195
  def update_project_member(project_member)
V
Valery Sizov 已提交
196
    mailer.project_access_granted_email(project_member.id).deliver_later
197
  end
198

D
Douwe Maan 已提交
199
  def invite_group_member(group_member, token)
V
Valery Sizov 已提交
200
    mailer.group_member_invited_email(group_member.id, token).deliver_later
D
Douwe Maan 已提交
201 202 203
  end

  def accept_group_invite(group_member)
V
Valery Sizov 已提交
204
    mailer.group_invite_accepted_email(group_member.id).deliver_later
D
Douwe Maan 已提交
205 206
  end

D
Douwe Maan 已提交
207
  def decline_group_invite(group_member)
V
Valery Sizov 已提交
208 209 210 211 212 213
    mailer.group_invite_declined_email(
      group_member.group.id,
      group_member.invite_email,
      group_member.access_level,
      group_member.created_by_id
    ).deliver_later
D
Douwe Maan 已提交
214 215
  end

216
  def new_group_member(group_member)
V
Valery Sizov 已提交
217
    mailer.group_access_granted_email(group_member.id).deliver_later
218 219
  end

220
  def update_group_member(group_member)
V
Valery Sizov 已提交
221
    mailer.group_access_granted_email(group_member.id).deliver_later
222 223
  end

224
  def project_was_moved(project, old_path_with_namespace)
225 226 227 228
    recipients = project.team.members
    recipients = reject_muted_users(recipients, project)

    recipients.each do |recipient|
V
Valery Sizov 已提交
229 230 231 232 233
      mailer.project_was_moved_email(
        project.id,
        recipient.id,
        old_path_with_namespace
      ).deliver_later
234 235 236
    end
  end

237 238
  protected

239 240
  # Get project users with WATCH notification level
  def project_watchers(project)
241
    project_members = project_member_notification(project)
242

243
    users_with_project_level_global = project_member_notification(project, Notification::N_GLOBAL)
244
    users_with_group_level_global = group_member_notification(project, Notification::N_GLOBAL)
245
    users = users_with_global_level_watch([users_with_project_level_global, users_with_group_level_global].flatten.uniq)
246

247
    users_with_project_setting = select_project_member_setting(project, users_with_project_level_global, users)
248
    users_with_group_setting = select_group_member_setting(project, project_members, users_with_group_level_global, users)
249

250
    User.where(id: users_with_project_setting.concat(users_with_group_setting).uniq).to_a
251 252
  end

253
  def project_member_notification(project, notification_level=nil)
254
    project_members = project.project_members
255

256 257 258 259 260
    if notification_level
      project_members.where(notification_level: notification_level).pluck(:user_id)
    else
      project_members.pluck(:user_id)
    end
261 262
  end

263
  def group_member_notification(project, notification_level)
264
    if project.group
265
      project.group.group_members.where(notification_level: notification_level).pluck(:user_id)
266 267
    else
      []
268
    end
269
  end
270

271 272 273 274 275 276 277 278
  def users_with_global_level_watch(ids)
    User.where(
      id: ids,
      notification_level: Notification::N_WATCH
    ).pluck(:id)
  end

  # Build a list of users based on project notifcation settings
279 280
  def select_project_member_setting(project, global_setting, users_global_level_watch)
    users = project_member_notification(project, Notification::N_WATCH)
281

282 283 284 285
    # If project setting is global, add to watch list if global setting is watch
    global_setting.each do |user_id|
      if users_global_level_watch.include?(user_id)
        users << user_id
286 287
      end
    end
288 289

    users
290 291
  end

S
Steven Burgart 已提交
292
  # Build a list of users based on group notification settings
293 294
  def select_group_member_setting(project, project_members, global_setting, users_global_level_watch)
    uids = group_member_notification(project, Notification::N_WATCH)
295 296 297 298 299 300

    # Group setting is watch, add to users list if user is not project member
    users = []
    uids.each do |user_id|
      if project_members.exclude?(user_id)
        users << user_id
301 302 303
      end
    end

304 305 306 307
    # Group setting is global, add to users list if global setting is watch
    global_setting.each do |user_id|
      if project_members.exclude?(user_id) && users_global_level_watch.include?(user_id)
        users << user_id
308 309 310
      end
    end

311
    users
312 313
  end

314 315 316 317
  def add_project_watchers(recipients, project)
    recipients.concat(project_watchers(project)).compact.uniq
  end

318 319
  # Remove users with disabled notifications from array
  # Also remove duplications and nil recipients
320
  def reject_muted_users(users, project = nil)
321
    reject_users(users, :disabled?, project)
322 323
  end

324 325
  # Remove users with notification level 'Mentioned'
  def reject_mention_users(users, project = nil)
326 327 328 329 330 331 332 333 334
    reject_users(users, :mention?, project)
  end

  # Reject users which method_name from notification object returns true.
  #
  # Example:
  #   reject_users(users, :watch?, project)
  #
  def reject_users(users, method_name, project = nil)
335
    users = users.to_a.compact.uniq
336
    users = users.reject(&:blocked?)
337 338

    users.reject do |user|
339
      next user.notification.send(method_name) unless project
340

D
Douwe Maan 已提交
341
      member = project.project_members.find_by(user_id: user.id)
342

D
Douwe Maan 已提交
343 344
      if !member && project.group
        member = project.group.group_members.find_by(user_id: user.id)
345 346 347
      end

      # reject users who globally set mention notification and has no membership
348
      next user.notification.send(method_name) unless member
349 350

      # reject users who set mention notification in project
351
      next true if member.notification.send(method_name)
352 353

      # reject users who have N_MENTION in project and disabled in global settings
354
      member.notification.global? && user.notification.send(method_name)
355 356 357
    end
  end

V
Valery Sizov 已提交
358
  def reject_unsubscribed_users(recipients, target)
V
Valery Sizov 已提交
359
    return recipients unless target.respond_to? :subscriptions
360

V
Valery Sizov 已提交
361
    recipients.reject do |user|
362 363
      subscription = target.subscriptions.find_by_user_id(user.id)
      subscription && !subscription.subscribed
V
Valery Sizov 已提交
364 365 366
    end
  end

V
Valery Sizov 已提交
367 368 369 370 371 372
  def add_subscribed_users(recipients, target)
    return recipients unless target.respond_to? :subscriptions

    subscriptions = target.subscriptions

    if subscriptions.any?
V
tests  
Valery Sizov 已提交
373
      recipients + subscriptions.where(subscribed: true).map(&:user)
V
Valery Sizov 已提交
374 375 376 377
    else
      recipients
    end
  end
378

379 380 381 382 383 384 385 386 387 388
  def add_label_subscriptions(recipients, target)
    return recipients unless target.respond_to? :labels

    target.labels.each do |label|
      recipients += label.subscriptions.where(subscribed: true).map(&:user)
    end

    recipients
  end

I
Izaak Alpert 已提交
389
  def new_resource_email(target, project, method)
390
    recipients = build_recipients(target, project, target.author)
391 392

    recipients.each do |recipient|
V
Valery Sizov 已提交
393
      mailer.send(method, recipient.id, target.id).deliver_later
394 395 396
    end
  end

I
Izaak Alpert 已提交
397
  def close_resource_email(target, project, current_user, method)
398
    recipients = build_recipients(target, project, current_user)
399 400

    recipients.each do |recipient|
V
Valery Sizov 已提交
401
      mailer.send(method, recipient.id, target.id, current_user.id).deliver_later
402 403 404
    end
  end

I
Izaak Alpert 已提交
405
  def reassign_resource_email(target, project, current_user, method)
406
    previous_assignee_id = previous_record(target, 'assignee_id')
407 408
    previous_assignee = User.find_by(id: previous_assignee_id) if previous_assignee_id

409
    recipients = build_recipients(target, project, current_user, action: :reassign, previous_assignee: previous_assignee)
410

411
    recipients.each do |recipient|
V
Valery Sizov 已提交
412 413 414 415 416 417 418
      mailer.send(
        method,
        recipient.id,
        target.id,
        previous_assignee_id,
        current_user.id
      ).deliver_later
419 420
    end
  end
D
Dmitriy Zaporozhets 已提交
421

422 423 424 425 426 427 428 429 430
  def relabel_resource_email(target, project, labels, current_user, method)
    recipients = build_relabel_recipients(target, project, labels, current_user)
    label_names = labels.map(&:name)

    recipients.each do |recipient|
      mailer.send(method, recipient.id, target.id, current_user.id, label_names).deliver_later
    end
  end

431
  def reopen_resource_email(target, project, current_user, method, status)
432
    recipients = build_recipients(target, project, current_user)
433 434

    recipients.each do |recipient|
V
Valery Sizov 已提交
435
      mailer.send(method, recipient.id, target.id, status, current_user.id).deliver_later
436 437 438
    end
  end

439
  def build_recipients(target, project, current_user, action: nil, previous_assignee: nil)
440
    recipients = target.participants(current_user)
441

D
Fix.  
Douwe Maan 已提交
442
    recipients = add_project_watchers(recipients, project)
443
    recipients = reject_mention_users(recipients, project)
444

445 446 447 448
    # Re-assign is considered as a mention of the new assignee so we add the
    # new assignee to the list of recipients after we rejected users with
    # the "on mention" notification level
    if action == :reassign
449
      recipients << previous_assignee if previous_assignee
450 451 452 453
      recipients << target.assignee
    end

    recipients = reject_muted_users(recipients, project)
V
Valery Sizov 已提交
454
    recipients = add_subscribed_users(recipients, target)
455
    recipients = add_label_subscriptions(recipients, target)
V
Valery Sizov 已提交
456
    recipients = reject_unsubscribed_users(recipients, target)
457

458 459
    recipients.delete(current_user)

460
    recipients.uniq
461 462
  end

463 464 465 466 467 468 469
  def build_relabel_recipients(target, project, labels, current_user)
    recipients = add_label_subscriptions([], target)
    recipients = reject_unsubscribed_users(recipients, target)
    recipients.delete(current_user)
    recipients.uniq
  end

D
Dmitriy Zaporozhets 已提交
470
  def mailer
V
Valery Sizov 已提交
471
    Notify
D
Dmitriy Zaporozhets 已提交
472
  end
473 474 475 476

  def previous_record(object, attribute)
    if object && attribute
      if object.previous_changes.include?(attribute)
M
Marin Jankovski 已提交
477
        object.previous_changes[attribute].first
478 479 480
      end
    end
  end
481
end