project.rb 48.4 KB
Newer Older
1 2
require 'carrierwave/orm/activerecord'

G
gitlabhq 已提交
3
class Project < ActiveRecord::Base
4
  include Gitlab::ConfigHelper
5
  include Gitlab::ShellAdapter
6
  include Gitlab::VisibilityLevel
7
  include Gitlab::CurrentSettings
R
Rémy Coutable 已提交
8
  include AccessRequestable
9
  include Avatarable
10
  include CacheMarkdownField
11 12
  include Referable
  include Sortable
13
  include AfterCommitQueue
14
  include CaseSensitivity
15
  include TokenAuthenticatable
16
  include ValidAttribute
F
Felipe Artur 已提交
17
  include ProjectFeaturesCompatibility
18
  include SelectForProjectAuthorization
19
  include Routable
R
Robert Speicher 已提交
20

21
  extend Gitlab::ConfigHelper
22
  extend Gitlab::CurrentSettings
23

24
  BoardLimitExceeded = Class.new(StandardError)
25

26
  NUMBER_OF_PERMITTED_BOARDS = 1
D
Douwe Maan 已提交
27
  UNKNOWN_IMPORT_URL = 'http://unknown.git'.freeze
28
  LATEST_STORAGE_VERSION = 1
J
Jared Szechy 已提交
29

30 31
  cache_markdown_field :description, pipeline: :description

32 33
  delegate :feature_available?, :builds_enabled?, :wiki_enabled?,
           :merge_requests_enabled?, :issues_enabled?, to: :project_feature,
34
                                                       allow_nil: true
F
Felipe Artur 已提交
35

36
  delegate :base_dir, :disk_path, :ensure_storage_path_exists, to: :storage
37

38
  default_value_for :archived, false
39
  default_value_for :visibility_level, gitlab_config_features.visibility_level
40
  default_value_for :resolve_outdated_diff_discussions, false
41
  default_value_for :container_registry_enabled, gitlab_config_features.container_registry
42
  default_value_for(:repository_storage) { current_application_settings.pick_repository_storage }
K
Kamil Trzcinski 已提交
43
  default_value_for(:shared_runners_enabled) { current_application_settings.shared_runners_enabled }
F
Felipe Artur 已提交
44 45 46 47 48
  default_value_for :issues_enabled, gitlab_config_features.issues
  default_value_for :merge_requests_enabled, gitlab_config_features.merge_requests
  default_value_for :builds_enabled, gitlab_config_features.builds
  default_value_for :wiki_enabled, gitlab_config_features.wiki
  default_value_for :snippets_enabled, gitlab_config_features.snippets
49
  default_value_for :only_allow_merge_if_all_discussions_are_resolved, false
50

51 52
  add_authentication_token_field :runners_token
  before_save :ensure_runners_token
53

54 55
  after_save :update_project_statistics, if: :namespace_id_changed?
  after_create :create_project_feature, unless: :project_feature
56
  after_create :set_last_activity_at
57
  after_create :set_last_repository_updated_at
58
  after_update :update_forks_visibility_level
59

60
  before_destroy :remove_private_deploy_keys
61
  after_destroy -> { run_after_commit { remove_pages } }
K
Kamil Trzcinski 已提交
62

63 64
  after_validation :check_pending_delete

65
  # Storage specific hooks
66
  after_initialize :use_hashed_storage
67 68
  after_create :ensure_storage_path_exists
  after_save :ensure_storage_path_exists, if: :namespace_id_changed?
69

70
  acts_as_taggable
D
Dmitriy Zaporozhets 已提交
71

72
  attr_accessor :old_path_with_namespace
73
  attr_accessor :template_name
74
  attr_writer :pipeline_status
75

76 77
  alias_attribute :title, :name

78
  # Relations
79
  belongs_to :creator, class_name: 'User'
80
  belongs_to :group, -> { where(type: 'Group') }, foreign_key: 'namespace_id'
81
  belongs_to :namespace
82

83
  has_one :last_event, -> {order 'events.created_at DESC'}, class_name: 'Event'
84
  has_many :boards, before_add: :validate_board_limit
F
Felipe Artur 已提交
85

86
  # Project services
87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119
  has_one :campfire_service
  has_one :drone_ci_service
  has_one :emails_on_push_service
  has_one :pipelines_email_service
  has_one :irker_service
  has_one :pivotaltracker_service
  has_one :hipchat_service
  has_one :flowdock_service
  has_one :assembla_service
  has_one :asana_service
  has_one :gemnasium_service
  has_one :mattermost_slash_commands_service
  has_one :mattermost_service
  has_one :slack_slash_commands_service
  has_one :slack_service
  has_one :buildkite_service
  has_one :bamboo_service
  has_one :teamcity_service
  has_one :pushover_service
  has_one :jira_service
  has_one :redmine_service
  has_one :custom_issue_tracker_service
  has_one :bugzilla_service
  has_one :gitlab_issue_tracker_service, inverse_of: :project
  has_one :external_wiki_service
  has_one :kubernetes_service, inverse_of: :project
  has_one :prometheus_service, inverse_of: :project
  has_one :mock_ci_service
  has_one :mock_deployment_service
  has_one :mock_monitoring_service
  has_one :microsoft_teams_service

  has_one  :forked_project_link,  foreign_key: "forked_to_project_id"
120 121 122 123
  has_one  :forked_from_project,  through:   :forked_project_link

  has_many :forked_project_links, foreign_key: "forked_from_project_id"
  has_many :forks,                through:     :forked_project_links, source: :forked_to_project
124

125
  # Merge Requests for target project should be removed with it
126 127 128 129 130 131 132 133 134 135 136
  has_many :merge_requests, foreign_key: 'target_project_id'
  has_many :issues
  has_many :labels, class_name: 'ProjectLabel'
  has_many :services
  has_many :events
  has_many :milestones
  has_many :notes
  has_many :snippets, class_name: 'ProjectSnippet'
  has_many :hooks, class_name: 'ProjectHook'
  has_many :protected_branches
  has_many :protected_tags
137

138
  has_many :project_authorizations
139
  has_many :authorized_users, through: :project_authorizations, source: :user, class_name: 'User'
140
  has_many :project_members, -> { where(requested_at: nil) },
141
    as: :source, dependent: :delete_all # rubocop:disable Cop/ActiveRecordDependent
142

R
Rémy Coutable 已提交
143
  alias_method :members, :project_members
144 145
  has_many :users, through: :project_members

146
  has_many :requesters, -> { where.not(requested_at: nil) },
147
    as: :source, class_name: 'ProjectMember', dependent: :delete_all # rubocop:disable Cop/ActiveRecordDependent
148
  has_many :members_and_requesters, as: :source, class_name: 'ProjectMember'
149

150
  has_many :deploy_keys_projects
151
  has_many :deploy_keys, through: :deploy_keys_projects
152
  has_many :users_star_projects
C
Ciro Santilli 已提交
153
  has_many :starrers, through: :users_star_projects, source: :user
154
  has_many :releases
155
  has_many :lfs_objects_projects, dependent: :destroy # rubocop:disable Cop/ActiveRecordDependent
M
Marin Jankovski 已提交
156
  has_many :lfs_objects, through: :lfs_objects_projects
157
  has_many :project_group_links
158
  has_many :invited_groups, through: :project_group_links, source: :group
159 160
  has_many :pages_domains
  has_many :todos
161
  has_many :notification_settings, as: :source, dependent: :delete_all # rubocop:disable Cop/ActiveRecordDependent
162

163
  has_one :import_data, class_name: 'ProjectImportData', inverse_of: :project, autosave: true
164
  has_one :project_feature, inverse_of: :project
165
  has_one :statistics, class_name: 'ProjectStatistics'
166

167 168 169
  # Container repositories need to remove data from the container registry,
  # which is not managed by the DB. Hence we're still using dependent: :destroy
  # here.
170
  has_many :container_repositories, dependent: :destroy # rubocop:disable Cop/ActiveRecordDependent
K
Kamil Trzcinski 已提交
171

172 173 174 175 176 177 178
  has_many :commit_statuses
  has_many :pipelines, class_name: 'Ci::Pipeline'

  # Ci::Build objects store data on the file system such as artifact files and
  # build traces. Currently there's no efficient way of removing this data in
  # bulk that doesn't involve loading the rows into memory. As a result we're
  # still using `dependent: :destroy` here.
179
  has_many :builds, class_name: 'Ci::Build', dependent: :destroy # rubocop:disable Cop/ActiveRecordDependent
180
  has_many :runner_projects, class_name: 'Ci::RunnerProject'
181
  has_many :runners, through: :runner_projects, source: :runner, class_name: 'Ci::Runner'
182
  has_many :variables, class_name: 'Ci::Variable'
183 184 185 186
  has_many :triggers, class_name: 'Ci::Trigger'
  has_many :environments
  has_many :deployments
  has_many :pipeline_schedules, class_name: 'Ci::PipelineSchedule'
K
Kamil Trzcinski 已提交
187

K
Kamil Trzcinski 已提交
188 189
  has_many :active_runners, -> { active }, through: :runner_projects, source: :runner, class_name: 'Ci::Runner'

190 191
  has_one :auto_devops, class_name: 'ProjectAutoDevops'

192
  accepts_nested_attributes_for :variables, allow_destroy: true
193
  accepts_nested_attributes_for :project_feature, update_only: true
194
  accepts_nested_attributes_for :import_data
195
  accepts_nested_attributes_for :auto_devops, update_only: true
196

197
  delegate :name, to: :owner, allow_nil: true, prefix: true
198
  delegate :members, to: :team, prefix: true
199
  delegate :add_user, :add_users, to: :team
200
  delegate :add_guest, :add_reporter, :add_developer, :add_master, to: :team
D
Douwe Maan 已提交
201
  delegate :empty_repo?, to: :repository
202

A
Andrey Kumanyaev 已提交
203
  # Validations
204
  validates :creator, presence: true, on: :create
205
  validates :description, length: { maximum: 2000 }, allow_blank: true
206
  validates :ci_config_path,
207
    format: { without: /\.{2}/,
208
              message: 'cannot include directory traversal.' },
209 210
    length: { maximum: 255 },
    allow_blank: true
211 212
  validates :name,
    presence: true,
213
    length: { maximum: 255 },
214
    format: { with: Gitlab::Regex.project_name_regex,
D
Douwe Maan 已提交
215
              message: Gitlab::Regex.project_name_regex_message }
216 217
  validates :path,
    presence: true,
218
    dynamic_path: true,
219
    length: { maximum: 255 },
220 221
    format: { with: Gitlab::PathRegex.project_path_format_regex,
              message: Gitlab::PathRegex.project_path_format_message },
222 223
    uniqueness: { scope: :namespace_id }

224
  validates :namespace, presence: true
D
Douwe Maan 已提交
225
  validates :name, uniqueness: { scope: :namespace_id }
J
James Lopez 已提交
226
  validates :import_url, addressable_url: true, if: :external_import?
D
Douwe Maan 已提交
227
  validates :import_url, importable_url: true, if: [:external_import?, :import_url_changed?]
228
  validates :star_count, numericality: { greater_than_or_equal_to: 0 }
229
  validate :check_limit, on: :create
230
  validate :can_create_repository?, on: [:create, :update], if: ->(project) { !project.persisted? || project.renamed? }
231
  validate :avatar_type,
232
    if: ->(project) { project.avatar.present? && project.avatar_changed? }
233
  validates :avatar, file_size: { maximum: 200.kilobytes.to_i }
D
Douwe Maan 已提交
234
  validate :visibility_level_allowed_by_group
D
Douwe Maan 已提交
235
  validate :visibility_level_allowed_as_fork
236
  validate :check_wiki_path_conflict
237 238 239
  validates :repository_storage,
    presence: true,
    inclusion: { in: ->(_object) { Gitlab.config.repositories.storages.keys } }
240

D
Douwe Maan 已提交
241
  mount_uploader :avatar, AvatarUploader
242
  has_many :uploads, as: :model, dependent: :destroy # rubocop:disable Cop/ActiveRecordDependent
A
Andrey Kumanyaev 已提交
243

244
  # Scopes
245
  scope :pending_delete, -> { where(pending_delete: true) }
246
  scope :without_deleted, -> { where(pending_delete: false) }
247

248 249 250
  scope :with_hashed_storage, -> { where('storage_version >= 1') }
  scope :with_legacy_storage, -> { where(storage_version: [nil, 0]) }

251
  scope :sorted_by_activity, -> { reorder(last_activity_at: :desc) }
252 253
  scope :sorted_by_stars, -> { reorder('projects.star_count DESC') }

254
  scope :in_namespace, ->(namespace_ids) { where(namespace_id: namespace_ids) }
D
Dmitriy Zaporozhets 已提交
255
  scope :personal, ->(user) { where(namespace_id: user.namespace_id) }
256
  scope :joined, ->(user) { where('namespace_id != ?', user.namespace_id) }
T
Toon Claes 已提交
257
  scope :starred_by, ->(user) { joins(:users_star_projects).where('users_star_projects.user_id': user.id) }
R
Rémy Coutable 已提交
258
  scope :visible_to_user, ->(user) { where(id: user.authorized_projects.select(:id).reorder(nil)) }
259
  scope :archived, -> { where(archived: true) }
260
  scope :non_archived, -> { where(archived: false) }
R
Rubén Dávila 已提交
261
  scope :for_milestones, ->(ids) { joins(:milestones).where('milestones.id' => ids).distinct }
R
Rémy Coutable 已提交
262 263
  scope :with_push, -> { joins(:events).where('events.action = ?', Event::PUSHED) }

264
  scope :with_project_feature, -> { joins('LEFT JOIN project_features ON projects.id = project_features.project_id') }
M
Markus Koller 已提交
265
  scope :with_statistics, -> { includes(:statistics) }
266
  scope :with_shared_runners, -> { where(shared_runners_enabled: true) }
267 268 269
  scope :inside_path, ->(path) do
    # We need routes alias rs for JOIN so it does not conflict with
    # includes(:route) which we use in ProjectsFinder.
270 271
    joins("INNER JOIN routes rs ON rs.source_id = projects.id AND rs.source_type = 'Project'")
      .where('rs.path LIKE ?', "#{sanitize_sql_like(path)}/%")
272
  end
273 274 275 276 277 278 279 280 281 282 283 284 285 286 287

  # "enabled" here means "not disabled". It includes private features!
  scope :with_feature_enabled, ->(feature) {
    access_level_attribute = ProjectFeature.access_level_attribute(feature)
    with_project_feature.where(project_features: { access_level_attribute => [nil, ProjectFeature::PRIVATE, ProjectFeature::ENABLED] })
  }

  # Picks a feature where the level is exactly that given.
  scope :with_feature_access_level, ->(feature, level) {
    access_level_attribute = ProjectFeature.access_level_attribute(feature)
    with_project_feature.where(project_features: { access_level_attribute => level })
  }

  scope :with_builds_enabled, -> { with_feature_enabled(:builds) }
  scope :with_issues_enabled, -> { with_feature_enabled(:issues) }
288
  scope :with_merge_requests_enabled, -> { with_feature_enabled(:merge_requests) }
289

290
  enum auto_cancel_pending_pipelines: { disabled: 0, enabled: 1 }
291

292 293 294 295
  # Returns a collection of projects that is either public or visible to the
  # logged in user.
  def self.public_or_visible_to_user(user = nil)
    if user
296 297 298 299
      authorized = user
        .project_authorizations
        .select(1)
        .where('project_authorizations.project_id = projects.id')
300 301 302 303 304 305 306 307 308

      levels = Gitlab::VisibilityLevel.levels_for_user(user)

      where('EXISTS (?) OR projects.visibility_level IN (?)', authorized, levels)
    else
      public_to_user
    end
  end

309 310 311
  # project features may be "disabled", "internal" or "enabled". If "internal",
  # they are only available to team members. This scope returns projects where
  # the feature is either enabled, or internal with permission for the user.
312 313 314 315
  #
  # This method uses an optimised version of `with_feature_access_level` for
  # logged in users to more efficiently get private projects with the given
  # feature.
316
  def self.with_feature_available_for_user(feature, user)
317 318 319 320 321 322 323
    visible = [nil, ProjectFeature::ENABLED]

    if user&.admin?
      with_feature_enabled(feature)
    elsif user
      column = ProjectFeature.quoted_access_level_column(feature)

324 325
      authorized = user.project_authorizations.select(1)
        .where('project_authorizations.project_id = projects.id')
326

327 328
      with_project_feature
        .where("#{column} IN (?) OR (#{column} = ? AND EXISTS (?))",
329 330 331 332 333 334
              visible,
              ProjectFeature::PRIVATE,
              authorized)
    else
      with_feature_access_level(feature, visible)
    end
335
  end
F
Felipe Artur 已提交
336

R
Rémy Coutable 已提交
337 338
  scope :active, -> { joins(:issues, :notes, :merge_requests).order('issues.created_at, notes.created_at, merge_requests.created_at DESC') }
  scope :abandoned, -> { where('projects.last_activity_at < ?', 6.months.ago) }
339

340 341
  scope :excluding_project, ->(project) { where.not(id: project) }

342
  state_machine :import_status, initial: :none do
343 344 345 346 347 348 349 350
    event :import_schedule do
      transition [:none, :finished, :failed] => :scheduled
    end

    event :force_import_start do
      transition [:none, :finished, :failed] => :started
    end

351
    event :import_start do
352
      transition scheduled: :started
353 354 355
    end

    event :import_finish do
356
      transition started: :finished
357 358 359
    end

    event :import_fail do
360
      transition [:scheduled, :started] => :failed
361 362 363
    end

    event :import_retry do
364
      transition failed: :started
365 366
    end

367
    state :scheduled
368 369
    state :started
    state :finished
370 371
    state :failed

372
    after_transition [:none, :finished, :failed] => :scheduled do |project, _|
373 374 375 376
      project.run_after_commit do
        job_id = add_import_job
        update(import_jid: job_id) if job_id
      end
377 378
    end

379 380
    after_transition started: :finished do |project, _|
      project.reset_cache_and_import_attrs
381 382 383

      if Gitlab::ImportSources.importer_names.include?(project.import_type) && project.repo_exists?
        project.run_after_commit do
L
Lin Jen-Shin 已提交
384
          Projects::AfterImportService.new(project).execute
385 386
        end
      end
387
    end
388 389
  end

A
Andrey Kumanyaev 已提交
390
  class << self
391 392 393 394 395 396 397
    # Searches for a list of projects based on the query given in `query`.
    #
    # On PostgreSQL this method uses "ILIKE" to perform a case-insensitive
    # search. On MySQL a regular "LIKE" is used as it's already
    # case-insensitive.
    #
    # query - The search query as a String.
398
    def search(query)
399
      ptable  = arel_table
400 401 402
      ntable  = Namespace.arel_table
      pattern = "%#{query}%"

M
mhasbini 已提交
403 404 405
      # unscoping unnecessary conditions that'll be applied
      # when executing `where("projects.id IN (#{union.to_sql})")`
      projects = unscoped.select(:id).where(
406 407 408
        ptable[:path].matches(pattern)
          .or(ptable[:name].matches(pattern))
          .or(ptable[:description].matches(pattern))
409 410
      )

411 412 413
      namespaces = unscoped.select(:id)
        .joins(:namespace)
        .where(ntable[:name].matches(pattern))
414 415 416

      union = Gitlab::SQL::Union.new([projects, namespaces])

417
      where("projects.id IN (#{union.to_sql})") # rubocop:disable GitlabSecurity/SqlInjection
A
Andrey Kumanyaev 已提交
418
    end
419

420
    def search_by_title(query)
421 422 423 424
      pattern = "%#{query}%"
      table   = Project.arel_table

      non_archived.where(table[:name].matches(pattern))
425 426
    end

427 428 429
    def visibility_levels
      Gitlab::VisibilityLevel.options
    end
430 431

    def sort(method)
432 433
      case method.to_s
      when 'storage_size_desc'
M
Markus Koller 已提交
434 435 436
        # storage_size is a joined column so we need to
        # pass a string to avoid AR adding the table name
        reorder('project_statistics.storage_size DESC, projects.id DESC')
437 438 439 440
      when 'latest_activity_desc'
        reorder(last_activity_at: :desc)
      when 'latest_activity_asc'
        reorder(last_activity_at: :asc)
441 442
      else
        order_by(method)
443 444
      end
    end
445 446

    def reference_pattern
447
      %r{
448 449
        ((?<namespace>#{Gitlab::PathRegex::FULL_NAMESPACE_FORMAT_REGEX})\/)?
        (?<project>#{Gitlab::PathRegex::PROJECT_PATH_FORMAT_REGEX})
450
      }x
451
    end
Y
Yorick Peterse 已提交
452

Y
Yorick Peterse 已提交
453
    def trending
454 455
      joins('INNER JOIN trending_projects ON projects.id = trending_projects.project_id')
        .reorder('trending_projects.id ASC')
Y
Yorick Peterse 已提交
456
    end
457 458 459 460 461 462

    def cached_count
      Rails.cache.fetch('total_project_count', expires_in: 5.minutes) do
        Project.count
      end
    end
463 464

    def group_ids
465
      joins(:namespace).where(namespaces: { type: 'Group' }).select(:namespace_id)
466
    end
467 468
  end

469
  def lfs_enabled?
470
    return namespace.lfs_enabled? if self[:lfs_enabled].nil?
P
Patricio Cano 已提交
471

472
    self[:lfs_enabled] && Gitlab.config.lfs.enabled
473 474
  end

475
  def auto_devops_enabled?
476
    if auto_devops&.enabled.nil?
Z
Zeger-Jan van de Weg 已提交
477
      current_application_settings.auto_devops_enabled?
478 479
    else
      auto_devops.enabled?
Z
Zeger-Jan van de Weg 已提交
480
    end
481 482
  end

483 484 485 486
  def has_auto_devops_implicitly_disabled?
    auto_devops&.enabled.nil? && !current_application_settings.auto_devops_enabled?
  end

487
  def repository_storage_path
488
    Gitlab.config.repositories.storages[repository_storage].try(:[], 'path')
489 490
  end

D
Dmitriy Zaporozhets 已提交
491
  def team
492
    @team ||= ProjectTeam.new(self)
D
Dmitriy Zaporozhets 已提交
493 494 495
  end

  def repository
496
    @repository ||= Repository.new(full_path, self, disk_path: disk_path)
497 498
  end

499 500 501 502
  def reload_repository!
    @repository = nil
  end

A
Andre Guedes 已提交
503
  def container_registry_url
K
Kamil Trzcinski 已提交
504
    if Gitlab.config.registry.enabled
505
      "#{Gitlab.config.registry.host_port}/#{full_path.downcase}"
506
    end
507 508
  end

509
  def has_container_registry_tags?
510 511 512
    return @images if defined?(@images)

    @images = container_repositories.to_a.any?(&:has_tags?) ||
513
      has_root_container_repository_tags?
514 515
  end

516 517
  def commit(ref = 'HEAD')
    repository.commit(ref)
D
Dmitriy Zaporozhets 已提交
518 519
  end

520
  # ref can't be HEAD, can only be branch/tag name or SHA
521
  def latest_successful_builds_for(ref = default_branch)
522
    latest_pipeline = pipelines.latest_successful_for(ref)
523 524 525 526 527 528

    if latest_pipeline
      latest_pipeline.builds.latest.with_artifacts
    else
      builds.none
    end
529 530
  end

531
  def merge_base_commit(first_commit_id, second_commit_id)
D
Douwe Maan 已提交
532 533
    sha = repository.merge_base(first_commit_id, second_commit_id)
    repository.commit(sha) if sha
534 535
  end

536
  def saved?
537
    id && persisted?
538 539
  end

540
  def add_import_job
D
Douwe Maan 已提交
541 542
    job_id =
      if forked?
543 544 545 546
        RepositoryForkWorker.perform_async(id,
                                           forked_from_project.repository_storage_path,
                                           forked_from_project.full_path,
                                           self.namespace.full_path)
D
Douwe Maan 已提交
547 548 549
      else
        RepositoryImportWorker.perform_async(self.id)
      end
550

551 552 553 554 555 556 557 558
    log_import_activity(job_id)

    job_id
  end

  def log_import_activity(job_id, type: :import)
    job_type = type.to_s.capitalize

559
    if job_id
560
      Rails.logger.info("#{job_type} job scheduled for #{full_path} with job ID #{job_id}.")
561
    else
562
      Rails.logger.error("#{job_type} job failed to create for #{full_path}.")
563
    end
564 565
  end

566 567 568 569 570
  def reset_cache_and_import_attrs
    run_after_commit do
      ProjectCacheWorker.perform_async(self.id)
    end

571
    update(import_error: nil)
572 573 574 575 576
    remove_import_data
  end

  # This method is overriden in EE::Project model
  def remove_import_data
577
    import_data&.destroy
578 579
  end

580
  def ci_config_path=(value)
581
    # Strip all leading slashes so that //foo -> foo
L
Lin Jen-Shin 已提交
582
    super(value&.sub(%r{\A/+}, '')&.delete("\0"))
583 584
  end

J
James Lopez 已提交
585
  def import_url=(value)
J
James Lopez 已提交
586 587
    return super(value) unless Gitlab::UrlSanitizer.valid?(value)

588
    import_url = Gitlab::UrlSanitizer.new(value)
J
James Lopez 已提交
589
    super(import_url.sanitized_url)
590
    create_or_update_import_data(credentials: import_url.credentials)
J
James Lopez 已提交
591 592 593
  end

  def import_url
J
James Lopez 已提交
594
    if import_data && super.present?
595
      import_url = Gitlab::UrlSanitizer.new(super, credentials: import_data.credentials)
J
James Lopez 已提交
596 597 598
      import_url.full_url
    else
      super
J
James Lopez 已提交
599 600
    end
  end
J
James Lopez 已提交
601

J
James Lopez 已提交
602
  def valid_import_url?
603
    valid?(:import_url) || errors.messages[:import_url].nil?
J
James Lopez 已提交
604 605
  end

606
  def create_or_update_import_data(data: nil, credentials: nil)
607
    return unless import_url.present? && valid_import_url?
608

J
James Lopez 已提交
609
    project_import_data = import_data || build_import_data
J
James Lopez 已提交
610 611 612 613
    if data
      project_import_data.data ||= {}
      project_import_data.data = project_import_data.data.merge(data)
    end
614 615 616 617
    if credentials
      project_import_data.credentials ||= {}
      project_import_data.credentials = project_import_data.credentials.merge(credentials)
    end
J
James Lopez 已提交
618
  end
J
James Lopez 已提交
619

620
  def import?
621
    external_import? || forked? || gitlab_project_import?
622 623
  end

624 625 626 627
  def no_import?
    import_status == 'none'
  end

628
  def external_import?
629 630 631
    import_url.present?
  end

632
  def imported?
633 634 635 636
    import_finished?
  end

  def import_in_progress?
637 638 639 640
    import_started? || import_scheduled?
  end

  def import_started?
641 642 643
    import? && import_status == 'started'
  end

644 645 646 647
  def import_scheduled?
    import_status == 'scheduled'
  end

648 649 650 651 652 653
  def import_failed?
    import_status == 'failed'
  end

  def import_finished?
    import_status == 'finished'
654 655
  end

D
Douwe Maan 已提交
656
  def safe_import_url
657
    Gitlab::UrlSanitizer.new(import_url).masked_url
D
Douwe Maan 已提交
658 659
  end

660 661 662 663
  def gitlab_project_import?
    import_type == 'gitlab_project'
  end

R
Rémy Coutable 已提交
664 665 666 667
  def gitea_import?
    import_type == 'gitea'
  end

668 669 670 671
  def github_import?
    import_type == 'github'
  end

672
  def check_limit
D
Douwe Maan 已提交
673
    unless creator.can_create_project? || namespace.kind == 'group'
674 675 676
      projects_limit = creator.projects_limit

      if projects_limit == 0
P
Phil Hughes 已提交
677
        self.errors.add(:limit_reached, "Personal project creation is not allowed. Please contact your administrator with questions")
678
      else
P
Phil Hughes 已提交
679
        self.errors.add(:limit_reached, "Your project limit is #{projects_limit} projects! Please contact your administrator to increase it")
680
      end
681 682
    end
  rescue
D
Douwe Maan 已提交
683
    self.errors.add(:base, "Can't check your ability to create project")
G
gitlabhq 已提交
684 685
  end

D
Douwe Maan 已提交
686 687 688 689 690 691 692 693 694 695 696 697 698
  def visibility_level_allowed_by_group
    return if visibility_level_allowed_by_group?

    level_name = Gitlab::VisibilityLevel.level_name(self.visibility_level).downcase
    group_level_name = Gitlab::VisibilityLevel.level_name(self.group.visibility_level).downcase
    self.errors.add(:visibility_level, "#{level_name} is not allowed in a #{group_level_name} group.")
  end

  def visibility_level_allowed_as_fork
    return if visibility_level_allowed_as_fork?

    level_name = Gitlab::VisibilityLevel.level_name(self.visibility_level).downcase
    self.errors.add(:visibility_level, "#{level_name} is not allowed since the fork source project has lower visibility.")
699 700
  end

701 702 703 704 705 706 707 708 709 710
  def check_wiki_path_conflict
    return if path.blank?

    path_to_check = path.ends_with?('.wiki') ? path.chomp('.wiki') : "#{path}.wiki"

    if Project.where(namespace_id: namespace_id, path: path_to_check).exists?
      errors.add(:name, 'has already been taken')
    end
  end

711
  def to_param
712 713 714 715 716
    if persisted? && errors.include?(:path)
      path_was
    else
      path
    end
717 718
  end

719
  # `from` argument can be a Namespace or Project.
720 721
  def to_reference(from = nil, full: false)
    if full || cross_namespace_reference?(from)
722
      full_path
723 724 725
    elsif cross_project_reference?(from)
      path
    end
726 727 728 729 730 731 732 733
  end

  def to_human_reference(from_project = nil)
    if cross_namespace_reference?(from_project)
      name_with_namespace
    elsif cross_project_reference?(from_project)
      name
    end
734 735
  end

736
  def web_url
737
    Gitlab::Routing.url_helpers.project_url(self)
738 739
  end

740
  def new_issue_address(author)
741
    return unless Gitlab::IncomingEmail.supports_issue_creation? && author
742

743 744 745
    author.ensure_incoming_email_token!

    Gitlab::IncomingEmail.reply_address(
746
      "#{full_path}+#{author.incoming_email_token}")
747 748
  end

749
  def build_commit_note(commit)
750
    notes.new(commit_id: commit.id, noteable_type: 'Commit')
G
gitlabhq 已提交
751
  end
N
Nihad Abbasov 已提交
752

N
Nihad Abbasov 已提交
753
  def last_activity
754
    last_event
G
gitlabhq 已提交
755 756 757
  end

  def last_activity_date
758
    last_repository_updated_at || last_activity_at || updated_at
D
Dmitriy Zaporozhets 已提交
759
  end
760

D
Dmitriy Zaporozhets 已提交
761 762 763
  def project_id
    self.id
  end
R
randx 已提交
764

765
  def get_issue(issue_id, current_user)
766 767 768 769 770
    issue = IssuesFinder.new(current_user, project_id: id).find_by(iid: issue_id) if issues_enabled?

    if issue
      issue
    elsif external_issue_tracker
R
Robert Speicher 已提交
771
      ExternalIssue.new(issue_id, self)
772 773 774
    end
  end

R
Robert Speicher 已提交
775
  def issue_exists?(issue_id)
776
    get_issue(issue_id)
R
Robert Speicher 已提交
777 778
  end

779
  def default_issue_tracker
780
    gitlab_issue_tracker_service || create_gitlab_issue_tracker_service
781 782 783 784 785 786 787 788 789 790
  end

  def issues_tracker
    if external_issue_tracker
      external_issue_tracker
    else
      default_issue_tracker
    end
  end

791
  def external_issue_reference_pattern
792
    external_issue_tracker.class.reference_pattern(only_long: issues_enabled?)
793 794
  end

795
  def default_issues_tracker?
796
    !external_issue_tracker
797 798 799
  end

  def external_issue_tracker
800 801 802 803 804 805 806 807 808 809 810 811 812 813 814
    if has_external_issue_tracker.nil? # To populate existing projects
      cache_has_external_issue_tracker
    end

    if has_external_issue_tracker?
      return @external_issue_tracker if defined?(@external_issue_tracker)

      @external_issue_tracker = services.external_issue_trackers.first
    else
      nil
    end
  end

  def cache_has_external_issue_tracker
    update_column(:has_external_issue_tracker, services.external_issue_trackers.any?)
815 816
  end

817 818 819 820
  def has_wiki?
    wiki_enabled? || has_external_wiki?
  end

821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836
  def external_wiki
    if has_external_wiki.nil?
      cache_has_external_wiki # Populate
    end

    if has_external_wiki
      @external_wiki ||= services.external_wikis.first
    else
      nil
    end
  end

  def cache_has_external_wiki
    update_column(:has_external_wiki, services.external_wikis.any?)
  end

837
  def find_or_initialize_services(exceptions: [])
M
Marin Jankovski 已提交
838 839
    services_templates = Service.where(template: true)

840 841 842
    available_services_names = Service.available_services_names - exceptions

    available_services_names.map do |service_name|
M
Marin Jankovski 已提交
843
      service = find_service(services, service_name)
844

845 846 847
      if service
        service
      else
M
Marin Jankovski 已提交
848 849 850 851
        # We should check if template for the service exists
        template = find_service(services_templates, service_name)

        if template.nil?
852
          # If no template, we should create an instance. Ex `build_gitlab_ci_service`
853
          public_send("build_#{service_name}_service") # rubocop:disable GitlabSecurity/PublicSend
M
Marin Jankovski 已提交
854
        else
855
          Service.build_from_template(id, template)
M
Marin Jankovski 已提交
856 857
        end
      end
858 859 860
    end
  end

861 862 863 864
  def find_or_initialize_service(name)
    find_or_initialize_services.find { |service| service.to_param == name }
  end

V
Valery Sizov 已提交
865 866
  def create_labels
    Label.templates.each do |label|
867
      params = label.attributes.except('id', 'template', 'created_at', 'updated_at')
868
      Labels::FindOrCreateService.new(nil, self, params).execute(skip_authorization: true)
V
Valery Sizov 已提交
869 870 871
    end
  end

M
Marin Jankovski 已提交
872 873 874
  def find_service(list, name)
    list.find { |service| service.to_param == name }
  end
D
Dmitriy Zaporozhets 已提交
875

876
  def ci_services
Y
Yorick Peterse 已提交
877
    services.where(category: :ci)
878 879 880
  end

  def ci_service
881
    @ci_service ||= ci_services.reorder(nil).find_by(active: true)
882 883
  end

884 885 886 887 888
  def deployment_services
    services.where(category: :deployment)
  end

  def deployment_service
889
    @deployment_service ||= deployment_services.reorder(nil).find_by(active: true)
890 891
  end

892 893 894 895 896
  def monitoring_services
    services.where(category: :monitoring)
  end

  def monitoring_service
897
    @monitoring_service ||= monitoring_services.reorder(nil).find_by(active: true)
898 899
  end

D
Drew Blessing 已提交
900 901 902 903
  def jira_tracker?
    issues_tracker.to_param == 'jira'
  end

904
  def avatar_type
905 906
    unless self.avatar.image?
      self.errors.add :avatar, 'only images allowed'
907 908 909 910
    end
  end

  def avatar_in_git
911
    repository.avatar
912 913
  end

914 915 916
  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
917
    avatar_path(args) || (Gitlab::Routing.url_helpers.project_avatar_url(self) if avatar_in_git)
S
sue445 已提交
918 919
  end

920 921 922 923 924
  # For compatibility with old code
  def code
    path
  end

925
  def items_for(entity)
D
Dmitriy Zaporozhets 已提交
926 927 928 929 930 931 932
    case entity
    when 'issue' then
      issues
    when 'merge_request' then
      merge_requests
    end
  end
933

934
  def send_move_instructions(old_path_with_namespace)
935 936 937
    # New project path needs to be committed to the DB or notification will
    # retrieve stale information
    run_after_commit { NotificationService.new.project_was_moved(self, old_path_with_namespace) }
938
  end
939 940

  def owner
941 942
    if group
      group
943
    else
944
      namespace.try(:owner)
945 946
    end
  end
D
Dmitriy Zaporozhets 已提交
947

948
  def execute_hooks(data, hooks_scope = :push_hooks)
949
    hooks.public_send(hooks_scope).each do |hook| # rubocop:disable GitlabSecurity/PublicSend
950
      hook.async_execute(data, hooks_scope.to_s)
951
    end
D
Dmitriy Zaporozhets 已提交
952 953
  end

954 955
  def execute_services(data, hooks_scope = :push_hooks)
    # Call only service hooks that are active for this scope
956
    services.public_send(hooks_scope).each do |service| # rubocop:disable GitlabSecurity/PublicSend
957
      service.async_execute(data)
D
Dmitriy Zaporozhets 已提交
958 959 960 961
    end
  end

  def valid_repo?
962
    repository.exists?
D
Dmitriy Zaporozhets 已提交
963
  rescue
964
    errors.add(:path, 'Invalid repository path')
D
Dmitriy Zaporozhets 已提交
965 966 967 968
    false
  end

  def repo
J
Jacob Vosmaer 已提交
969
    repository.rugged
D
Dmitriy Zaporozhets 已提交
970 971 972
  end

  def url_to_repo
973
    gitlab_shell.url_to_repo(full_path)
D
Dmitriy Zaporozhets 已提交
974 975 976
  end

  def repo_exists?
977
    @repo_exists ||= repository.exists?
D
Dmitriy Zaporozhets 已提交
978 979 980 981 982
  rescue
    @repo_exists = false
  end

  def root_ref?(branch)
D
Dmitriy Zaporozhets 已提交
983
    repository.root_ref == branch
D
Dmitriy Zaporozhets 已提交
984 985 986 987 988 989
  end

  def ssh_url_to_repo
    url_to_repo
  end

990 991
  def http_url_to_repo
    "#{web_url}.git"
D
Dmitriy Zaporozhets 已提交
992 993
  end

994
  def user_can_push_to_empty_repo?(user)
995
    !ProtectedBranch.default_branch_protected? || team.max_member_access(user.id) > Gitlab::Access::DEVELOPER
996 997
  end

998 999 1000
  def forked?
    !(forked_project_link.nil? || forked_project_link.forked_from_project.nil?)
  end
D
Dmitriy Zaporozhets 已提交
1001

1002 1003 1004 1005
  def personal?
    !group
  end

1006 1007 1008 1009 1010 1011
  # Expires various caches before a project is renamed.
  def expire_caches_before_rename(old_path)
    repo = Repository.new(old_path, self)
    wiki = Repository.new("#{old_path}.wiki", self)

    if repo.exists?
1012
      repo.before_delete
1013 1014 1015
    end

    if wiki.exists?
1016
      wiki.before_delete
1017 1018 1019
    end
  end

1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033
  # Check if repository already exists on disk
  def can_create_repository?
    return false unless repository_storage_path

    expires_full_path_cache # we need to clear cache to validate renames correctly

    if gitlab_shell.exists?(repository_storage_path, "#{disk_path}.git")
      errors.add(:base, 'There is already a repository with that name on disk')
      return false
    end

    true
  end

1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046
  def create_repository(force: false)
    # Forked import is handled asynchronously
    return if forked? && !force

    if gitlab_shell.add_repository(repository_storage_path, disk_path)
      repository.after_create
      true
    else
      errors.add(:base, 'Failed to create repository via gitlab-shell')
      false
    end
  end

1047 1048
  def hook_attrs(backward: true)
    attrs = {
K
Kirill Zaitsev 已提交
1049
      name: name,
1050
      description: description,
K
Kirilll Zaitsev 已提交
1051
      web_url: web_url,
1052
      avatar_url: avatar_url(only_path: false),
1053 1054
      git_ssh_url: ssh_url_to_repo,
      git_http_url: http_url_to_repo,
K
Kirill Zaitsev 已提交
1055
      namespace: namespace.name,
1056
      visibility_level: visibility_level,
1057
      path_with_namespace: full_path,
1058
      default_branch: default_branch,
1059
      ci_config_path: ci_config_path
K
Kirill Zaitsev 已提交
1060
    }
1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072

    # Backward compatibility
    if backward
      attrs.merge!({
                    homepage: web_url,
                    url: url_to_repo,
                    ssh_url: ssh_url_to_repo,
                    http_url: http_url_to_repo
                  })
    end

    attrs
K
Kirill Zaitsev 已提交
1073 1074
  end

1075
  def project_member(user)
G
Gabriel Mazetto 已提交
1076
    project_members.find_by(user_id: user)
1077
  end
1078 1079 1080 1081

  def default_branch
    @default_branch ||= repository.root_ref if repository.exists?
  end
1082 1083 1084 1085 1086

  def reload_default_branch
    @default_branch = nil
    default_branch
  end
1087

1088
  def visibility_level_field
1089
    :visibility_level
1090
  end
1091 1092 1093 1094 1095 1096 1097 1098

  def archive!
    update_attribute(:archived, true)
  end

  def unarchive!
    update_attribute(:archived, false)
  end
1099

1100
  def change_head(branch)
1101 1102
    if repository.branch_exists?(branch)
      repository.before_change_head
1103
      repository.write_ref('HEAD', "refs/heads/#{branch}")
1104 1105 1106 1107 1108 1109 1110
      repository.copy_gitattributes(branch)
      repository.after_change_head
      reload_default_branch
    else
      errors.add(:base, "Could not change HEAD: branch '#{branch}' does not exist")
      false
    end
1111
  end
1112 1113 1114 1115

  def forked_from?(project)
    forked? && project == forked_from_project
  end
1116

1117 1118 1119
  def origin_merge_requests
    merge_requests.where(source_project_id: self.id)
  end
1120

1121
  def ensure_repository
1122
    create_repository(force: true) unless repository_exists?
1123 1124
  end

1125 1126 1127 1128
  def repository_exists?
    !!repository.exists?
  end

1129
  # update visibility_level of forks
1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140
  def update_forks_visibility_level
    return unless visibility_level < visibility_level_was

    forks.each do |forked_project|
      if forked_project.visibility_level > visibility_level
        forked_project.visibility_level = visibility_level
        forked_project.save!
      end
    end
  end

1141 1142 1143
  def create_wiki
    ProjectWiki.new(self, self.owner).wiki
    true
G
Guilherme Garnier 已提交
1144
  rescue ProjectWiki::CouldNotCreateWikiError
1145
    errors.add(:base, 'Failed create wiki')
1146 1147
    false
  end
1148

1149 1150 1151 1152
  def wiki
    @wiki ||= ProjectWiki.new(self, self.owner)
  end

D
Drew Blessing 已提交
1153 1154 1155 1156
  def jira_tracker_active?
    jira_tracker? && jira_service.active
  end

1157
  def allowed_to_share_with_group?
1158
    !namespace.share_with_group_lock
1159 1160
  end

1161 1162 1163
  def pipeline_for(ref, sha = nil)
    sha ||= commit(ref).try(:sha)

1164
    return unless sha
1165

1166
    pipelines.order(id: :desc).find_by(sha: sha, ref: ref)
K
Kamil Trzcinski 已提交
1167 1168
  end

1169 1170 1171 1172 1173 1174 1175 1176 1177 1178 1179 1180 1181 1182 1183 1184 1185
  def latest_successful_pipeline_for_default_branch
    if defined?(@latest_successful_pipeline_for_default_branch)
      return @latest_successful_pipeline_for_default_branch
    end

    @latest_successful_pipeline_for_default_branch =
      pipelines.latest_successful_for(default_branch)
  end

  def latest_successful_pipeline_for(ref = nil)
    if ref && ref != default_branch
      pipelines.latest_successful_for(ref)
    else
      latest_successful_pipeline_for_default_branch
    end
  end

1186
  def enable_ci
F
Felipe Artur 已提交
1187
    project_feature.update_attribute(:builds_access_level, ProjectFeature::ENABLED)
1188
  end
M
Marin Jankovski 已提交
1189

1190 1191 1192 1193 1194
  def shared_runners_available?
    shared_runners_enabled?
  end

  def shared_runners
1195
    @shared_runners ||= shared_runners_available? ? Ci::Runner.shared : Ci::Runner.none
1196 1197
  end

1198 1199
  def active_shared_runners
    @active_shared_runners ||= shared_runners.active
1200
  end
K
Kamil Trzcinski 已提交
1201 1202

  def any_runners?(&block)
K
Kamil Trzcinski 已提交
1203
    active_runners.any?(&block) || active_shared_runners.any?(&block)
K
Kamil Trzcinski 已提交
1204 1205
  end

1206
  def valid_runners_token?(token)
J
James Lopez 已提交
1207
    self.runners_token && ActiveSupport::SecurityUtils.variable_size_secure_compare(token, self.runners_token)
K
Kamil Trzcinski 已提交
1208 1209
  end

K
Kamil Trzcinski 已提交
1210 1211 1212 1213 1214 1215 1216
  def build_timeout_in_minutes
    build_timeout / 60
  end

  def build_timeout_in_minutes=(value)
    self.build_timeout = value.to_i * 60
  end
1217

S
Stan Hu 已提交
1218
  def open_issues_count
1219 1220 1221 1222 1223
    Projects::OpenIssuesCountService.new(self).count
  end

  def open_merge_requests_count
    Projects::OpenMergeRequestsCountService.new(self).count
S
Stan Hu 已提交
1224
  end
1225

D
Douwe Maan 已提交
1226
  def visibility_level_allowed_as_fork?(level = self.visibility_level)
D
Douwe Maan 已提交
1227
    return true unless forked?
D
Douwe Maan 已提交
1228

D
Douwe Maan 已提交
1229 1230 1231 1232 1233 1234
    # self.forked_from_project will be nil before the project is saved, so
    # we need to go through the relation
    original_project = forked_project_link.forked_from_project
    return true unless original_project

    level <= original_project.visibility_level
D
Douwe Maan 已提交
1235
  end
1236

D
Douwe Maan 已提交
1237 1238
  def visibility_level_allowed_by_group?(level = self.visibility_level)
    return true unless group
1239

D
Douwe Maan 已提交
1240
    level <= group.visibility_level
M
Marin Jankovski 已提交
1241
  end
1242

D
Douwe Maan 已提交
1243 1244
  def visibility_level_allowed?(level = self.visibility_level)
    visibility_level_allowed_as_fork?(level) && visibility_level_allowed_by_group?(level)
F
Felipe Artur 已提交
1245 1246
  end

1247 1248 1249
  def runners_token
    ensure_runners_token!
  end
V
Valery Sizov 已提交
1250

1251 1252 1253
  def pages_deployed?
    Dir.exist?(public_pages_path)
  end
1254

1255
  def pages_url
1256 1257
    subdomain, _, url_path = full_path.partition('/')

K
Kamil Trzcinski 已提交
1258 1259
    # The hostname always needs to be in downcased
    # All web servers convert hostname to lowercase
1260
    host = "#{subdomain}.#{Settings.pages.host}".downcase
K
Kamil Trzcinski 已提交
1261 1262

    # The host in URL always needs to be downcased
1263
    url = Gitlab.config.pages.url.sub(/^https?:\/\//) do |prefix|
1264
      "#{prefix}#{subdomain}."
K
Kamil Trzcinski 已提交
1265
    end.downcase
1266

K
Kamil Trzcinski 已提交
1267
    # If the project path is the same as host, we serve it as group page
1268 1269 1270 1271
    return url if host == url_path

    "#{url}/#{url_path}"
  end
1272

1273 1274
  def pages_subdomain
    full_path.partition('/').first
1275
  end
K
Kamil Trzcinski 已提交
1276 1277

  def pages_path
1278 1279
    # TODO: when we migrate Pages to work with new storage types, change here to use disk_path
    File.join(Settings.pages.path, full_path)
K
Kamil Trzcinski 已提交
1280 1281 1282 1283 1284 1285
  end

  def public_pages_path
    File.join(pages_path, 'public')
  end

1286 1287 1288 1289
  def pages_available?
    Gitlab.config.pages.enabled && !namespace.subgroup?
  end

1290
  def remove_private_deploy_keys
1291 1292 1293 1294 1295 1296 1297 1298 1299 1300 1301 1302
    exclude_keys_linked_to_other_projects = <<-SQL
      NOT EXISTS (
        SELECT 1
        FROM deploy_keys_projects dkp2
        WHERE dkp2.deploy_key_id = deploy_keys_projects.deploy_key_id
        AND dkp2.project_id != deploy_keys_projects.project_id
      )
    SQL

    deploy_keys.where(public: false)
               .where(exclude_keys_linked_to_other_projects)
               .delete_all
1303 1304
  end

1305
  # TODO: what to do here when not using Legacy Storage? Do we still need to rename and delay removal?
K
Kamil Trzcinski 已提交
1306
  def remove_pages
1307 1308 1309
    # Projects with a missing namespace cannot have their pages removed
    return unless namespace

1310 1311
    ::Projects::UpdatePagesConfigurationService.new(self).execute

1312 1313 1314
    # 1. We rename pages to temporary directory
    # 2. We wait 5 minutes, due to NFS caching
    # 3. We asynchronously remove pages with force
K
Kamil Trzcinski 已提交
1315
    temp_path = "#{path}.#{SecureRandom.hex}.deleted"
K
Kamil Trzcinski 已提交
1316

1317 1318
    if Gitlab::PagesTransfer.new.rename_project(path, temp_path, namespace.full_path)
      PagesWorker.perform_in(5.minutes, :remove, namespace.full_path, temp_path)
K
Kamil Trzcinski 已提交
1319
    end
K
Kamil Trzcinski 已提交
1320 1321
  end

1322 1323 1324 1325 1326 1327 1328 1329 1330 1331 1332 1333 1334 1335 1336 1337 1338 1339 1340 1341 1342 1343 1344 1345 1346 1347 1348 1349 1350 1351 1352 1353 1354 1355 1356 1357 1358 1359 1360 1361 1362 1363 1364 1365
  def rename_repo
    new_full_path = build_full_path

    Rails.logger.error "Attempting to rename #{full_path_was} -> #{new_full_path}"

    if has_container_registry_tags?
      Rails.logger.error "Project #{full_path_was} cannot be renamed because container registry tags are present!"

      # we currently doesn't support renaming repository if it contains images in container registry
      raise StandardError.new('Project cannot be renamed, because images are present in its container registry')
    end

    expire_caches_before_rename(full_path_was)

    if storage.rename_repo
      Gitlab::AppLogger.info "Project was renamed: #{full_path_was} -> #{new_full_path}"
      rename_repo_notify!
      after_rename_repo
    else
      Rails.logger.error "Repository could not be renamed: #{full_path_was} -> #{new_full_path}"

      # if we cannot move namespace directory we should rollback
      # db changes in order to prevent out of sync between db and fs
      raise StandardError.new('repository cannot be renamed')
    end
  end

  def rename_repo_notify!
    send_move_instructions(full_path_was)
    expires_full_path_cache

    self.old_path_with_namespace = full_path_was
    SystemHooksService.new.execute_hooks_for(self, :rename)

    reload_repository!
  end

  def after_rename_repo
    path_before_change = previous_changes['path'].first

    Gitlab::UploadsTransfer.new.rename_project(path_before_change, self.path, namespace.full_path)
    Gitlab::PagesTransfer.new.rename_project(path_before_change, self.path, namespace.full_path)
  end

J
Josh Frye 已提交
1366 1367
  def running_or_pending_build_count(force: false)
    Rails.cache.fetch(['projects', id, 'running_or_pending_build_count'], force: force) do
1368 1369 1370
      builds.running_or_pending.count(:all)
    end
  end
J
James Lopez 已提交
1371

1372
  # Lazy loading of the `pipeline_status` attribute
1373
  def pipeline_status
1374
    @pipeline_status ||= Gitlab::Cache::Ci::ProjectPipelineStatus.load_for_project(self)
1375 1376
  end

J
James Lopez 已提交
1377
  def mark_import_as_failed(error_message)
1378 1379 1380
    original_errors = errors.dup
    sanitized_message = Gitlab::UrlSanitizer.sanitize(error_message)

J
James Lopez 已提交
1381
    import_fail
1382 1383 1384 1385 1386
    update_column(:import_error, sanitized_message)
  rescue ActiveRecord::ActiveRecordError => e
    Rails.logger.error("Error setting import status to failed: #{e.message}. Original error: #{sanitized_message}")
  ensure
    @errors = original_errors
J
James Lopez 已提交
1387
  end
J
James Lopez 已提交
1388

1389 1390
  def add_export_job(current_user:)
    job_id = ProjectExportWorker.perform_async(current_user.id, self.id)
1391 1392 1393 1394 1395 1396 1397

    if job_id
      Rails.logger.info "Export job started for project ID #{self.id} with job ID #{job_id}"
    else
      Rails.logger.error "Export job failed to start for project ID #{self.id}"
    end
  end
J
James Lopez 已提交
1398 1399

  def export_path
1400
    File.join(Gitlab::ImportExport.storage_path, disk_path)
J
James Lopez 已提交
1401
  end
1402 1403 1404 1405 1406 1407 1408 1409 1410

  def export_project_path
    Dir.glob("#{export_path}/*export.tar.gz").max_by { |f| File.ctime(f) }
  end

  def remove_exports
    _, status = Gitlab::Popen.popen(%W(find #{export_path} -not -path #{export_path} -delete))
    status.zero?
  end
1411

V
vanadium23 已提交
1412 1413 1414 1415
  def full_path_slug
    Gitlab::Utils.slugify(full_path.to_s)
  end

K
Kamil Trzcinski 已提交
1416
  def has_ci?
1417
    repository.gitlab_ci_yml || auto_devops_enabled?
K
Kamil Trzcinski 已提交
1418 1419
  end

1420
  def predefined_variables
1421
    [
1422 1423
      { key: 'CI_PROJECT_ID', value: id.to_s, public: true },
      { key: 'CI_PROJECT_NAME', value: path, public: true },
1424
      { key: 'CI_PROJECT_PATH', value: full_path, public: true },
V
vanadium23 已提交
1425
      { key: 'CI_PROJECT_PATH_SLUG', value: full_path_slug, public: true },
1426
      { key: 'CI_PROJECT_NAMESPACE', value: namespace.full_path, public: true },
1427
      { key: 'CI_PROJECT_URL', value: web_url, public: true }
1428 1429 1430 1431 1432 1433 1434 1435 1436 1437
    ]
  end

  def container_registry_variables
    return [] unless Gitlab.config.registry.enabled

    variables = [
      { key: 'CI_REGISTRY', value: Gitlab.config.registry.host_port, public: true }
    ]

K
Kamil Trzcinski 已提交
1438
    if container_registry_enabled?
A
Andre Guedes 已提交
1439
      variables << { key: 'CI_REGISTRY_IMAGE', value: container_registry_url, public: true }
K
Kamil Trzcinski 已提交
1440 1441
    end

1442 1443 1444
    variables
  end

L
Lin Jen-Shin 已提交
1445 1446
  def secret_variables_for(ref:, environment: nil)
    # EE would use the environment
1447 1448 1449 1450 1451 1452
    if protected_for?(ref)
      variables
    else
      variables.unprotected
    end
  end
1453

1454 1455 1456
  def protected_for?(ref)
    ProtectedBranch.protected?(self, ref) ||
      ProtectedTag.protected?(self, ref)
1457
  end
1458

1459 1460 1461 1462 1463 1464
  def deployment_variables
    return [] unless deployment_service

    deployment_service.predefined_variables
  end

1465 1466 1467 1468 1469 1470
  def auto_devops_variables
    return [] unless auto_devops_enabled?

    auto_devops&.variables || []
  end

1471
  def append_or_update_attribute(name, value)
1472
    old_values = public_send(name.to_s) # rubocop:disable GitlabSecurity/PublicSend
1473 1474 1475 1476 1477 1478

    if Project.reflect_on_association(name).try(:macro) == :has_many && old_values.any?
      update_attribute(name, old_values + value)
    else
      update_attribute(name, value)
    end
1479 1480 1481

  rescue ActiveRecord::RecordNotSaved => e
    handle_update_attribute_error(e, value)
1482 1483
  end

Y
Yorick Peterse 已提交
1484
  def pushes_since_gc
1485
    Gitlab::Redis::SharedState.with { |redis| redis.get(pushes_since_gc_redis_shared_state_key).to_i }
Y
Yorick Peterse 已提交
1486 1487 1488
  end

  def increment_pushes_since_gc
1489
    Gitlab::Redis::SharedState.with { |redis| redis.incr(pushes_since_gc_redis_shared_state_key) }
Y
Yorick Peterse 已提交
1490 1491 1492
  end

  def reset_pushes_since_gc
1493
    Gitlab::Redis::SharedState.with { |redis| redis.del(pushes_since_gc_redis_shared_state_key) }
Y
Yorick Peterse 已提交
1494 1495
  end

D
Douwe Maan 已提交
1496
  def route_map_for(commit_sha)
1497 1498
    @route_maps_by_commit ||= Hash.new do |h, sha|
      h[sha] = begin
D
Douwe Maan 已提交
1499
        data = repository.route_map_for(sha)
1500 1501
        next unless data

D
Douwe Maan 已提交
1502 1503 1504
        Gitlab::RouteMap.new(data)
      rescue Gitlab::RouteMap::FormatError
        nil
1505 1506 1507 1508 1509 1510 1511
      end
    end

    @route_maps_by_commit[commit_sha]
  end

  def public_path_for_source_path(path, commit_sha)
D
Douwe Maan 已提交
1512
    map = route_map_for(commit_sha)
1513 1514
    return unless map

D
Douwe Maan 已提交
1515
    map.public_path_for_source_path(path)
1516 1517
  end

1518 1519 1520 1521 1522 1523 1524 1525
  def parent
    namespace
  end

  def parent_changed?
    namespace_id_changed?
  end

1526 1527 1528 1529 1530 1531 1532 1533
  def default_merge_request_target
    if forked_from_project&.merge_requests_enabled?
      forked_from_project
    else
      self
    end
  end

F
Felipe Artur 已提交
1534 1535 1536 1537 1538 1539 1540 1541
  def multiple_issue_boards_available?(user)
    feature_available?(:multiple_issue_boards, user)
  end

  def issue_board_milestone_available?(user = nil)
    feature_available?(:issue_board_milestone, user)
  end

1542 1543 1544 1545
  def full_path_was
    File.join(namespace.full_path, previous_changes['path'].first)
  end

1546 1547
  alias_method :name_with_namespace, :full_name
  alias_method :human_name, :full_name
1548
  # @deprecated cannot remove yet because it has an index with its name in elasticsearch
1549 1550
  alias_method :path_with_namespace, :full_path

1551 1552 1553 1554
  def forks_count
    Projects::ForksCountService.new(self).count
  end

1555
  def legacy_storage?
1556 1557 1558 1559 1560
    [nil, 0].include?(self.storage_version)
  end

  def hashed_storage?
    self.storage_version && self.storage_version >= 1
1561 1562
  end

1563 1564 1565 1566
  def renamed?
    persisted? && path_changed?
  end

1567 1568 1569 1570 1571 1572 1573 1574 1575 1576 1577 1578 1579 1580 1581 1582 1583 1584 1585 1586 1587 1588
  def migrate_to_hashed_storage!
    return if hashed_storage?

    update!(repository_read_only: true)

    if repo_reference_count > 0 || wiki_reference_count > 0
      ProjectMigrateHashedStorageWorker.perform_in(Gitlab::ReferenceCounter::REFERENCE_EXPIRE_TIME, id)
    else
      ProjectMigrateHashedStorageWorker.perform_async(id)
    end
  end

  def storage_version=(value)
    super

    @storage = nil if storage_version_changed?
  end

  def gl_repository(is_wiki:)
    Gitlab::GlRepository.gl_repository(self, is_wiki)
  end

1589 1590
  private

1591 1592
  def storage
    @storage ||=
1593
      if hashed_storage?
1594 1595 1596 1597 1598
        Storage::HashedProject.new(self)
      else
        Storage::LegacyProject.new(self)
      end
  end
1599

1600
  def use_hashed_storage
1601
    if self.new_record? && current_application_settings.hashed_storage_enabled
1602
      self.storage_version = LATEST_STORAGE_VERSION
G
Gabriel Mazetto 已提交
1603 1604 1605
    end
  end

1606 1607 1608 1609 1610 1611 1612 1613
  def repo_reference_count
    Gitlab::ReferenceCounter.new(gl_repository(is_wiki: false)).value
  end

  def wiki_reference_count
    Gitlab::ReferenceCounter.new(gl_repository(is_wiki: true)).value
  end

1614 1615 1616 1617 1618 1619 1620 1621 1622
  # set last_activity_at to the same as created_at
  def set_last_activity_at
    update_column(:last_activity_at, self.created_at)
  end

  def set_last_repository_updated_at
    update_column(:last_repository_updated_at, self.created_at)
  end

1623
  def cross_namespace_reference?(from)
1624 1625 1626 1627 1628
    case from
    when Project
      namespace != from.namespace
    when Namespace
      namespace != from
1629 1630 1631
    end
  end

1632
  # Check if a reference is being done cross-project
1633 1634 1635 1636
  def cross_project_reference?(from)
    return true if from.is_a?(Namespace)

    from && self != from
1637 1638
  end

1639
  def pushes_since_gc_redis_shared_state_key
Y
Yorick Peterse 已提交
1640 1641 1642
    "projects/#{id}/pushes_since_gc"
  end

1643 1644 1645 1646 1647 1648 1649
  # Similar to the normal callbacks that hook into the life cycle of an
  # Active Record object, you can also define callbacks that get triggered
  # when you add an object to an association collection. If any of these
  # callbacks throw an exception, the object will not be added to the
  # collection. Before you add a new board to the boards collection if you
  # already have 1, 2, or n it will fail, but it if you have 0 that is lower
  # than the number of permitted boards per project it won't fail.
1650
  def validate_board_limit(board)
1651
    raise BoardLimitExceeded, 'Number of permitted boards exceeded' if boards.size >= NUMBER_OF_PERMITTED_BOARDS
1652
  end
1653

M
Markus Koller 已提交
1654 1655 1656 1657
  def update_project_statistics
    stats = statistics || build_statistics
    stats.update(namespace_id: namespace_id)
  end
1658 1659 1660 1661 1662 1663 1664 1665 1666 1667 1668 1669 1670 1671 1672

  def check_pending_delete
    return if valid_attribute?(:name) && valid_attribute?(:path)
    return unless pending_delete_twin

    %i[route route.path name path].each do |error|
      errors.delete(error)
    end

    errors.add(:base, "The project is still being deleted. Please try again later.")
  end

  def pending_delete_twin
    return false unless path

1673
    Project.pending_delete.find_by_full_path(full_path)
1674
  end
1675 1676 1677 1678 1679 1680 1681 1682 1683

  ##
  # This method is here because of support for legacy container repository
  # which has exactly the same path like project does, but which might not be
  # persisted in `container_repositories` table.
  #
  def has_root_container_repository_tags?
    return false unless Gitlab.config.registry.enabled

1684
    ContainerRepository.build_root_repository(self).has_tags?
1685
  end
1686 1687 1688 1689 1690 1691 1692 1693 1694 1695 1696 1697

  def handle_update_attribute_error(ex, value)
    if ex.message.start_with?('Failed to replace')
      if value.respond_to?(:each)
        invalid = value.detect(&:invalid?)

        raise ex, ([ex.message] + invalid.errors.full_messages).join(' ') if invalid
      end
    end

    raise ex
  end
G
gitlabhq 已提交
1698
end