project.rb 20.4 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 24
#  archived               :boolean          default(FALSE), not null
#  import_status          :string(255)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
25 26
#  repository_size        :float            default(0.0)
#  star_count             :integer          default(0), not null
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
27 28
#  import_type            :string(255)
#  import_source          :string(255)
29
#  avatar                 :string(255)
30 31
#

32 33 34
require 'carrierwave/orm/activerecord'
require 'file_size_validator'

gitlabhq's avatar
gitlabhq committed
35
class Project < ActiveRecord::Base
36
  include Gitlab::ConfigHelper
37
  include Gitlab::ShellAdapter
38
  include Gitlab::VisibilityLevel
39 40
  include Referable
  include Sortable
sue445's avatar
sue445 committed
41

42
  extend Gitlab::ConfigHelper
43
  extend Enumerize
44

45
  default_value_for :archived, false
46 47 48 49
  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
  default_value_for :wiki_enabled, gitlab_config_features.wiki
50
  default_value_for :wall_enabled, false
51
  default_value_for :snippets_enabled, gitlab_config_features.snippets
52

53 54
  # set last_activity_at to the same as created_at
  after_create :set_last_activity_at
55
  def set_last_activity_at
56
    update_column(:last_activity_at, self.created_at)
57 58
  end

59
  ActsAsTaggableOn.strict_case_match = true
60
  acts_as_taggable_on :tags
61

62 63
  attr_accessor :new_default_branch

64
  # Relations
65
  belongs_to :creator, foreign_key: 'creator_id', class_name: 'User'
66
  belongs_to :group, -> { where(type: Group) }, foreign_key: 'namespace_id'
67
  belongs_to :namespace
68

69
  has_one :last_event, -> {order 'events.created_at DESC'}, class_name: 'Event', foreign_key: 'project_id'
70 71 72

  # Project services
  has_many :services
73
  has_one :gitlab_ci_service, dependent: :destroy
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
74
  has_one :campfire_service, dependent: :destroy
75
  has_one :emails_on_push_service, dependent: :destroy
Aorimn's avatar
Aorimn committed
76
  has_one :irker_service, dependent: :destroy
77
  has_one :pivotaltracker_service, dependent: :destroy
78
  has_one :hipchat_service, dependent: :destroy
79
  has_one :flowdock_service, dependent: :destroy
Carlos Paramio's avatar
Carlos Paramio committed
80
  has_one :assembla_service, dependent: :destroy
Jeremy's avatar
Jeremy committed
81
  has_one :asana_service, dependent: :destroy
82
  has_one :gemnasium_service, dependent: :destroy
83
  has_one :slack_service, dependent: :destroy
84
  has_one :buildkite_service, dependent: :destroy
Drew Blessing's avatar
Drew Blessing committed
85
  has_one :bamboo_service, dependent: :destroy
86
  has_one :teamcity_service, dependent: :destroy
87
  has_one :pushover_service, dependent: :destroy
88 89
  has_one :jira_service, dependent: :destroy
  has_one :redmine_service, dependent: :destroy
90
  has_one :custom_issue_tracker_service, dependent: :destroy
91
  has_one :gitlab_issue_tracker_service, dependent: :destroy
92
  has_one :external_wiki_service, dependent: :destroy
93

94
  has_one :forked_project_link, dependent: :destroy, foreign_key: "forked_to_project_id"
95

96
  has_one :forked_from_project, through: :forked_project_link
97
  # Merge Requests for target project should be removed with it
98
  has_many :merge_requests,     dependent: :destroy, foreign_key: 'target_project_id'
99
  # Merge requests from source project should be kept when source project was removed
100
  has_many :fork_merge_requests, foreign_key: 'source_project_id', class_name: MergeRequest
101
  has_many :issues,             dependent: :destroy
102
  has_many :labels,             dependent: :destroy
103 104
  has_many :services,           dependent: :destroy
  has_many :events,             dependent: :destroy
105 106
  has_many :milestones,         dependent: :destroy
  has_many :notes,              dependent: :destroy
107 108
  has_many :snippets,           dependent: :destroy, class_name: 'ProjectSnippet'
  has_many :hooks,              dependent: :destroy, class_name: 'ProjectHook'
109
  has_many :protected_branches, dependent: :destroy
110 111
  has_many :project_members, dependent: :destroy, as: :source, class_name: 'ProjectMember'
  has_many :users, through: :project_members
112 113
  has_many :deploy_keys_projects, dependent: :destroy
  has_many :deploy_keys, through: :deploy_keys_projects
Ciro Santilli's avatar
Ciro Santilli committed
114 115
  has_many :users_star_projects, dependent: :destroy
  has_many :starrers, through: :users_star_projects, source: :user
116

117 118
  has_one :import_data, dependent: :destroy, class_name: "ProjectImportData"

119
  delegate :name, to: :owner, allow_nil: true, prefix: true
120
  delegate :members, to: :team, prefix: true
121

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
122
  # Validations
123
  validates :creator, presence: true, on: :create
124
  validates :description, length: { maximum: 2000 }, allow_blank: true
125 126 127 128
  validates :name,
    presence: true,
    length: { within: 0..255 },
    format: { with: Gitlab::Regex.project_name_regex,
Douwe Maan's avatar
Douwe Maan committed
129
              message: Gitlab::Regex.project_name_regex_message }
130 131 132
  validates :path,
    presence: true,
    length: { within: 0..255 },
Douwe Maan's avatar
Douwe Maan committed
133 134
    format: { with: Gitlab::Regex.project_path_regex,
              message: Gitlab::Regex.project_path_regex_message }
135
  validates :issues_enabled, :merge_requests_enabled,
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
136
            :wiki_enabled, inclusion: { in: [true, false] }
137
  validates :issues_tracker_id, length: { maximum: 255 }, allow_blank: true
138
  validates :namespace, presence: true
139 140
  validates_uniqueness_of :name, scope: :namespace_id
  validates_uniqueness_of :path, scope: :namespace_id
141
  validates :import_url,
142
    format: { with: /\A#{URI.regexp(%w(ssh git http https))}\z/, message: 'should be a valid url' },
143
    if: :import?
144
  validates :star_count, numericality: { greater_than_or_equal_to: 0 }
145
  validate :check_limit, on: :create
146
  validate :avatar_type,
147
    if: ->(project) { project.avatar.present? && project.avatar_changed? }
148
  validates :avatar, file_size: { maximum: 200.kilobytes.to_i }
149

Douwe Maan's avatar
Douwe Maan committed
150
  mount_uploader :avatar, AvatarUploader
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
151

152
  # Scopes
153
  scope :sorted_by_activity, -> { reorder(last_activity_at: :desc) }
154 155 156
  scope :sorted_by_stars, -> { reorder('projects.star_count DESC') }
  scope :sorted_by_names, -> { joins(:namespace).reorder('namespaces.name ASC, projects.name ASC') }

157 158 159
  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 ) }
160
  scope :in_namespace, ->(namespace_ids) { where(namespace_id: namespace_ids) }
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
161
  scope :in_group_namespace, -> { joins(:group) }
162
  scope :personal, ->(user) { where(namespace_id: user.namespace_id) }
163
  scope :joined, ->(user) { where('namespace_id != ?', user.namespace_id) }
164 165
  scope :public_only, -> { where(visibility_level: Project::PUBLIC) }
  scope :public_and_internal_only, -> { where(visibility_level: Project.public_and_internal_levels) }
166 167
  scope :non_archived, -> { where(archived: false) }

168 169
  state_machine :import_status, initial: :none do
    event :import_start do
170
      transition [:none, :finished] => :started
171 172 173
    end

    event :import_finish do
174
      transition started: :finished
175 176 177
    end

    event :import_fail do
178
      transition started: :failed
179 180 181
    end

    event :import_retry do
182
      transition failed: :started
183 184 185 186
    end

    state :started
    state :finished
187 188
    state :failed

189
    after_transition any => :started, do: :add_import_job
190
    after_transition any => :finished, do: :clear_import_data
191 192
  end

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
193
  class << self
194 195 196 197
    def public_and_internal_levels
      [Project::PUBLIC, Project::INTERNAL]
    end

198
    def abandoned
199
      where('projects.last_activity_at < ?', 6.months.ago)
200
    end
201

202 203
    def publicish(user)
      visibility_levels = [Project::PUBLIC]
204
      visibility_levels << Project::INTERNAL if user
205 206
      where(visibility_level: visibility_levels)
    end
207

208
    def with_push
209
      joins(:events).where('events.action = ?', Event::PUSHED)
210 211
    end

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
212
    def active
213
      joins(:issues, :notes, :merge_requests).order('issues.created_at, notes.created_at, merge_requests.created_at DESC')
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
214
    end
215

216
    def search(query)
217 218
      joins(:namespace).where('projects.archived = ?', false).
        where('LOWER(projects.name) LIKE :query OR
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
219 220
              LOWER(projects.path) LIKE :query OR
              LOWER(namespaces.name) LIKE :query OR
221
              LOWER(projects.description) LIKE :query',
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
222
              query: "%#{query.try(:downcase)}%")
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
223
    end
224

225
    def search_by_title(query)
226
      where('projects.archived = ?', false).where('LOWER(projects.name) LIKE :query', query: "%#{query.downcase}%")
227 228
    end

229
    def find_with_namespace(id)
230
      return nil unless id.include?('/')
231

232
      id = id.split('/')
233 234 235 236
      namespace = Namespace.find_by(path: id.first)
      return nil unless namespace

      where(namespace_id: namespace.id).find_by(path: id.second)
237
    end
238

239 240 241
    def visibility_levels
      Gitlab::VisibilityLevel.options
    end
242 243

    def sort(method)
244 245 246 247
      if method == 'repository_size_desc'
        reorder(repository_size: :desc, id: :desc)
      else
        order_by(method)
248 249
      end
    end
250 251 252 253 254

    def reference_pattern
      name_pattern = Gitlab::Regex::NAMESPACE_REGEX_STR
      %r{(?<project>#{name_pattern}/#{name_pattern})}
    end
255 256
  end

Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
257
  def team
258
    @team ||= ProjectTeam.new(self)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
259 260 261
  end

  def repository
262 263 264
    @repository ||= Repository.new(path_with_namespace, nil, self)
  end

265
  def commit(id = 'HEAD')
266
    repository.commit(id)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
267 268
  end

269
  def saved?
270
    id && persisted?
271 272
  end

273 274 275 276
  def add_import_job
    RepositoryImportWorker.perform_in(2.seconds, id)
  end

277
  def clear_import_data
278
    self.import_data.destroy if self.import_data
279 280
  end

281 282 283 284
  def import?
    import_url.present?
  end

285
  def imported?
286 287 288 289 290 291 292 293 294 295 296 297 298
    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'
299 300
  end

301
  def check_limit
302
    unless creator.can_create_project? or namespace.kind == 'group'
303
      errors[:limit_reached] << ("Your project limit is #{creator.projects_limit} projects! Please contact your administrator to increase it")
304 305
    end
  rescue
Robert Speicher's avatar
Robert Speicher committed
306
    errors[:base] << ("Can't check your ability to create project")
gitlabhq's avatar
gitlabhq committed
307 308
  end

309
  def to_param
Vinnie Okada's avatar
Vinnie Okada committed
310
    path
311 312
  end

313 314 315 316
  def to_reference(_from_project = nil)
    path_with_namespace
  end

317
  def web_url
Douwe Maan's avatar
Douwe Maan committed
318
    Rails.application.routes.url_helpers.namespace_project_url(self.namespace, self)
319 320
  end

321
  def web_url_without_protocol
322
    web_url.split('://')[1]
323 324
  end

325
  def build_commit_note(commit)
326
    notes.new(commit_id: commit.id, noteable_type: 'Commit')
gitlabhq's avatar
gitlabhq committed
327
  end
Nihad Abbasov's avatar
Nihad Abbasov committed
328

329
  def last_activity
330
    last_event
gitlabhq's avatar
gitlabhq committed
331 332 333
  end

  def last_activity_date
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
334
    last_activity_at || updated_at
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
335
  end
336

337 338 339
  def project_id
    self.id
  end
randx's avatar
randx committed
340

Robert Speicher's avatar
Robert Speicher committed
341
  def get_issue(issue_id)
342
    if default_issues_tracker?
Robert Speicher's avatar
Robert Speicher committed
343
      issues.find_by(iid: issue_id)
344
    else
Robert Speicher's avatar
Robert Speicher committed
345
      ExternalIssue.new(issue_id, self)
346 347 348
    end
  end

Robert Speicher's avatar
Robert Speicher committed
349
  def issue_exists?(issue_id)
350
    get_issue(issue_id)
Robert Speicher's avatar
Robert Speicher committed
351 352
  end

353
  def default_issue_tracker
354
    gitlab_issue_tracker_service || create_gitlab_issue_tracker_service
355 356 357 358 359 360 361 362 363 364
  end

  def issues_tracker
    if external_issue_tracker
      external_issue_tracker
    else
      default_issue_tracker
    end
  end

365
  def default_issues_tracker?
366
    !external_issue_tracker
367 368 369
  end

  def external_issues_trackers
370
    services.select(&:issue_tracker?).reject(&:default?)
371 372 373 374 375 376
  end

  def external_issue_tracker
    @external_issues_tracker ||= external_issues_trackers.select(&:activated?).first
  end

Andrew8xx8's avatar
Andrew8xx8 committed
377
  def can_have_issues_tracker_id?
378
    self.issues_enabled && !self.default_issues_tracker?
Andrew8xx8's avatar
Andrew8xx8 committed
379 380
  end

381
  def build_missing_services
382 383
    services_templates = Service.where(template: true)

384
    Service.available_services_names.each do |service_name|
385
      service = find_service(services, service_name)
386 387

      # If service is available but missing in db
388 389 390 391 392 393 394 395 396 397 398
      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`
          service = self.send :"create_#{service_name}_service"
        else
          Service.create_from_template(self.id, template)
        end
      end
399 400 401
    end
  end

402 403 404
  def find_service(list, name)
    list.find { |service| service.to_param == name }
  end
405 406 407 408

  def gitlab_ci?
    gitlab_ci_service && gitlab_ci_service.active
  end
409

410 411 412 413 414
  def ci_services
    services.select { |service| service.category == :ci }
  end

  def ci_service
415
    @ci_service ||= ci_services.select(&:activated?).first
416 417
  end

418
  def avatar_type
419 420
    unless self.avatar.image?
      self.errors.add :avatar, 'only images allowed'
421 422 423 424 425 426 427 428 429 430
    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
431 432 433 434
  def avatar_url
    if avatar.present?
      [gitlab_config.url, avatar.url].join
    elsif avatar_in_git
Douwe Maan's avatar
Douwe Maan committed
435
      Rails.application.routes.url_helpers.namespace_project_avatar_url(namespace, self)
sue445's avatar
sue445 committed
436 437 438
    end
  end

439 440 441 442 443
  # For compatibility with old code
  def code
    path
  end

444
  def items_for(entity)
445 446 447 448 449 450 451
    case entity
    when 'issue' then
      issues
    when 'merge_request' then
      merge_requests
    end
  end
452 453

  def send_move_instructions
454
    NotificationService.new.project_was_moved(self)
455
  end
456 457

  def owner
458 459
    if group
      group
460
    else
461
      namespace.try(:owner)
462 463
    end
  end
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
464

465
  def project_member_by_name_or_email(name = nil, email = nil)
466
    user = users.where('name like ? or email like ?', name, email).first
467
    project_members.where(user: user) if user
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
468 469 470
  end

  # Get Team Member record by user id
471
  def project_member_by_id(user_id)
472
    project_members.find_by(user_id: user_id)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
473 474 475 476 477
  end

  def name_with_namespace
    @name_with_namespace ||= begin
                               if namespace
478
                                 namespace.human_name + ' / ' + name
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
479 480 481 482 483 484 485 486 487 488 489 490 491 492
                               else
                                 name
                               end
                             end
  end

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

493 494
  def execute_hooks(data, hooks_scope = :push_hooks)
    hooks.send(hooks_scope).each do |hook|
495
      hook.async_execute(data, hooks_scope.to_s)
496
    end
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
497 498
  end

499 500 501
  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|
502
      service.async_execute(data)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
503 504 505 506
    end
  end

  def update_merge_requests(oldrev, newrev, ref, user)
507 508
    MergeRequests::RefreshService.new(self, user).
      execute(oldrev, newrev, ref)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
509 510 511
  end

  def valid_repo?
512
    repository.exists?
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
513
  rescue
514
    errors.add(:path, 'Invalid repository path')
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
515 516 517 518
    false
  end

  def empty_repo?
519
    !repository.exists? || repository.empty?
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
520 521
  end

522 523 524 525
  def ensure_satellite_exists
    self.satellite.create unless self.satellite.exists?
  end

Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
526 527 528 529 530
  def satellite
    @satellite ||= Gitlab::Satellite::Satellite.new(self)
  end

  def repo
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
531
    repository.raw
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
532 533 534
  end

  def url_to_repo
535
    gitlab_shell.url_to_repo(path_with_namespace)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
536 537 538 539 540 541 542
  end

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

  def repo_exists?
543
    @repo_exists ||= repository.exists?
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
544 545 546 547 548
  rescue
    @repo_exists = false
  end

  def open_branches
549 550 551 552 553 554 555 556 557 558 559 560 561
    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
562 563 564
  end

  def root_ref?(branch)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
565
    repository.root_ref == branch
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
566 567 568 569 570 571 572
  end

  def ssh_url_to_repo
    url_to_repo
  end

  def http_url_to_repo
Douwe Maan's avatar
Douwe Maan committed
573
    "#{web_url}.git"
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
574 575 576
  end

  # Check if current branch name is marked as protected in the system
577
  def protected_branch?(branch_name)
578
    protected_branches_names.include?(branch_name)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
579
  end
580

581
  def developers_can_push_to_protected_branch?(branch_name)
582
    protected_branches.any? { |pb| pb.name == branch_name && pb.developers_can_push }
583 584
  end

585 586 587
  def forked?
    !(forked_project_link.nil? || forked_project_link.forked_from_project.nil?)
  end
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
588

589 590 591 592
  def personal?
    !group
  end

Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
593
  def rename_repo
594
    path_was = previous_changes['path'].first
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
595 596 597 598 599 600 601 602 603
    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)
      # If repository moved successfully we need to remove old satellite
      # and send update instructions to users.
      # However we cannot allow rollback since we moved repository
      # So we basically we mute exceptions in next actions
      begin
604
        gitlab_shell.mv_repository("#{old_path_with_namespace}.wiki", "#{new_path_with_namespace}.wiki")
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
605
        gitlab_shell.rm_satellites(old_path_with_namespace)
606
        ensure_satellite_exists
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
607
        send_move_instructions
608
        reset_events_cache
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
609
      rescue
Johannes Schleifenbaum's avatar
Johannes Schleifenbaum committed
610
        # Returning false does not rollback after_* transaction but gives
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
611 612 613 614 615 616 617 618 619
        # 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
  end
620

Kirill Zaitsev's avatar
Kirill Zaitsev committed
621 622 623 624 625 626 627 628 629 630
  def hook_attrs
    {
      name: name,
      ssh_url: ssh_url_to_repo,
      http_url: http_url_to_repo,
      namespace: namespace.name,
      visibility_level: visibility_level
    }
  end

631 632 633 634 635
  # 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
636
  # * when the project avatar changes
637 638 639 640 641 642 643 644 645
  # 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
646 647

  def project_member(user)
648
    project_members.where(user_id: user).first
649
  end
650 651 652 653

  def default_branch
    @default_branch ||= repository.root_ref if repository.exists?
  end
654 655 656 657 658

  def reload_default_branch
    @default_branch = nil
    default_branch
  end
659

660 661 662
  def visibility_level_field
    visibility_level
  end
663 664 665 666 667 668 669 670

  def archive!
    update_attribute(:archived, true)
  end

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

672 673 674 675
  def change_head(branch)
    gitlab_shell.update_repository_head(self.path_with_namespace, branch)
    reload_default_branch
  end
676 677 678 679

  def forked_from?(project)
    forked? && project == forked_from_project
  end
680 681 682 683

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

685 686 687 688
  def update_commit_count
    update_attribute(:commit_count, repository.commit_count)
  end

689 690 691
  def forks_count
    ForkedProjectLink.where(forked_from_project_id: self.id).count
  end
692 693 694 695

  def find_label(name)
    labels.find_by(name: name)
  end
696 697 698 699

  def origin_merge_requests
    merge_requests.where(source_project_id: self.id)
  end
700 701

  def create_repository
702 703 704 705 706 707 708 709
    if forked?
      if gitlab_shell.fork_repository(forked_from_project.path_with_namespace, self.namespace.path)
        ensure_satellite_exists
        true
      else
        errors.add(:base, 'Failed to fork repository')
        false
      end
710
    else
711 712 713 714 715 716
      if gitlab_shell.add_repository(path_with_namespace)
        true
      else
        errors.add(:base, 'Failed to create repository')
        false
      end
717 718 719 720 721 722 723 724 725 726 727
    end
  end

  def repository_exists?
    !!repository.exists?
  end

  def create_wiki
    ProjectWiki.new(self, self.owner).wiki
    true
  rescue ProjectWiki::CouldNotCreateWikiError => ex
728
    errors.add(:base, 'Failed create wiki')
729 730
    false
  end
gitlabhq's avatar
gitlabhq committed
731
end