BigW Consortium Gitlab

participable.rb 2.83 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12
# == Participable concern
#
# Contains functionality related to objects that can have participants, such as
# an author, an assignee and people mentioned in its description or comments.
#
# Usage:
#
#     class Issue < ActiveRecord::Base
#       include Participable
#
#       # ...
#
Yorick Peterse committed
13 14 15 16 17 18 19
#       participant :author
#       participant :assignee
#       participant :notes
#
#       participant -> (current_user, ext) do
#         ext.analyze('...')
#       end
20
#     end
21
#
22 23
#     issue = Issue.last
#     users = issue.participants
24 25 26 27
module Participable
  extend ActiveSupport::Concern

  module ClassMethods
Yorick Peterse committed
28 29 30 31 32 33 34 35 36 37 38 39 40 41 42
    # Adds a list of participant attributes. Attributes can either be symbols or
    # Procs.
    #
    # When using a Proc instead of a Symbol the Proc will be given two
    # arguments:
    #
    # 1. The current user (as an instance of User)
    # 2. An instance of `Gitlab::ReferenceExtractor`
    #
    # It is expected that a Proc populates the given reference extractor
    # instance with data. The return value of the Proc is ignored.
    #
    # attr - The name of the attribute or a Proc
    def participant(attr)
      participant_attrs << attr
43
    end
44
  end
45

46 47 48 49
  included do
    # Accessor for participant attributes.
    cattr_accessor :participant_attrs, instance_accessor: false do
      []
50 51 52
    end
  end

Yorick Peterse committed
53 54 55 56 57 58
  # Returns the users participating in a discussion.
  #
  # This method processes attributes of objects in breadth-first order.
  #
  # Returns an Array of User instances.
  def participants(current_user = nil)
59 60 61 62 63 64 65 66 67 68
    @participants ||= Hash.new do |hash, user|
      hash[user] = raw_participants(user)
    end

    @participants[current_user]
  end

  private

  def raw_participants(current_user = nil)
Yorick Peterse committed
69 70 71 72
    current_user ||= author
    ext = Gitlab::ReferenceExtractor.new(project, current_user)
    participants = Set.new
    process = [self]
73

Yorick Peterse committed
74 75
    until process.empty?
      source = process.pop
76

Yorick Peterse committed
77 78 79 80 81 82 83 84
      case source
      when User
        participants << source
      when Participable
        source.class.participant_attrs.each do |attr|
          if attr.respond_to?(:call)
            source.instance_exec(current_user, ext, &attr)
          else
85
            process << source.__send__(attr) # rubocop:disable GitlabSecurity/PublicSend
Yorick Peterse committed
86 87 88 89 90 91 92 93
          end
        end
      when Enumerable, ActiveRecord::Relation
        # This uses reverse_each so we can use "pop" to get the next value to
        # process (in order). Using unshift instead of pop would require
        # moving all Array values one index to the left (which can be
        # expensive).
        source.reverse_each { |obj| process << obj }
94 95 96
      end
    end

Yorick Peterse committed
97
    participants.merge(ext.users)
98

Jarka Kadlecova committed
99 100
    case self
    when PersonalSnippet
101 102 103 104
      Ability.users_that_can_read_personal_snippet(participants.to_a, self)
    else
      Ability.users_that_can_read_project(participants.to_a, project)
    end
Douwe Maan committed
105
  end
106
end