project.rb 19.2 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
Jeremy's avatar
Jeremy committed
72
  has_one :asana_service, dependent: :destroy
73
  has_one :gemnasium_service, dependent: :destroy
74
  has_one :slack_service, dependent: :destroy
Keith Pitt's avatar
Keith Pitt committed
75
  has_one :buildbox_service, dependent: :destroy
Drew Blessing's avatar
Drew Blessing committed
76
  has_one :bamboo_service, dependent: :destroy
77
  has_one :teamcity_service, dependent: :destroy
78
  has_one :pushover_service, dependent: :destroy
79 80
  has_one :jira_service, dependent: :destroy
  has_one :redmine_service, dependent: :destroy
81
  has_one :custom_issue_tracker_service, dependent: :destroy
82
  has_one :gitlab_issue_tracker_service, dependent: :destroy
83

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

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

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

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

141
  mount_uploader :avatar, AvatarUploader
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
142

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

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

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

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

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

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

    state :started
    state :finished
179 180
    state :failed

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

  def issues_tracker
    if external_issue_tracker
      external_issue_tracker
    else
      default_issue_tracker
    end
  end

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

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

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

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

355
  def build_missing_services
356 357
    services_templates = Service.where(template: true)

358
    Service.available_services_names.each do |service_name|
359
      service = find_service(services, service_name)
360 361

      # If service is available but missing in db
362 363 364 365 366 367 368 369 370 371 372
      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
373 374 375
    end
  end

376 377 378
  def find_service(list, name)
    list.find { |service| service.to_param == name }
  end
379 380 381 382

  def gitlab_ci?
    gitlab_ci_service && gitlab_ci_service.active
  end
383

384 385 386 387 388
  def ci_services
    services.select { |service| service.category == :ci }
  end

  def ci_service
389
    @ci_service ||= ci_services.select(&:activated?).first
390 391
  end

392
  def avatar_type
393 394
    unless self.avatar.image?
      self.errors.add :avatar, 'only images allowed'
395 396 397 398 399 400 401 402 403 404 405 406 407 408 409
    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

410
  def items_for(entity)
411 412 413 414 415 416 417
    case entity
    when 'issue' then
      issues
    when 'merge_request' then
      merge_requests
    end
  end
418 419

  def send_move_instructions
420
    NotificationService.new.project_was_moved(self)
421
  end
422 423

  def owner
424 425
    if group
      group
426
    else
427
      namespace.try(:owner)
428 429
    end
  end
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
430 431

  def team_member_by_name_or_email(name = nil, email = nil)
432
    user = users.where('name like ? or email like ?', name, email).first
433
    project_members.where(user: user) if user
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
434 435 436 437
  end

  # Get Team Member record by user id
  def team_member_by_id(user_id)
438
    project_members.find_by(user_id: user_id)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
439 440 441 442 443
  end

  def name_with_namespace
    @name_with_namespace ||= begin
                               if namespace
444
                                 namespace.human_name + ' / ' + name
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
445 446 447 448 449 450 451 452 453 454 455 456 457 458
                               else
                                 name
                               end
                             end
  end

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

459 460 461 462
  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
463 464 465
  end

  def execute_services(data)
466 467
    services.select(&:active).each do |service|
      service.async_execute(data)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
468 469 470 471
    end
  end

  def update_merge_requests(oldrev, newrev, ref, user)
472 473
    MergeRequests::RefreshService.new(self, user).
      execute(oldrev, newrev, ref)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
474 475 476
  end

  def valid_repo?
477
    repository.exists?
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
478
  rescue
479
    errors.add(:path, 'Invalid repository path')
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
480 481 482 483
    false
  end

  def empty_repo?
484
    !repository.exists? || repository.empty?
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
485 486
  end

487 488 489 490
  def ensure_satellite_exists
    self.satellite.create unless self.satellite.exists?
  end

Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
491 492 493 494 495
  def satellite
    @satellite ||= Gitlab::Satellite::Satellite.new(self)
  end

  def repo
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
496
    repository.raw
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
497 498 499
  end

  def url_to_repo
500
    gitlab_shell.url_to_repo(path_with_namespace)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
501 502 503 504 505 506 507
  end

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

  def repo_exists?
508
    @repo_exists ||= repository.exists?
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
509 510 511 512 513
  rescue
    @repo_exists = false
  end

  def open_branches
514 515 516 517 518 519 520 521 522 523 524 525 526
    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
527 528 529
  end

  def root_ref?(branch)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
530
    repository.root_ref == branch
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
531 532 533 534 535 536 537
  end

  def ssh_url_to_repo
    url_to_repo
  end

  def http_url_to_repo
538
    [gitlab_config.url, '/', path_with_namespace, '.git'].join('')
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
539 540 541
  end

  # Check if current branch name is marked as protected in the system
542
  def protected_branch?(branch_name)
543
    protected_branches_names.include?(branch_name)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
544
  end
545

546
  def developers_can_push_to_protected_branch?(branch_name)
547
    protected_branches.any? { |pb| pb.name == branch_name && pb.developers_can_push }
548 549
  end

550 551 552
  def forked?
    !(forked_project_link.nil? || forked_project_link.forked_from_project.nil?)
  end
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
553

554 555 556 557
  def personal?
    !group
  end

Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
558
  def rename_repo
559
    path_was = previous_changes['path'].first
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
560 561 562 563 564 565 566 567 568
    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
569
        gitlab_shell.mv_repository("#{old_path_with_namespace}.wiki", "#{new_path_with_namespace}.wiki")
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
570
        gitlab_shell.rm_satellites(old_path_with_namespace)
571
        ensure_satellite_exists
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
572
        send_move_instructions
573
        reset_events_cache
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
574
      rescue
Johannes Schleifenbaum's avatar
Johannes Schleifenbaum committed
575
        # Returning false does not rollback after_* transaction but gives
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
576 577 578 579 580 581 582 583 584
        # 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
585

Kirill Zaitsev's avatar
Kirill Zaitsev committed
586 587 588 589 590 591 592 593 594 595
  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

596 597 598 599 600
  # 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
601
  # * when the project avatar changes
602 603 604 605 606 607 608 609 610
  # 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
611 612

  def project_member(user)
613
    project_members.where(user_id: user).first
614
  end
615 616 617 618

  def default_branch
    @default_branch ||= repository.root_ref if repository.exists?
  end
619 620 621 622 623

  def reload_default_branch
    @default_branch = nil
    default_branch
  end
624

625 626 627
  def visibility_level_field
    visibility_level
  end
628 629 630 631 632 633 634 635

  def archive!
    update_attribute(:archived, true)
  end

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

637 638 639 640
  def change_head(branch)
    gitlab_shell.update_repository_head(self.path_with_namespace, branch)
    reload_default_branch
  end
641 642 643 644

  def forked_from?(project)
    forked? && project == forked_from_project
  end
645 646 647 648

  def update_repository_size
    update_attribute(:repository_size, repository.size)
  end
649 650 651 652

  def forks_count
    ForkedProjectLink.where(forked_from_project_id: self.id).count
  end
653 654 655 656

  def find_label(name)
    labels.find_by(name: name)
  end
657 658 659 660

  def origin_merge_requests
    merge_requests.where(source_project_id: self.id)
  end
661 662 663 664 665

  def create_repository
    if gitlab_shell.add_repository(path_with_namespace)
      true
    else
666
      errors.add(:base, 'Failed to create repository')
667 668 669 670 671 672 673 674 675 676 677 678
      false
    end
  end

  def repository_exists?
    !!repository.exists?
  end

  def create_wiki
    ProjectWiki.new(self, self.owner).wiki
    true
  rescue ProjectWiki::CouldNotCreateWikiError => ex
679
    errors.add(:base, 'Failed create wiki')
680 681
    false
  end
gitlabhq's avatar
gitlabhq committed
682
end