group.rb 6.9 KB
Newer Older
S
Steven Thonus 已提交
1 2
require 'carrierwave/orm/activerecord'

3
class Group < Namespace
4
  include Gitlab::ConfigHelper
F
Felipe Artur 已提交
5
  include Gitlab::VisibilityLevel
R
Rémy Coutable 已提交
6
  include AccessRequestable
7
  include Avatarable
8
  include Referable
9
  include SelectForProjectAuthorization
F
Felipe Artur 已提交
10

11
  has_many :group_members, -> { where(requested_at: nil) }, dependent: :destroy, as: :source
12
  alias_method :members, :group_members
13
  has_many :users, through: :group_members
14
  has_many :owners,
15
    -> { where(members: { access_level: Gitlab::Access::OWNER }) },
16 17 18
    through: :group_members,
    source: :user

19 20
  has_many :requesters, -> { where.not(requested_at: nil) }, dependent: :destroy, as: :source, class_name: 'GroupMember'

21 22
  has_many :project_group_links, dependent: :destroy
  has_many :shared_projects, through: :project_group_links, source: :project
23
  has_many :notification_settings, dependent: :destroy, as: :source
D
Douglas Barbosa Alexandre 已提交
24
  has_many :labels, class_name: 'GroupLabel'
A
Andrey Kumanyaev 已提交
25

26
  validate :avatar_type, if: ->(user) { user.avatar.present? && user.avatar_changed? }
27 28
  validate :visibility_level_allowed_by_projects

29
  validates :avatar, file_size: { maximum: 200.kilobytes.to_i }
S
Steven Thonus 已提交
30

31 32
  validates :two_factor_grace_period, presence: true, numericality: { greater_than_or_equal_to: 0 }

D
Douwe Maan 已提交
33
  mount_uploader :avatar, AvatarUploader
34
  has_many :uploads, as: :model, dependent: :destroy
35

36 37
  after_create :post_create_hook
  after_destroy :post_destroy_hook
38
  after_save :update_two_factor_requirement
39

40
  class << self
41 42 43 44
    def supports_nested_groups?
      Gitlab::Database.postgresql?
    end

45 46 47 48 49 50 51
    # Searches for groups matching the given query.
    #
    # This method uses ILIKE on PostgreSQL and LIKE on MySQL.
    #
    # query - The search query as a String
    #
    # Returns an ActiveRecord::Relation.
52
    def search(query)
53
      table   = Namespace.arel_table
54 55 56
      pattern = "%#{query}%"

      where(table[:name].matches(pattern).or(table[:path].matches(pattern)))
57 58 59
    end

    def sort(method)
M
Markus Koller 已提交
60 61 62 63 64 65 66
      if method == 'storage_size_desc'
        # storage_size is a virtual column so we need to
        # pass a string to avoid AR adding the table name
        reorder('storage_size DESC, namespaces.id DESC')
      else
        order_by(method)
      end
67
    end
68 69

    def reference_prefix
70 71 72 73 74
      User.reference_prefix
    end

    def reference_pattern
      User.reference_pattern
75
    end
76 77 78 79

    def visible_to_user(user)
      where(id: user.authorized_groups.select(:id).reorder(nil))
    end
80 81 82

    def select_for_project_authorization
      if current_scope.joins_values.include?(:shared_projects)
83 84
        joins('INNER JOIN namespaces project_namespace ON project_namespace.id = projects.namespace_id')
          .where('project_namespace.share_with_group_lock = ?',  false)
85
          .select("projects.id AS project_id, LEAST(project_group_links.group_access, members.access_level) AS access_level")
86 87 88 89
      else
        super
      end
    end
90 91
  end

92
  def to_reference(_from_project = nil, full: nil)
93
    "#{self.class.reference_prefix}#{full_path}"
94 95
  end

96
  def web_url
97
    Gitlab::Routing.url_helpers.group_canonical_url(self)
98 99
  end

100
  def human_name
101
    full_name
102
  end
103

F
Felipe Artur 已提交
104
  def visibility_level_field
105
    :visibility_level
F
Felipe Artur 已提交
106 107
  end

108
  def visibility_level_allowed_by_projects
D
Douwe Maan 已提交
109
    allowed_by_projects = self.projects.where('visibility_level > ?', self.visibility_level).none?
110 111 112 113 114 115 116 117 118

    unless allowed_by_projects
      level_name = Gitlab::VisibilityLevel.level_name(visibility_level).downcase
      self.errors.add(:visibility_level, "#{level_name} is not allowed since there are projects with higher visibility.")
    end

    allowed_by_projects
  end

119 120 121 122
  def avatar_url(**args)
    # We use avatar_path instead of overriding avatar_url because of carrierwave.
    # See https://gitlab.com/gitlab-org/gitlab-ce/merge_requests/11001/diffs#note_28659864
    avatar_path(args)
123 124
  end

125 126 127 128 129 130 131
  def lfs_enabled?
    return false unless Gitlab.config.lfs.enabled
    return Gitlab.config.lfs.enabled if self[:lfs_enabled].nil?

    self[:lfs_enabled]
  end

132
  def add_users(users, access_level, current_user: nil, expires_at: nil)
133
    GroupMember.add_users(
134 135 136 137 138 139
      self,
      users,
      access_level,
      current_user: current_user,
      expires_at: expires_at
    )
140 141
  end

142
  def add_user(user, access_level, current_user: nil, expires_at: nil)
143 144 145 146 147 148 149
    GroupMember.add_user(
      self,
      user,
      access_level,
      current_user: current_user,
      expires_at: expires_at
    )
150 151
  end

152
  def add_guest(user, current_user = nil)
153
    add_user(user, :guest, current_user: current_user)
154 155 156
  end

  def add_reporter(user, current_user = nil)
157
    add_user(user, :reporter, current_user: current_user)
158 159 160
  end

  def add_developer(user, current_user = nil)
161
    add_user(user, :developer, current_user: current_user)
162 163 164
  end

  def add_master(user, current_user = nil)
165
    add_user(user, :master, current_user: current_user)
166 167
  end

D
Douwe Maan 已提交
168
  def add_owner(user, current_user = nil)
169
    add_user(user, :owner, current_user: current_user)
D
Douwe Maan 已提交
170 171 172
  end

  def has_owner?(user)
173
    members_with_parents.owners.where(user_id: user).any?
D
Douwe Maan 已提交
174 175 176
  end

  def has_master?(user)
177
    members_with_parents.masters.where(user_id: user).any?
D
Douwe Maan 已提交
178 179
  end

180 181
  # Check if user is a last owner of the group.
  # Parent owners are ignored for nested groups.
D
Douwe Maan 已提交
182
  def last_owner?(user)
183
    owners.include?(user) && owners.size == 1
D
Douwe Maan 已提交
184 185
  end

S
Steven Thonus 已提交
186 187 188 189 190
  def avatar_type
    unless self.avatar.image?
      self.errors.add :avatar, "only images allowed"
    end
  end
191

192
  def post_create_hook
193 194
    Gitlab::AppLogger.info("Group \"#{name}\" was created")

195 196 197 198
    system_hook_service.execute_hooks_for(self, :create)
  end

  def post_destroy_hook
199 200
    Gitlab::AppLogger.info("Group \"#{name}\" was removed")

201 202 203 204 205 206
    system_hook_service.execute_hooks_for(self, :destroy)
  end

  def system_hook_service
    SystemHooksService.new
  end
207 208

  def refresh_members_authorized_projects
209 210
    UserProjectAccessChangedService.new(user_ids_for_project_authorizations)
      .execute
211 212 213 214
  end

  def user_ids_for_project_authorizations
    users_with_parents.pluck(:id)
215 216 217
  end

  def members_with_parents
218
    GroupMember.non_request.where(source_id: ancestors.pluck(:id).push(id))
219 220 221
  end

  def users_with_parents
222
    User.where(id: members_with_parents.select(:user_id))
223
  end
Z
Z.J. van de Weg 已提交
224

225 226 227 228 229 230
  def users_with_descendants
    members_with_descendants = GroupMember.non_request.where(source_id: descendants.pluck(:id).push(id))

    User.where(id: members_with_descendants.select(:user_id))
  end

231 232 233
  def max_member_access_for_user(user)
    return GroupMember::OWNER if user.admin?

234 235 236 237
    members_with_parents
      .where(user_id: user)
      .reorder(access_level: :desc)
      .first&.
238 239 240
      access_level || GroupMember::NO_ACCESS
  end

Z
Z.J. van de Weg 已提交
241 242 243 244 245 246 247 248 249
  def mattermost_team_params
    max_length = 59

    {
      name: path[0..max_length],
      display_name: name[0..max_length],
      type: public? ? 'O' : 'I' # Open vs Invite-only
    }
  end
250 251 252 253 254 255 256 257

  protected

  def update_two_factor_requirement
    return unless require_two_factor_authentication_changed? || two_factor_grace_period_changed?

    users.find_each(&:update_two_factor_requirement)
  end
258
end