project.rb 25.9 KB
Newer Older
1 2 3 4 5 6 7 8
# == Schema Information
#
# Table name: projects
#
#  id                     :integer          not null, primary key
#  name                   :string(255)
#  path                   :string(255)
#  description            :text
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
9 10
#  created_at             :datetime
#  updated_at             :datetime
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
11
#  creator_id             :integer
12 13 14 15
#  issues_enabled         :boolean          default(TRUE), not null
#  wall_enabled           :boolean          default(TRUE), not null
#  merge_requests_enabled :boolean          default(TRUE), not null
#  wiki_enabled           :boolean          default(TRUE), not null
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
16
#  namespace_id           :integer
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
17
#  issues_tracker         :string(255)      default("gitlab"), not null
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
18
#  issues_tracker_id      :string(255)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
19
#  snippets_enabled       :boolean          default(TRUE), not null
20
#  last_activity_at       :datetime
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
21
#  import_url             :string(255)
22
#  visibility_level       :integer          default(0), not null
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
23
#  archived               :boolean          default(FALSE), not null
Atsushi Ishida's avatar
Atsushi Ishida committed
24
#  avatar                 :string(255)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
25
#  import_status          :string(255)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
26 27
#  repository_size        :float            default(0.0)
#  star_count             :integer          default(0), not null
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
28 29
#  import_type            :string(255)
#  import_source          :string(255)
Atsushi Ishida's avatar
Atsushi Ishida committed
30
#  commit_count           :integer          default(0)
Stan Hu's avatar
Stan Hu committed
31
#  import_error           :text
Stan Hu's avatar
Stan Hu committed
32 33 34 35 36 37 38
#  ci_id                  :integer
#  builds_enabled         :boolean          default(TRUE), not null
#  shared_runners_enabled :boolean          default(TRUE), not null
#  runners_token          :string
#  build_coverage_regex   :string
#  build_allow_git_fetch  :boolean          default(TRUE), not null
#  build_timeout          :integer          default(3600), not null
39 40
#

41 42 43
require 'carrierwave/orm/activerecord'
require 'file_size_validator'

gitlabhq's avatar
gitlabhq committed
44
class Project < ActiveRecord::Base
45
  include Gitlab::ConfigHelper
46
  include Gitlab::ShellAdapter
47
  include Gitlab::VisibilityLevel
48
  include Gitlab::CurrentSettings
49 50
  include Referable
  include Sortable
51
  include AfterCommitQueue
52
  include CaseSensitivity
53
  include TokenAuthenticatable
Robert Speicher's avatar
Robert Speicher committed
54

55
  extend Gitlab::ConfigHelper
56

Jared Szechy's avatar
Jared Szechy committed
57 58
  UNKNOWN_IMPORT_URL = 'http://unknown.git'

59
  default_value_for :archived, false
60 61 62
  default_value_for :visibility_level, gitlab_config_features.visibility_level
  default_value_for :issues_enabled, gitlab_config_features.issues
  default_value_for :merge_requests_enabled, gitlab_config_features.merge_requests
63
  default_value_for :builds_enabled, gitlab_config_features.builds
64
  default_value_for :wiki_enabled, gitlab_config_features.wiki
65
  default_value_for :wall_enabled, false
66
  default_value_for :snippets_enabled, gitlab_config_features.snippets
67
  default_value_for(:shared_runners_enabled) { current_application_settings.shared_runners_enabled }
68

69 70
  # set last_activity_at to the same as created_at
  after_create :set_last_activity_at
71
  def set_last_activity_at
72
    update_column(:last_activity_at, self.created_at)
73 74
  end

75 76 77 78 79 80 81 82 83 84 85 86 87
  # update visibility_levet of forks
  after_update :update_forks_visibility_level
  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

88
  ActsAsTaggableOn.strict_case_match = true
89
  acts_as_taggable_on :tags
90

91
  attr_accessor :new_default_branch
92
  attr_accessor :old_path_with_namespace
93

94
  # Relations
95
  belongs_to :creator, foreign_key: 'creator_id', class_name: 'User'
96
  belongs_to :group, -> { where(type: Group) }, foreign_key: 'namespace_id'
97
  belongs_to :namespace
98

99
  has_one :last_event, -> {order 'events.created_at DESC'}, class_name: 'Event', foreign_key: 'project_id'
100 101 102

  # Project services
  has_many :services
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
103
  has_one :campfire_service, dependent: :destroy
Kirilll Zaitsev's avatar
Kirilll Zaitsev committed
104
  has_one :drone_ci_service, dependent: :destroy
105
  has_one :emails_on_push_service, dependent: :destroy
106
  has_one :builds_email_service, dependent: :destroy
Aorimn's avatar
Aorimn committed
107
  has_one :irker_service, dependent: :destroy
108
  has_one :pivotaltracker_service, dependent: :destroy
109
  has_one :hipchat_service, dependent: :destroy
110
  has_one :flowdock_service, dependent: :destroy
Carlos Paramio's avatar
Carlos Paramio committed
111
  has_one :assembla_service, dependent: :destroy
Jeremy's avatar
Jeremy committed
112
  has_one :asana_service, dependent: :destroy
113
  has_one :gemnasium_service, dependent: :destroy
114
  has_one :slack_service, dependent: :destroy
115
  has_one :buildkite_service, dependent: :destroy
Drew Blessing's avatar
Drew Blessing committed
116
  has_one :bamboo_service, dependent: :destroy
117
  has_one :teamcity_service, dependent: :destroy
118
  has_one :pushover_service, dependent: :destroy
119 120
  has_one :jira_service, dependent: :destroy
  has_one :redmine_service, dependent: :destroy
121
  has_one :custom_issue_tracker_service, dependent: :destroy
122
  has_one :gitlab_issue_tracker_service, dependent: :destroy
123
  has_one :external_wiki_service, dependent: :destroy
124

125 126 127 128 129
  has_one  :forked_project_link,  dependent: :destroy, foreign_key: "forked_to_project_id"
  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
130

131
  # Merge Requests for target project should be removed with it
132
  has_many :merge_requests,     dependent: :destroy, foreign_key: 'target_project_id'
133
  # Merge requests from source project should be kept when source project was removed
134
  has_many :fork_merge_requests, foreign_key: 'source_project_id', class_name: MergeRequest
135
  has_many :issues,             dependent: :destroy
136
  has_many :labels,             dependent: :destroy
137 138
  has_many :services,           dependent: :destroy
  has_many :events,             dependent: :destroy
139 140
  has_many :milestones,         dependent: :destroy
  has_many :notes,              dependent: :destroy
141 142
  has_many :snippets,           dependent: :destroy, class_name: 'ProjectSnippet'
  has_many :hooks,              dependent: :destroy, class_name: 'ProjectHook'
143
  has_many :protected_branches, dependent: :destroy
144 145
  has_many :project_members, dependent: :destroy, as: :source, class_name: 'ProjectMember'
  has_many :users, through: :project_members
146 147
  has_many :deploy_keys_projects, dependent: :destroy
  has_many :deploy_keys, through: :deploy_keys_projects
Ciro Santilli's avatar
Ciro Santilli committed
148 149
  has_many :users_star_projects, dependent: :destroy
  has_many :starrers, through: :users_star_projects, source: :user
150
  has_many :releases, dependent: :destroy
Marin Jankovski's avatar
Marin Jankovski committed
151 152
  has_many :lfs_objects_projects, dependent: :destroy
  has_many :lfs_objects, through: :lfs_objects_projects
153

154
  has_one :import_data, dependent: :destroy, class_name: "ProjectImportData"
155

156
  has_many :commit_statuses, dependent: :destroy, class_name: 'CommitStatus', foreign_key: :gl_project_id
157
  has_many :ci_commits, dependent: :destroy, class_name: 'Ci::Commit', foreign_key: :gl_project_id
158 159 160 161 162
  has_many :builds, class_name: 'Ci::Build', foreign_key: :gl_project_id # the builds are created from the commit_statuses
  has_many :runner_projects, dependent: :destroy, class_name: 'Ci::RunnerProject', foreign_key: :gl_project_id
  has_many :runners, through: :runner_projects, source: :runner, class_name: 'Ci::Runner'
  has_many :variables, dependent: :destroy, class_name: 'Ci::Variable', foreign_key: :gl_project_id
  has_many :triggers, dependent: :destroy, class_name: 'Ci::Trigger', foreign_key: :gl_project_id
163

164
  accepts_nested_attributes_for :variables, allow_destroy: true
165

166
  delegate :name, to: :owner, allow_nil: true, prefix: true
167
  delegate :members, to: :team, prefix: true
168

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
169
  # Validations
170
  validates :creator, presence: true, on: :create
171
  validates :description, length: { maximum: 2000 }, allow_blank: true
172 173 174 175
  validates :name,
    presence: true,
    length: { within: 0..255 },
    format: { with: Gitlab::Regex.project_name_regex,
Douwe Maan's avatar
Douwe Maan committed
176
              message: Gitlab::Regex.project_name_regex_message }
177 178 179
  validates :path,
    presence: true,
    length: { within: 0..255 },
Douwe Maan's avatar
Douwe Maan committed
180 181
    format: { with: Gitlab::Regex.project_path_regex,
              message: Gitlab::Regex.project_path_regex_message }
182
  validates :issues_enabled, :merge_requests_enabled,
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
183
            :wiki_enabled, inclusion: { in: [true, false] }
184
  validates :issues_tracker_id, length: { maximum: 255 }, allow_blank: true
185
  validates :namespace, presence: true
186 187
  validates_uniqueness_of :name, scope: :namespace_id
  validates_uniqueness_of :path, scope: :namespace_id
188
  validates :import_url,
Robert Speicher's avatar
Robert Speicher committed
189
    url: { protocols: %w(ssh git http https) },
190
    if: :external_import?
191
  validates :star_count, numericality: { greater_than_or_equal_to: 0 }
192
  validate :check_limit, on: :create
193
  validate :avatar_type,
194
    if: ->(project) { project.avatar.present? && project.avatar_changed? }
195
  validates :avatar, file_size: { maximum: 200.kilobytes.to_i }
196

197
  add_authentication_token_field :runners_token
198
  before_save :ensure_runners_token
199

Douwe Maan's avatar
Douwe Maan committed
200
  mount_uploader :avatar, AvatarUploader
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
201

202
  # Scopes
203
  scope :sorted_by_activity, -> { reorder(last_activity_at: :desc) }
204 205 206
  scope :sorted_by_stars, -> { reorder('projects.star_count DESC') }
  scope :sorted_by_names, -> { joins(:namespace).reorder('namespaces.name ASC, projects.name ASC') }

207 208 209
  scope :without_user, ->(user)  { where('projects.id NOT IN (:ids)', ids: user.authorized_projects.map(&:id) ) }
  scope :without_team, ->(team) { team.projects.present? ? where('projects.id NOT IN (:ids)', ids: team.projects.map(&:id)) : scoped  }
  scope :not_in_group, ->(group) { where('projects.id NOT IN (:ids)', ids: group.project_ids ) }
210
  scope :in_namespace, ->(namespace_ids) { where(namespace_id: namespace_ids) }
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
211
  scope :in_group_namespace, -> { joins(:group) }
212
  scope :personal, ->(user) { where(namespace_id: user.namespace_id) }
213
  scope :joined, ->(user) { where('namespace_id != ?', user.namespace_id) }
214 215
  scope :public_only, -> { where(visibility_level: Project::PUBLIC) }
  scope :public_and_internal_only, -> { where(visibility_level: Project.public_and_internal_levels) }
216 217
  scope :non_archived, -> { where(archived: false) }

218 219
  state_machine :import_status, initial: :none do
    event :import_start do
220
      transition [:none, :finished] => :started
221 222 223
    end

    event :import_finish do
224
      transition started: :finished
225 226 227
    end

    event :import_fail do
228
      transition started: :failed
229 230 231
    end

    event :import_retry do
232
      transition failed: :started
233 234 235 236
    end

    state :started
    state :finished
237 238
    state :failed

239
    after_transition any => :started, do: :schedule_add_import_job
240
    after_transition any => :finished, do: :clear_import_data
241 242
  end

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
243
  class << self
244 245 246 247
    def public_and_internal_levels
      [Project::PUBLIC, Project::INTERNAL]
    end

248
    def abandoned
249
      where('projects.last_activity_at < ?', 6.months.ago)
250
    end
251

252 253
    def publicish(user)
      visibility_levels = [Project::PUBLIC]
254
      visibility_levels << Project::INTERNAL if user
255 256
      where(visibility_level: visibility_levels)
    end
257

258
    def with_push
259
      joins(:events).where('events.action = ?', Event::PUSHED)
260 261
    end

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
262
    def active
263
      joins(:issues, :notes, :merge_requests).order('issues.created_at, notes.created_at, merge_requests.created_at DESC')
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
264
    end
265

266
    def search(query)
267
      joins(:namespace).
268
        where('LOWER(projects.name) LIKE :query OR
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
269 270
              LOWER(projects.path) LIKE :query OR
              LOWER(namespaces.name) LIKE :query OR
271
              LOWER(projects.description) LIKE :query',
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
272
              query: "%#{query.try(:downcase)}%")
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
273
    end
274

275
    def search_by_title(query)
276
      where('projects.archived = ?', false).where('LOWER(projects.name) LIKE :query', query: "%#{query.downcase}%")
277 278
    end

279
    def find_with_namespace(id)
280 281 282 283 284 285 286
      namespace_path, project_path = id.split('/')

      return nil if !namespace_path || !project_path

      # Use of unscoped ensures we're not secretly adding any ORDER BYs, which
      # have a negative impact on performance (and aren't needed for this
      # query).
287
      projects = unscoped.
288
        joins(:namespace).
289 290
        iwhere('namespaces.path' => namespace_path)

Gabriel Mazetto's avatar
Gabriel Mazetto committed
291
      projects.find_by('projects.path' => project_path) ||
292
        projects.iwhere('projects.path' => project_path).take
293
    end
294

295
    def find_by_ci_id(id)
Kamil Trzcinski's avatar
Kamil Trzcinski committed
296
      find_by(ci_id: id.to_i)
297 298
    end

299 300 301
    def visibility_levels
      Gitlab::VisibilityLevel.options
    end
302 303

    def sort(method)
304 305 306 307
      if method == 'repository_size_desc'
        reorder(repository_size: :desc, id: :desc)
      else
        order_by(method)
308 309
      end
    end
310 311 312 313 314

    def reference_pattern
      name_pattern = Gitlab::Regex::NAMESPACE_REGEX_STR
      %r{(?<project>#{name_pattern}/#{name_pattern})}
    end
315 316 317 318 319 320 321 322 323 324 325 326 327 328

    def trending(since = 1.month.ago)
      # By counting in the JOIN we don't expose the GROUP BY to the outer query.
      # This means that calls such as "any?" and "count" just return a number of
      # the total count, instead of the counts grouped per project as a Hash.
      join_body = "INNER JOIN (
        SELECT project_id, COUNT(*) AS amount
        FROM notes
        WHERE created_at >= #{sanitize(since)}
        GROUP BY project_id
      ) join_note_counts ON projects.id = join_note_counts.project_id"

      joins(join_body).reorder('join_note_counts.amount DESC')
    end
329 330 331 332

    def visible_to_user(user)
      where(id: user.authorized_projects.select(:id).reorder(nil))
    end
333 334
  end

Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
335
  def team
336
    @team ||= ProjectTeam.new(self)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
337 338 339
  end

  def repository
340 341 342
    @repository ||= Repository.new(path_with_namespace, nil, self)
  end

343
  def commit(id = 'HEAD')
344
    repository.commit(id)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
345 346
  end

347
  def saved?
348
    id && persisted?
349 350
  end

351 352 353 354
  def schedule_add_import_job
    run_after_commit(:add_import_job)
  end

355
  def add_import_job
356
    if forked?
357
      RepositoryForkWorker.perform_async(self.id, forked_from_project.path_with_namespace, self.namespace.path)
358
    else
359
      RepositoryImportWorker.perform_async(self.id)
360
    end
361 362
  end

363
  def clear_import_data
364 365 366 367
    update(import_error: nil)

    ProjectCacheWorker.perform_async(self.id)

368
    self.import_data.destroy if self.import_data
369 370
  end

371
  def import?
372 373 374 375
    external_import? || forked?
  end

  def external_import?
376 377 378
    import_url.present?
  end

379
  def imported?
380 381 382 383 384 385 386 387 388 389 390 391 392
    import_finished?
  end

  def import_in_progress?
    import? && import_status == 'started'
  end

  def import_failed?
    import_status == 'failed'
  end

  def import_finished?
    import_status == 'finished'
393 394
  end

395 396 397 398 399 400 401 402
  def safe_import_url
    result = URI.parse(self.import_url)
    result.password = '*****' unless result.password.nil?
    result.to_s
  rescue
    original_url
  end

403
  def check_limit
404
    unless creator.can_create_project? or namespace.kind == 'group'
405
      errors[:limit_reached] << ("Your project limit is #{creator.projects_limit} projects! Please contact your administrator to increase it")
406 407
    end
  rescue
Robert Speicher's avatar
Robert Speicher committed
408
    errors[:base] << ("Can't check your ability to create project")
gitlabhq's avatar
gitlabhq committed
409 410
  end

411
  def to_param
Vinnie Okada's avatar
Vinnie Okada committed
412
    path
413 414
  end

415 416 417 418
  def to_reference(_from_project = nil)
    path_with_namespace
  end

419
  def web_url
420
    Gitlab::Application.routes.url_helpers.namespace_project_url(self.namespace, self)
421 422
  end

423
  def web_url_without_protocol
424
    web_url.split('://')[1]
425 426
  end

427
  def build_commit_note(commit)
428
    notes.new(commit_id: commit.id, noteable_type: 'Commit')
gitlabhq's avatar
gitlabhq committed
429
  end
Nihad Abbasov's avatar
Nihad Abbasov committed
430

431
  def last_activity
432
    last_event
gitlabhq's avatar
gitlabhq committed
433 434 435
  end

  def last_activity_date
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
436
    last_activity_at || updated_at
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
437
  end
438

439 440 441
  def project_id
    self.id
  end
randx's avatar
randx committed
442

Robert Speicher's avatar
Robert Speicher committed
443
  def get_issue(issue_id)
444
    if default_issues_tracker?
Robert Speicher's avatar
Robert Speicher committed
445
      issues.find_by(iid: issue_id)
446
    else
Robert Speicher's avatar
Robert Speicher committed
447
      ExternalIssue.new(issue_id, self)
448 449 450
    end
  end

Robert Speicher's avatar
Robert Speicher committed
451
  def issue_exists?(issue_id)
452
    get_issue(issue_id)
Robert Speicher's avatar
Robert Speicher committed
453 454
  end

455
  def default_issue_tracker
456
    gitlab_issue_tracker_service || create_gitlab_issue_tracker_service
457 458 459 460 461 462 463 464 465 466
  end

  def issues_tracker
    if external_issue_tracker
      external_issue_tracker
    else
      default_issue_tracker
    end
  end

467
  def default_issues_tracker?
468
    !external_issue_tracker
469 470 471
  end

  def external_issues_trackers
472
    services.select(&:issue_tracker?).reject(&:default?)
473 474 475
  end

  def external_issue_tracker
Gabriel Mazetto's avatar
Gabriel Mazetto committed
476
    @external_issues_tracker ||= external_issues_trackers.find(&:activated?)
477 478
  end

Andrew8xx8's avatar
Andrew8xx8 committed
479
  def can_have_issues_tracker_id?
480
    self.issues_enabled && !self.default_issues_tracker?
Andrew8xx8's avatar
Andrew8xx8 committed
481 482
  end

483
  def build_missing_services
484 485
    services_templates = Service.where(template: true)

486
    Service.available_services_names.each do |service_name|
487
      service = find_service(services, service_name)
488 489

      # If service is available but missing in db
490 491 492 493 494 495
      if service.nil?
        # We should check if template for the service exists
        template = find_service(services_templates, service_name)

        if template.nil?
          # If no template, we should create an instance. Ex `create_gitlab_ci_service`
496
          self.send :"create_#{service_name}_service"
497 498 499 500
        else
          Service.create_from_template(self.id, template)
        end
      end
501 502 503
    end
  end

504 505 506 507 508 509 510 511 512
  def create_labels
    Label.templates.each do |label|
      label = label.dup
      label.template = nil
      label.project_id = self.id
      label.save
    end
  end

513 514 515
  def find_service(list, name)
    list.find { |service| service.to_param == name }
  end
516

517 518 519 520 521
  def ci_services
    services.select { |service| service.category == :ci }
  end

  def ci_service
Gabriel Mazetto's avatar
Gabriel Mazetto committed
522
    @ci_service ||= ci_services.find(&:activated?)
523 524
  end

Drew Blessing's avatar
Drew Blessing committed
525 526 527 528
  def jira_tracker?
    issues_tracker.to_param == 'jira'
  end

529
  def avatar_type
530 531
    unless self.avatar.image?
      self.errors.add :avatar, 'only images allowed'
532 533 534 535 536 537 538 539 540 541
    end
  end

  def avatar_in_git
    @avatar_file ||= 'logo.png' if repository.blob_at_branch('master', 'logo.png')
    @avatar_file ||= 'logo.jpg' if repository.blob_at_branch('master', 'logo.jpg')
    @avatar_file ||= 'logo.gif' if repository.blob_at_branch('master', 'logo.gif')
    @avatar_file
  end

sue445's avatar
sue445 committed
542 543 544 545
  def avatar_url
    if avatar.present?
      [gitlab_config.url, avatar.url].join
    elsif avatar_in_git
546
      Gitlab::Application.routes.url_helpers.namespace_project_avatar_url(namespace, self)
sue445's avatar
sue445 committed
547 548 549
    end
  end

550 551 552 553 554
  # For compatibility with old code
  def code
    path
  end

555
  def items_for(entity)
556 557 558 559 560 561 562
    case entity
    when 'issue' then
      issues
    when 'merge_request' then
      merge_requests
    end
  end
563

564
  def send_move_instructions(old_path_with_namespace)
565 566 567
    # 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) }
568
  end
569 570

  def owner
571 572
    if group
      group
573
    else
574
      namespace.try(:owner)
575 576
    end
  end
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
577

578
  def project_member_by_name_or_email(name = nil, email = nil)
Gabriel Mazetto's avatar
Gabriel Mazetto committed
579
    user = users.find_by('name like ? or email like ?', name, email)
580
    project_members.where(user: user) if user
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
581 582 583
  end

  # Get Team Member record by user id
584
  def project_member_by_id(user_id)
585
    project_members.find_by(user_id: user_id)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
586 587 588 589 590
  end

  def name_with_namespace
    @name_with_namespace ||= begin
                               if namespace
591
                                 namespace.human_name + ' / ' + name
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
592 593 594 595 596 597 598 599 600 601 602 603 604 605
                               else
                                 name
                               end
                             end
  end

  def path_with_namespace
    if namespace
      namespace.path + '/' + path
    else
      path
    end
  end

606 607
  def execute_hooks(data, hooks_scope = :push_hooks)
    hooks.send(hooks_scope).each do |hook|
608
      hook.async_execute(data, hooks_scope.to_s)
609
    end
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
610 611
  end

612 613 614
  def execute_services(data, hooks_scope = :push_hooks)
    # Call only service hooks that are active for this scope
    services.send(hooks_scope).each do |service|
615
      service.async_execute(data)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
616 617 618 619
    end
  end

  def update_merge_requests(oldrev, newrev, ref, user)
620 621
    MergeRequests::RefreshService.new(self, user).
      execute(oldrev, newrev, ref)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
622 623 624
  end

  def valid_repo?
625
    repository.exists?
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
626
  rescue
627
    errors.add(:path, 'Invalid repository path')
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
628 629 630 631
    false
  end

  def empty_repo?
632
    !repository.exists? || !repository.has_visible_content?
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
633 634 635
  end

  def repo
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
636
    repository.raw
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
637 638 639
  end

  def url_to_repo
640
    gitlab_shell.url_to_repo(path_with_namespace)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
641 642 643 644 645 646 647
  end

  def namespace_dir
    namespace.try(:path) || ''
  end

  def repo_exists?
648
    @repo_exists ||= repository.exists?
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
649 650 651 652 653
  rescue
    @repo_exists = false
  end

  def open_branches
654 655 656 657 658 659 660 661 662 663 664 665 666
    all_branches = repository.branches

    if protected_branches.present?
      all_branches.reject! do |branch|
        protected_branches_names.include?(branch.name)
      end
    end

    all_branches
  end

  def protected_branches_names
    @protected_branches_names ||= protected_branches.map(&:name)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
667 668 669
  end

  def root_ref?(branch)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
670
    repository.root_ref == branch
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
671 672 673 674 675 676 677
  end

  def ssh_url_to_repo
    url_to_repo
  end

  def http_url_to_repo
Douwe Maan's avatar
Douwe Maan committed
678
    "#{web_url}.git"
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
679 680 681
  end

  # Check if current branch name is marked as protected in the system
682
  def protected_branch?(branch_name)
683
    protected_branches_names.include?(branch_name)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
684
  end
685

686
  def developers_can_push_to_protected_branch?(branch_name)
687
    protected_branches.any? { |pb| pb.name == branch_name && pb.developers_can_push }
688 689
  end

690 691 692
  def forked?
    !(forked_project_link.nil? || forked_project_link.forked_from_project.nil?)
  end
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
693

694 695 696 697
  def personal?
    !group
  end

Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
698
  def rename_repo
699
    path_was = previous_changes['path'].first
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
700 701 702 703
    old_path_with_namespace = File.join(namespace_dir, path_was)
    new_path_with_namespace = File.join(namespace_dir, path)

    if gitlab_shell.mv_repository(old_path_with_namespace, new_path_with_namespace)
704
      # If repository moved successfully we need to send update instructions to users.
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
705 706 707
      # However we cannot allow rollback since we moved repository
      # So we basically we mute exceptions in next actions
      begin
708
        gitlab_shell.mv_repository("#{old_path_with_namespace}.wiki", "#{new_path_with_namespace}.wiki")
709
        send_move_instructions(old_path_with_namespace)
710
        reset_events_cache
711 712 713 714 715

        @old_path_with_namespace = old_path_with_namespace

        SystemHooksService.new.execute_hooks_for(self, :rename)

716
        @repository = nil
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
717
      rescue
Johannes Schleifenbaum's avatar
Johannes Schleifenbaum committed
718
        # Returning false does not rollback after_* transaction but gives
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
719 720 721 722 723 724 725 726
        # us information about failing some of tasks
        false
      end
    else
      # if we cannot move namespace directory we should rollback
      # db changes in order to prevent out of sync between db and fs
      raise Exception.new('repository cannot be renamed')
    end
727 728

    Gitlab::UploadsTransfer.new.rename_project(path_was, path, namespace.path)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
729
  end
730

Kirill Zaitsev's avatar
Kirill Zaitsev committed
731 732 733 734 735
  def hook_attrs
    {
      name: name,
      ssh_url: ssh_url_to_repo,
      http_url: http_url_to_repo,
Kirilll Zaitsev's avatar
Kirilll Zaitsev committed
736
      web_url: web_url,
Kirill Zaitsev's avatar
Kirill Zaitsev committed
737 738 739 740 741
      namespace: namespace.name,
      visibility_level: visibility_level
    }
  end

742 743 744 745 746
  # Reset events cache related to this project
  #
  # Since we do cache @event we need to reset cache in special cases:
  # * when project was moved
  # * when project was renamed
747
  # * when the project avatar changes
748 749 750 751 752 753 754 755 756
  # Events cache stored like  events/23-20130109142513.
  # The cache key includes updated_at timestamp.
  # Thus it will automatically generate a new fragment
  # when the event is updated because the key changes.
  def reset_events_cache
    Event.where(project_id: self.id).
      order('id DESC').limit(100).
      update_all(updated_at: Time.now)
  end
757 758

  def project_member(user)
Gabriel Mazetto's avatar
Gabriel Mazetto committed
759
    project_members.find_by(user_id: user)
760
  end
761 762 763 764

  def default_branch
    @default_branch ||= repository.root_ref if repository.exists?
  end
765 766 767 768 769

  def reload_default_branch
    @default_branch = nil
    default_branch
  end
770

771 772 773
  def visibility_level_field
    visibility_level
  end
774 775 776 777 778 779 780 781

  def archive!
    update_attribute(:archived, true)
  end

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

783
  def change_head(branch)
784 785
    # Cached divergent commit counts are based on repository head
    repository.expire_branch_cache
786 787 788
    gitlab_shell.update_repository_head(self.path_with_namespace, branch)
    reload_default_branch
  end
789 790 791 792

  def forked_from?(project)
    forked? && project == forked_from_project
  end
793 794 795 796

  def update_repository_size
    update_attribute(:repository_size, repository.size)
  end
797

798 799 800 801
  def update_commit_count
    update_attribute(:commit_count, repository.commit_count)
  end

802
  def forks_count
803
    forks.count
804
  end
805 806 807 808

  def find_label(name)
    labels.find_by(name: name)
  end
809 810 811 812

  def origin_merge_requests
    merge_requests.where(source_project_id: self.id)
  end
813 814

  def create_repository
815 816
    # Forked import is handled asynchronously
    unless forked?
817 818 819
      if gitlab_shell.add_repository(path_with_namespace)
        true
      else
820
        errors.add(:base, 'Failed to create repository via gitlab-shell')
821 822
        false
      end
823 824 825 826 827 828 829 830 831 832
    end
  end

  def repository_exists?
    !!repository.exists?
  end

  def create_wiki
    ProjectWiki.new(self, self.owner).wiki
    true
833
  rescue ProjectWiki::CouldNotCreateWikiError
834
    errors.add(:base, 'Failed create wiki')
835 836
    false
  end
837

Drew Blessing's avatar
Drew Blessing committed
838 839 840 841
  def jira_tracker_active?
    jira_tracker? && jira_service.active
  end

842
  def ci_commit(sha)
843 844 845 846 847
    ci_commits.find_by(sha: sha)
  end

  def ensure_ci_commit(sha)
    ci_commit(sha) || ci_commits.create(sha: sha)
848
  end
849

850 851 852
  def enable_ci
    self.builds_enabled = true
  end
Marin Jankovski's avatar
Marin Jankovski committed
853 854 855 856 857 858 859 860 861 862

  def unlink_fork
    if forked?
      forked_from_project.lfs_objects.find_each do |lfs_object|
        lfs_object.projects << self
      end

      forked_project_link.destroy
    end
  end
863 864

  def any_runners?(&block)
865
    if runners.active.any?(&block)
866 867 868 869 870 871
      return true
    end

    shared_runners_enabled? && Ci::Runner.shared.active.any?(&block)
  end

Kamil Trzcinski's avatar
Kamil Trzcinski committed
872
  def valid_runners_token? token
Kamil Trzcinski's avatar
Kamil Trzcinski committed
873
    self.runners_token && self.runners_token == token
Kamil Trzcinski's avatar
Kamil Trzcinski committed
874 875
  end

Kamil Trzcinski's avatar
Kamil Trzcinski committed
876 877
  # TODO (ayufan): For now we use runners_token (backward compatibility)
  # In 8.4 every build will have its own individual token valid for time of build
Kamil Trzcinski's avatar
Kamil Trzcinski committed
878
  def valid_build_token? token
Kamil Trzcinski's avatar
Kamil Trzcinski committed
879
    self.builds_enabled? && self.runners_token && self.runners_token == token
880 881 882 883 884 885 886 887 888 889 890 891 892
  end

  def build_coverage_enabled?
    build_coverage_regex.present?
  end

  def build_timeout_in_minutes
    build_timeout / 60
  end

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

894 895 896
  def open_issues_count
    issues.opened.count
  end
897 898 899 900

  def visibility_level_allowed?(level)
    return true unless forked?
    Gitlab::VisibilityLevel.allowed_fork_levels(forked_from_project.visibility_level).include?(level.to_i)
Marin Jankovski's avatar
Marin Jankovski committed
901
  end
902 903 904 905

  def runners_token
    ensure_runners_token!
  end
gitlabhq's avatar
gitlabhq committed
906
end