group.rb 9.41 KB
Newer Older
Steven Thonus's avatar
Steven Thonus committed
1 2
require 'carrierwave/orm/activerecord'

3
class Group < Namespace
4
  include Gitlab::ConfigHelper
5
  include AccessRequestable
6
  include Avatarable
7
  include Referable
8
  include SelectForProjectAuthorization
9
  include LoadedInGroupList
10
  include GroupDescendant
11

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

20
  has_many :requesters, -> { where.not(requested_at: nil) }, dependent: :destroy, as: :source, class_name: 'GroupMember' # rubocop:disable Cop/ActiveRecordDependent
21
  has_many :members_and_requesters, as: :source, class_name: 'GroupMember'
22

Felipe Artur's avatar
Felipe Artur committed
23
  has_many :milestones
24
  has_many :project_group_links, dependent: :destroy # rubocop:disable Cop/ActiveRecordDependent
25
  has_many :shared_projects, through: :project_group_links, source: :project
26
  has_many :notification_settings, dependent: :destroy, as: :source # rubocop:disable Cop/ActiveRecordDependent
27
  has_many :labels, class_name: 'GroupLabel'
Shinya Maeda's avatar
Shinya Maeda committed
28
  has_many :variables, class_name: 'Ci::GroupVariable'
29
  has_many :custom_attributes, class_name: 'GroupCustomAttribute'
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
30

31
  validate :avatar_type, if: ->(user) { user.avatar.present? && user.avatar_changed? }
32
  validate :visibility_level_allowed_by_projects
33
  validate :visibility_level_allowed_by_sub_groups
34
  validate :visibility_level_allowed_by_parent
35

36
  validates :avatar, file_size: { maximum: 200.kilobytes.to_i }
Steven Thonus's avatar
Steven Thonus committed
37

38 39
  validates :two_factor_grace_period, presence: true, numericality: { greater_than_or_equal_to: 0 }

Douwe Maan's avatar
Douwe Maan committed
40
  mount_uploader :avatar, AvatarUploader
41
  has_many :uploads, as: :model, dependent: :destroy # rubocop:disable Cop/ActiveRecordDependent
42

43 44
  after_create :post_create_hook
  after_destroy :post_destroy_hook
45
  after_save :update_two_factor_requirement
46
  after_update :path_changed_hook, if: :path_changed?
47

48
  class << self
49 50 51 52
    def supports_nested_groups?
      Gitlab::Database.postgresql?
    end

53 54 55 56 57 58 59
    # 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.
60
    def search(query)
61
      table   = Namespace.arel_table
62 63 64
      pattern = "%#{query}%"

      where(table[:name].matches(pattern).or(table[:path].matches(pattern)))
65 66 67
    end

    def sort(method)
68 69 70 71 72 73 74
      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
75
    end
76 77

    def reference_prefix
78 79 80 81 82
      User.reference_prefix
    end

    def reference_pattern
      User.reference_pattern
83
    end
84 85 86 87

    def visible_to_user(user)
      where(id: user.authorized_groups.select(:id).reorder(nil))
    end
88 89 90

    def select_for_project_authorization
      if current_scope.joins_values.include?(:shared_projects)
91 92
        joins('INNER JOIN namespaces project_namespace ON project_namespace.id = projects.namespace_id')
          .where('project_namespace.share_with_group_lock = ?',  false)
93
          .select("projects.id AS project_id, LEAST(project_group_links.group_access, members.access_level) AS access_level")
94 95 96 97
      else
        super
      end
    end
98 99
  end

100
  def to_reference(_from_project = nil, full: nil)
101
    "#{self.class.reference_prefix}#{full_path}"
102 103
  end

104
  def web_url
105
    Gitlab::Routing.url_helpers.group_canonical_url(self)
106 107
  end

108
  def human_name
109
    full_name
110
  end
111

112
  def visibility_level_allowed_by_parent?(level = self.visibility_level)
Rubén Dávila's avatar
Rubén Dávila committed
113
    return true unless parent_id && parent_id.nonzero?
114

115 116
    level <= parent.visibility_level
  end
117

118
  def visibility_level_allowed_by_projects?(level = self.visibility_level)
119
    !projects.where('visibility_level > ?', level).exists?
120
  end
121

122
  def visibility_level_allowed_by_sub_groups?(level = self.visibility_level)
123
    !children.where('visibility_level > ?', level).exists?
124 125
  end

126 127 128 129
  def visibility_level_allowed?(level = self.visibility_level)
    visibility_level_allowed_by_parent?(level) &&
      visibility_level_allowed_by_projects?(level) &&
      visibility_level_allowed_by_sub_groups?(level)
130 131
  end

132 133 134 135
  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)
136 137
  end

138 139 140 141 142 143 144
  def lfs_enabled?
    return false unless Gitlab.config.lfs.enabled
    return Gitlab.config.lfs.enabled if self[:lfs_enabled].nil?

    self[:lfs_enabled]
  end

145
  def add_users(users, access_level, current_user: nil, expires_at: nil)
146
    GroupMember.add_users(
147 148 149 150 151 152
      self,
      users,
      access_level,
      current_user: current_user,
      expires_at: expires_at
    )
153 154
  end

155
  def add_user(user, access_level, current_user: nil, expires_at: nil)
156 157 158 159 160 161 162
    GroupMember.add_user(
      self,
      user,
      access_level,
      current_user: current_user,
      expires_at: expires_at
    )
163 164
  end

165
  def add_guest(user, current_user = nil)
166
    add_user(user, :guest, current_user: current_user)
167 168 169
  end

  def add_reporter(user, current_user = nil)
170
    add_user(user, :reporter, current_user: current_user)
171 172 173
  end

  def add_developer(user, current_user = nil)
174
    add_user(user, :developer, current_user: current_user)
175 176 177
  end

  def add_master(user, current_user = nil)
178
    add_user(user, :master, current_user: current_user)
179 180
  end

Douwe Maan's avatar
Douwe Maan committed
181
  def add_owner(user, current_user = nil)
182
    add_user(user, :owner, current_user: current_user)
Douwe Maan's avatar
Douwe Maan committed
183 184
  end

185 186 187 188 189 190
  def member?(user, min_access_level = Gitlab::Access::GUEST)
    return false unless user

    max_member_access_for_user(user) >= min_access_level
  end

Douwe Maan's avatar
Douwe Maan committed
191
  def has_owner?(user)
192 193
    return false unless user

194
    members_with_parents.owners.where(user_id: user).any?
Douwe Maan's avatar
Douwe Maan committed
195 196 197
  end

  def has_master?(user)
198 199
    return false unless user

200
    members_with_parents.masters.where(user_id: user).any?
Douwe Maan's avatar
Douwe Maan committed
201 202
  end

203 204
  # Check if user is a last owner of the group.
  # Parent owners are ignored for nested groups.
Douwe Maan's avatar
Douwe Maan committed
205
  def last_owner?(user)
206
    owners.include?(user) && owners.size == 1
Douwe Maan's avatar
Douwe Maan committed
207 208
  end

Steven Thonus's avatar
Steven Thonus committed
209 210 211 212 213
  def avatar_type
    unless self.avatar.image?
      self.errors.add :avatar, "only images allowed"
    end
  end
214

215
  def post_create_hook
216 217
    Gitlab::AppLogger.info("Group \"#{name}\" was created")

218 219 220 221
    system_hook_service.execute_hooks_for(self, :create)
  end

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

224 225 226 227 228 229
    system_hook_service.execute_hooks_for(self, :destroy)
  end

  def system_hook_service
    SystemHooksService.new
  end
230

231
  def refresh_members_authorized_projects(blocking: true)
232
    UserProjectAccessChangedService.new(user_ids_for_project_authorizations)
233
      .execute(blocking: blocking)
234 235 236
  end

  def user_ids_for_project_authorizations
237
    members_with_parents.pluck(:user_id)
238 239 240
  end

  def members_with_parents
241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257
    # Avoids an unnecessary SELECT when the group has no parents
    source_ids =
      if parent_id
        self_and_ancestors.reorder(nil).select(:id)
      else
        id
      end

    GroupMember
      .active_without_invites
      .where(source_id: source_ids)
  end

  def members_with_descendants
    GroupMember
      .active_without_invites
      .where(source_id: self_and_descendants.reorder(nil).select(:id))
258 259 260
  end

  def users_with_parents
261 262 263
    User
      .where(id: members_with_parents.select(:user_id))
      .reorder(nil)
264
  end
Z.J. van de Weg's avatar
Z.J. van de Weg committed
265

266
  def users_with_descendants
267 268 269
    User
      .where(id: members_with_descendants.select(:user_id))
      .reorder(nil)
270 271
  end

272 273 274
  def max_member_access_for_user(user)
    return GroupMember::OWNER if user.admin?

275 276 277 278
    members_with_parents
      .where(user_id: user)
      .reorder(access_level: :desc)
      .first&.
279 280 281
      access_level || GroupMember::NO_ACCESS
  end

Z.J. van de Weg's avatar
Z.J. van de Weg committed
282 283 284 285 286 287 288 289 290
  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
291

Shinya Maeda's avatar
Shinya Maeda committed
292
  def secret_variables_for(ref, project)
293 294 295 296 297
    list_of_ids = [self] + ancestors
    variables = Ci::GroupVariable.where(group: list_of_ids)
    variables = variables.unprotected unless project.protected_for?(ref)
    variables = variables.group_by(&:group_id)
    list_of_ids.reverse.map { |group| variables[group.id] }.compact.flatten
Shinya Maeda's avatar
Shinya Maeda committed
298 299
  end

300 301 302 303 304 305
  def full_path_was
    return path_was unless has_parent?

    "#{parent.full_path}/#{path_was}"
  end

306
  private
307 308 309 310 311 312

  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
313

314 315 316 317
  def path_changed_hook
    system_hook_service.execute_hooks_for(self, :rename)
  end

318 319 320
  def visibility_level_allowed_by_parent
    return if visibility_level_allowed_by_parent?

321
    errors.add(:visibility_level, "#{visibility} is not allowed since the parent group has a #{parent.visibility} visibility.")
322 323 324 325 326
  end

  def visibility_level_allowed_by_projects
    return if visibility_level_allowed_by_projects?

327
    errors.add(:visibility_level, "#{visibility} is not allowed since this group contains projects with higher visibility.")
328 329 330 331 332
  end

  def visibility_level_allowed_by_sub_groups
    return if visibility_level_allowed_by_sub_groups?

333
    errors.add(:visibility_level, "#{visibility} is not allowed since there are sub-groups with higher visibility.")
334
  end
335
end