project.rb 18.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 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 Sortable
37
  include Gitlab::ShellAdapter
38
  include Gitlab::VisibilityLevel
39 40
  include Gitlab::ConfigHelper
  extend Gitlab::ConfigHelper
41
  extend Enumerize
42

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

51
  ActsAsTaggableOn.strict_case_match = true
52
  acts_as_taggable_on :tags
53

54 55
  attr_accessor :new_default_branch

56
  # Relations
57
  belongs_to :creator, foreign_key: 'creator_id', class_name: 'User'
58
  belongs_to :group, -> { where(type: Group) }, foreign_key: 'namespace_id'
59
  belongs_to :namespace
60

61
  has_one :last_event, -> {order 'events.created_at DESC'}, class_name: 'Event', foreign_key: 'project_id'
62 63 64

  # Project services
  has_many :services
65
  has_one :gitlab_ci_service, dependent: :destroy
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
66
  has_one :campfire_service, dependent: :destroy
67
  has_one :emails_on_push_service, dependent: :destroy
68
  has_one :pivotaltracker_service, dependent: :destroy
69
  has_one :hipchat_service, dependent: :destroy
70
  has_one :flowdock_service, dependent: :destroy
Carlos Paramio's avatar
Carlos Paramio committed
71
  has_one :assembla_service, dependent: :destroy
72
  has_one :gemnasium_service, dependent: :destroy
73
  has_one :slack_service, dependent: :destroy
Keith Pitt's avatar
Keith Pitt committed
74
  has_one :buildbox_service, dependent: :destroy
Drew Blessing's avatar
Drew Blessing committed
75
  has_one :bamboo_service, dependent: :destroy
76
  has_one :teamcity_service, dependent: :destroy
77
  has_one :pushover_service, dependent: :destroy
78 79
  has_one :jira_service, dependent: :destroy
  has_one :redmine_service, dependent: :destroy
80
  has_one :custom_issue_tracker_service, dependent: :destroy
81
  has_one :gitlab_issue_tracker_service, dependent: :destroy
82

83
  has_one :forked_project_link, dependent: :destroy, foreign_key: "forked_to_project_id"
84

85
  has_one :forked_from_project, through: :forked_project_link
86
  # Merge Requests for target project should be removed with it
87
  has_many :merge_requests,     dependent: :destroy, foreign_key: 'target_project_id'
88
  # Merge requests from source project should be kept when source project was removed
89
  has_many :fork_merge_requests, foreign_key: 'source_project_id', class_name: MergeRequest
90
  has_many :issues,             dependent: :destroy
91
  has_many :labels,             dependent: :destroy
92 93
  has_many :services,           dependent: :destroy
  has_many :events,             dependent: :destroy
94 95
  has_many :milestones,         dependent: :destroy
  has_many :notes,              dependent: :destroy
96 97
  has_many :snippets,           dependent: :destroy, class_name: 'ProjectSnippet'
  has_many :hooks,              dependent: :destroy, class_name: 'ProjectHook'
98
  has_many :protected_branches, dependent: :destroy
99 100
  has_many :project_members, dependent: :destroy, as: :source, class_name: 'ProjectMember'
  has_many :users, through: :project_members
101 102
  has_many :deploy_keys_projects, dependent: :destroy
  has_many :deploy_keys, through: :deploy_keys_projects
Ciro Santilli's avatar
Ciro Santilli committed
103 104
  has_many :users_star_projects, dependent: :destroy
  has_many :starrers, through: :users_star_projects, source: :user
105

106
  delegate :name, to: :owner, allow_nil: true, prefix: true
107
  delegate :members, to: :team, prefix: true
108

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
109
  # Validations
110
  validates :creator, presence: true, on: :create
111
  validates :description, length: { maximum: 2000 }, allow_blank: true
112 113 114 115 116 117 118 119 120 121
  validates :name,
    presence: true,
    length: { within: 0..255 },
    format: { with: Gitlab::Regex.project_name_regex,
              message: Gitlab::Regex.project_regex_message }
  validates :path,
    presence: true,
    length: { within: 0..255 },
    format: { with: Gitlab::Regex.path_regex,
              message: Gitlab::Regex.path_regex_message }
122
  validates :issues_enabled, :merge_requests_enabled,
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
123
            :wiki_enabled, inclusion: { in: [true, false] }
124 125 126
  validates :visibility_level,
    exclusion: { in: gitlab_config.restricted_visibility_levels },
    if: -> { gitlab_config.restricted_visibility_levels.any? }
127
  validates :issues_tracker_id, length: { maximum: 255 }, allow_blank: true
128
  validates :namespace, presence: true
129 130
  validates_uniqueness_of :name, scope: :namespace_id
  validates_uniqueness_of :path, scope: :namespace_id
131
  validates :import_url,
132
    format: { with: URI::regexp(%w(git http https)), message: 'should be a valid url' },
133
    if: :import?
134
  validates :star_count, numericality: { greater_than_or_equal_to: 0 }
135
  validate :check_limit, on: :create
136 137
  validate :avatar_type,
    if: ->(project) { project.avatar && project.avatar_changed? }
138
  validates :avatar, file_size: { maximum: 200.kilobytes.to_i }
139 140

  mount_uploader :avatar, AttachmentUploader
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
141

142
  # Scopes
143
  scope :sorted_by_activity, -> { reorder(last_activity_at: :desc) }
144 145 146
  scope :sorted_by_stars, -> { reorder('projects.star_count DESC') }
  scope :sorted_by_names, -> { joins(:namespace).reorder('namespaces.name ASC, projects.name ASC') }

147 148 149 150
  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 ) }
  scope :in_team, ->(team) { where('projects.id IN (:ids)', ids: team.projects.map(&:id)) }
151
  scope :in_namespace, ->(namespace) { where(namespace_id: namespace.id) }
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
152
  scope :in_group_namespace, -> { joins(:group) }
153
  scope :personal, ->(user) { where(namespace_id: user.namespace_id) }
154
  scope :joined, ->(user) { where('namespace_id != ?', user.namespace_id) }
155 156
  scope :public_only, -> { where(visibility_level: Project::PUBLIC) }
  scope :public_and_internal_only, -> { where(visibility_level: Project.public_and_internal_levels) }
157 158
  scope :non_archived, -> { where(archived: false) }

159 160
  state_machine :import_status, initial: :none do
    event :import_start do
161
      transition [:none, :finished] => :started
162 163 164
    end

    event :import_finish do
165
      transition started: :finished
166 167 168
    end

    event :import_fail do
169
      transition started: :failed
170 171 172
    end

    event :import_retry do
173
      transition failed: :started
174 175 176 177
    end

    state :started
    state :finished
178 179
    state :failed

180
    after_transition any => :started, do: :add_import_job
181 182
  end

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
183
  class << self
184 185 186 187
    def public_and_internal_levels
      [Project::PUBLIC, Project::INTERNAL]
    end

188
    def abandoned
189
      where('projects.last_activity_at < ?', 6.months.ago)
190
    end
191

192 193
    def publicish(user)
      visibility_levels = [Project::PUBLIC]
194
      visibility_levels << Project::INTERNAL if user
195 196
      where(visibility_level: visibility_levels)
    end
197

198
    def with_push
199
      joins(:events).where('events.action = ?', Event::PUSHED)
200 201
    end

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
202
    def active
203
      joins(:issues, :notes, :merge_requests).order('issues.created_at, notes.created_at, merge_requests.created_at DESC')
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
204
    end
205

206
    def search(query)
207 208
      joins(:namespace).where('projects.archived = ?', false).
        where('LOWER(projects.name) LIKE :query OR
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
209 210
              LOWER(projects.path) LIKE :query OR
              LOWER(namespaces.name) LIKE :query OR
211
              LOWER(projects.description) LIKE :query',
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
212
              query: "%#{query.try(:downcase)}%")
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
213
    end
214

215
    def search_by_title(query)
216
      where('projects.archived = ?', false).where('LOWER(projects.name) LIKE :query', query: "%#{query.downcase}%")
217 218
    end

219
    def find_with_namespace(id)
220
      return nil unless id.include?('/')
221

222
      id = id.split('/')
223 224 225 226
      namespace = Namespace.find_by(path: id.first)
      return nil unless namespace

      where(namespace_id: namespace.id).find_by(path: id.second)
227
    end
228

229 230 231
    def visibility_levels
      Gitlab::VisibilityLevel.options
    end
232 233

    def sort(method)
234 235 236 237
      if method == 'repository_size_desc'
        reorder(repository_size: :desc, id: :desc)
      else
        order_by(method)
238 239
      end
    end
240 241
  end

Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
242
  def team
243
    @team ||= ProjectTeam.new(self)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
244 245 246
  end

  def repository
247
    @repository ||= Repository.new(path_with_namespace)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
248 249
  end

250
  def saved?
251
    id && persisted?
252 253
  end

254 255 256 257
  def add_import_job
    RepositoryImportWorker.perform_in(2.seconds, id)
  end

258 259 260 261
  def import?
    import_url.present?
  end

262
  def imported?
263 264 265 266 267 268 269 270 271 272 273 274 275
    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'
276 277
  end

278
  def check_limit
279
    unless creator.can_create_project? or namespace.kind == 'group'
280
      errors[:limit_reached] << ("Your project limit is #{creator.projects_limit} projects! Please contact your administrator to increase it")
281 282
    end
  rescue
Robert Speicher's avatar
Robert Speicher committed
283
    errors[:base] << ("Can't check your ability to create project")
gitlabhq's avatar
gitlabhq committed
284 285
  end

286
  def to_param
287
    namespace.path + '/' + path
288 289
  end

290
  def web_url
291
    [gitlab_config.url, path_with_namespace].join('/')
292 293
  end

294
  def web_url_without_protocol
295
    web_url.split('://')[1]
296 297
  end

298
  def build_commit_note(commit)
299
    notes.new(commit_id: commit.id, noteable_type: 'Commit')
gitlabhq's avatar
gitlabhq committed
300
  end
Nihad Abbasov's avatar
Nihad Abbasov committed
301

302
  def last_activity
303
    last_event
gitlabhq's avatar
gitlabhq committed
304 305 306
  end

  def last_activity_date
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
307
    last_activity_at || updated_at
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
308
  end
309

310 311 312
  def project_id
    self.id
  end
randx's avatar
randx committed
313

314
  def issue_exists?(issue_id)
315
    if default_issues_tracker?
316
      self.issues.where(iid: issue_id).first.present?
317 318 319 320 321
    else
      true
    end
  end

322
  def default_issue_tracker
323
    gitlab_issue_tracker_service ||= create_gitlab_issue_tracker_service
324 325 326 327 328 329 330 331 332 333
  end

  def issues_tracker
    if external_issue_tracker
      external_issue_tracker
    else
      default_issue_tracker
    end
  end

334
  def default_issues_tracker?
335 336 337 338 339
    if external_issue_tracker
      false
    else
      true
    end
340 341 342
  end

  def external_issues_trackers
343
    services.select(&:issue_tracker?).reject(&:default?)
344 345 346 347 348 349
  end

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

Andrew8xx8's avatar
Andrew8xx8 committed
350
  def can_have_issues_tracker_id?
351
    self.issues_enabled && !self.default_issues_tracker?
Andrew8xx8's avatar
Andrew8xx8 committed
352 353
  end

354 355 356 357 358 359 360 361 362 363 364
  def build_missing_services
    available_services_names.each do |service_name|
      service = services.find { |service| service.to_param == service_name }

      # If service is available but missing in db
      # we should create an instance. Ex `create_gitlab_ci_service`
      service = self.send :"create_#{service_name}_service" if service.nil?
    end
  end

  def available_services_names
365
    %w(gitlab_ci campfire hipchat pivotaltracker flowdock assembla
366
       emails_on_push gemnasium slack pushover buildbox bamboo teamcity jira redmine custom_issue_tracker)
367
  end
368 369 370 371

  def gitlab_ci?
    gitlab_ci_service && gitlab_ci_service.active
  end
372

373 374 375 376 377
  def ci_services
    services.select { |service| service.category == :ci }
  end

  def ci_service
378
    @ci_service ||= ci_services.select(&:activated?).first
379 380
  end

381
  def avatar_type
382 383
    unless self.avatar.image?
      self.errors.add :avatar, 'only images allowed'
384 385 386 387 388 389 390 391 392 393 394 395 396 397 398
    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

  # For compatibility with old code
  def code
    path
  end

399
  def items_for(entity)
400 401 402 403 404 405 406
    case entity
    when 'issue' then
      issues
    when 'merge_request' then
      merge_requests
    end
  end
407 408

  def send_move_instructions
409
    NotificationService.new.project_was_moved(self)
410
  end
411 412

  def owner
413 414
    if group
      group
415
    else
416
      namespace.try(:owner)
417 418
    end
  end
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
419 420

  def team_member_by_name_or_email(name = nil, email = nil)
421
    user = users.where('name like ? or email like ?', name, email).first
422
    project_members.where(user: user) if user
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
423 424 425 426
  end

  # Get Team Member record by user id
  def team_member_by_id(user_id)
427
    project_members.find_by(user_id: user_id)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
428 429 430 431 432
  end

  def name_with_namespace
    @name_with_namespace ||= begin
                               if namespace
433
                                 namespace.human_name + ' / ' + name
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
434 435 436 437 438 439 440 441 442 443 444 445 446 447
                               else
                                 name
                               end
                             end
  end

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

448 449 450 451
  def execute_hooks(data, hooks_scope = :push_hooks)
    hooks.send(hooks_scope).each do |hook|
      hook.async_execute(data)
    end
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
452 453 454
  end

  def execute_services(data)
455 456
    services.select(&:active).each do |service|
      service.async_execute(data)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
457 458 459 460
    end
  end

  def update_merge_requests(oldrev, newrev, ref, user)
461 462
    MergeRequests::RefreshService.new(self, user).
      execute(oldrev, newrev, ref)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
463 464 465
  end

  def valid_repo?
466
    repository.exists?
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
467
  rescue
468
    errors.add(:path, 'Invalid repository path')
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
469 470 471 472
    false
  end

  def empty_repo?
473
    !repository.exists? || repository.empty?
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
474 475
  end

476 477 478 479
  def ensure_satellite_exists
    self.satellite.create unless self.satellite.exists?
  end

Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
480 481 482 483 484
  def satellite
    @satellite ||= Gitlab::Satellite::Satellite.new(self)
  end

  def repo
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
485
    repository.raw
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
486 487 488
  end

  def url_to_repo
489
    gitlab_shell.url_to_repo(path_with_namespace)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
490 491 492 493 494 495 496
  end

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

  def repo_exists?
497
    @repo_exists ||= repository.exists?
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
498 499 500 501 502
  rescue
    @repo_exists = false
  end

  def open_branches
503 504 505 506 507 508 509 510 511 512 513 514 515
    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
516 517 518
  end

  def root_ref?(branch)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
519
    repository.root_ref == branch
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
520 521 522 523 524 525 526
  end

  def ssh_url_to_repo
    url_to_repo
  end

  def http_url_to_repo
527
    [gitlab_config.url, '/', path_with_namespace, '.git'].join('')
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
528 529 530
  end

  # Check if current branch name is marked as protected in the system
531
  def protected_branch?(branch_name)
532
    protected_branches_names.include?(branch_name)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
533
  end
534

535
  def developers_can_push_to_protected_branch?(branch_name)
536
    protected_branches.any? { |pb| pb.name == branch_name && pb.developers_can_push }
537 538
  end

539 540 541
  def forked?
    !(forked_project_link.nil? || forked_project_link.forked_from_project.nil?)
  end
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
542

543 544 545 546
  def personal?
    !group
  end

Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
547
  def rename_repo
548
    path_was = previous_changes['path'].first
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
549 550 551 552 553 554 555 556 557
    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
558
        gitlab_shell.mv_repository("#{old_path_with_namespace}.wiki", "#{new_path_with_namespace}.wiki")
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
559
        gitlab_shell.rm_satellites(old_path_with_namespace)
560
        ensure_satellite_exists
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
561
        send_move_instructions
562
        reset_events_cache
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
563
      rescue
Johannes Schleifenbaum's avatar
Johannes Schleifenbaum committed
564
        # Returning false does not rollback after_* transaction but gives
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
565 566 567 568 569 570 571 572 573
        # 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
574

Kirill Zaitsev's avatar
Kirill Zaitsev committed
575 576 577 578 579 580 581 582 583 584
  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

585 586 587 588 589
  # 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
590
  # * when the project avatar changes
591 592 593 594 595 596 597 598 599
  # 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
600 601

  def project_member(user)
602
    project_members.where(user_id: user).first
603
  end
604 605 606 607

  def default_branch
    @default_branch ||= repository.root_ref if repository.exists?
  end
608 609 610 611 612

  def reload_default_branch
    @default_branch = nil
    default_branch
  end
613

614 615 616
  def visibility_level_field
    visibility_level
  end
617 618 619 620 621 622 623 624

  def archive!
    update_attribute(:archived, true)
  end

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

626 627 628 629
  def change_head(branch)
    gitlab_shell.update_repository_head(self.path_with_namespace, branch)
    reload_default_branch
  end
630 631 632 633

  def forked_from?(project)
    forked? && project == forked_from_project
  end
634 635 636 637

  def update_repository_size
    update_attribute(:repository_size, repository.size)
  end
638 639 640 641

  def forks_count
    ForkedProjectLink.where(forked_from_project_id: self.id).count
  end
642 643 644 645

  def find_label(name)
    labels.find_by(name: name)
  end
646 647 648 649

  def origin_merge_requests
    merge_requests.where(source_project_id: self.id)
  end
650 651 652 653 654

  def create_repository
    if gitlab_shell.add_repository(path_with_namespace)
      true
    else
655
      errors.add(:base, 'Failed to create repository')
656 657 658 659 660 661 662 663 664 665 666 667
      false
    end
  end

  def repository_exists?
    !!repository.exists?
  end

  def create_wiki
    ProjectWiki.new(self, self.owner).wiki
    true
  rescue ProjectWiki::CouldNotCreateWikiError => ex
668
    errors.add(:base, 'Failed create wiki')
669 670
    false
  end
gitlabhq's avatar
gitlabhq committed
671
end