project.rb 7.85 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
# == Schema Information
#
# Table name: projects
#
#  id                     :integer          not null, primary key
#  name                   :string(255)
#  path                   :string(255)
#  description            :text
#  created_at             :datetime         not null
#  updated_at             :datetime         not null
#  private_flag           :boolean          default(TRUE), not null
#  owner_id               :integer
#  default_branch         :string(255)
#  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
18
#  namespace_id           :integer
19 20
#

gitlabhq's avatar
gitlabhq committed
21 22 23
require "grit"

class Project < ActiveRecord::Base
24
  include Repository
randx's avatar
randx committed
25
  include PushObserver
26 27 28
  include Authority
  include Team

29
  attr_accessible :name, :path, :description, :default_branch, :issues_enabled,
30 31 32 33
                  :wall_enabled, :merge_requests_enabled, :wiki_enabled, as: [:default, :admin]

  attr_accessible :namespace_id, as: :admin

Nihad Abbasov's avatar
Nihad Abbasov committed
34
  attr_accessor :error_code
35

36
  # Relations
37
  belongs_to :group, foreign_key: "namespace_id", conditions: "type = 'Group'"
38
  belongs_to :namespace
39 40 41 42 43 44 45 46 47 48 49 50 51
  belongs_to :owner, class_name: "User"
  has_many :users,          through: :users_projects
  has_many :events,         dependent: :destroy
  has_many :merge_requests, dependent: :destroy
  has_many :issues,         dependent: :destroy, order: "closed, created_at DESC"
  has_many :milestones,     dependent: :destroy
  has_many :users_projects, dependent: :destroy
  has_many :notes,          dependent: :destroy
  has_many :snippets,       dependent: :destroy
  has_many :deploy_keys,    dependent: :destroy, foreign_key: "project_id", class_name: "Key"
  has_many :hooks,          dependent: :destroy, class_name: "ProjectHook"
  has_many :wikis,          dependent: :destroy
  has_many :protected_branches, dependent: :destroy
52
  has_one :last_event, class_name: 'Event', order: 'events.created_at DESC', foreign_key: 'project_id'
53
  has_one :gitlab_ci_service, dependent: :destroy
Aleksei Kvitinskii's avatar
Aleksei Kvitinskii committed
54

55 56
  delegate :name, to: :owner, allow_nil: true, prefix: true

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
57 58 59
  # Validations
  validates :owner, presence: true
  validates :description, length: { within: 0..2000 }
60 61
  validates :name, presence: true, length: { within: 0..255 }
  validates :path, presence: true, length: { within: 0..255 },
62
            format: { with: Gitlab::Regex.path_regex,
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
63 64 65
                      message: "only letters, digits & '_' '-' '.' allowed. Letter should be first" }
  validates :issues_enabled, :wall_enabled, :merge_requests_enabled,
            :wiki_enabled, inclusion: { in: [true, false] }
66

67 68 69
  validates_uniqueness_of :name, scope: :namespace_id
  validates_uniqueness_of :path, scope: :namespace_id

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
70 71
  validate :check_limit, :repo_name

72
  # Scopes
73
  scope :public_only, where(private_flag: false)
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
74 75
  scope :without_user, ->(user)  { where("id NOT IN (:ids)", ids: user.projects.map(&:id) ) }
  scope :not_in_group, ->(group) { where("id NOT IN (:ids)", ids: group.project_ids ) }
76
  scope :authorized_for, ->(user) { joins(:users_projects) { where(user_id: user.id) } }
gitlabhq's avatar
gitlabhq committed
77

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
78 79 80 81
  class << self
    def active
      joins(:issues, :notes, :merge_requests).order("issues.created_at, notes.created_at, merge_requests.created_at DESC")
    end
82

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
83
    def search query
84
      where("projects.name LIKE :query OR projects.path LIKE :query", query: "%#{query}%")
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
85
    end
86

87 88 89 90 91 92 93 94 95 96
    def find_with_namespace(id)
      if id.include?("/")
        id = id.split("/")
        namespace_id = Namespace.find_by_path(id.first).id
        where(namespace_id: namespace_id).find_by_path(id.last)
      else
        find_by_path(id)
      end
    end

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
97
    def create_by_user(params, user)
98
      namespace_id = params.delete(:namespace_id)
99

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
100 101 102
      project = Project.new params

      Project.transaction do
103

104
        # Parametrize path for project
105
        #
106 107 108 109
        # Ex.
        #  'GitLab HQ'.parameterize => "gitlab-hq"
        #
        project.path = project.name.dup.parameterize
110

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
111
        project.owner = user
112 113 114

        # Apply namespace if user has access to it
        # else fallback to user namespace
115 116 117 118 119 120 121 122
        if namespace_id != Namespace.global_id
          project.namespace_id = user.namespace_id

          if namespace_id
            group = Group.find_by_id(namespace_id)
            if user.can? :manage_group, group
              project.namespace_id = namespace_id
            end
123 124 125
          end
        end

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143
        project.save!

        # Add user as project master
        project.users_projects.create!(project_access: UsersProject::MASTER, user: user)

        # when project saved no team member exist so
        # project repository should be updated after first user add
        project.update_repository
      end

      project
    rescue Gitlab::Gitolite::AccessDenied => ex
      project.error_code = :gitolite
      project
    rescue => ex
      project.error_code = :db
      project.errors.add(:base, "Can't save project. Please try again later")
      project
144 145
    end

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
146 147 148
    def access_options
      UsersProject.access_roles
    end
149 150 151 152 153 154 155 156
  end

  def git_error?
    error_code == :gitolite
  end

  def saved?
    id && valid?
157 158
  end

159 160 161 162 163
  def check_limit
    unless owner.can_create_project?
      errors[:base] << ("Your own projects limit is #{owner.projects_limit}! Please contact administrator to increase it")
    end
  rescue
Robert Speicher's avatar
Robert Speicher committed
164
    errors[:base] << ("Can't check your ability to create project")
gitlabhq's avatar
gitlabhq committed
165 166
  end

167
  def repo_name
168 169 170 171
    denied_paths = %w(gitolite-admin groups projects dashboard)

    if denied_paths.include?(path)
      errors.add(:path, "like #{path} is not allowed")
172 173
    end
  end
Valeriy Sizov's avatar
Valeriy Sizov committed
174

175
  def to_param
176
    if namespace
177
      namespace.path + "/" + path
178
    else
179
      path
180
    end
181 182
  end

183
  def web_url
184
    [Gitlab.config.url, path].join("/")
185 186
  end

gitlabhq's avatar
gitlabhq committed
187
  def common_notes
188
    notes.where(noteable_type: ["", nil]).inc_author_project
gitlabhq's avatar
gitlabhq committed
189 190
  end

191
  def build_commit_note(commit)
192
    notes.new(noteable_id: commit.id, noteable_type: "Commit")
gitlabhq's avatar
gitlabhq committed
193
  end
Nihad Abbasov's avatar
Nihad Abbasov committed
194

195
  def commit_notes(commit)
196
    notes.where(noteable_id: commit.id, noteable_type: "Commit", line_code: nil)
197 198 199
  end

  def commit_line_notes(commit)
Valeriy Sizov's avatar
Valeriy Sizov committed
200
    notes.where(noteable_id: commit.id, noteable_type: "Commit").where("line_code IS NOT NULL")
gitlabhq's avatar
gitlabhq committed
201
  end
Nihad Abbasov's avatar
Nihad Abbasov committed
202

gitlabhq's avatar
gitlabhq committed
203 204 205 206 207 208 209 210
  def public?
    !private_flag
  end

  def private?
    private_flag
  end

211
  def last_activity
212
    last_event
gitlabhq's avatar
gitlabhq committed
213 214 215
  end

  def last_activity_date
216
    last_event.try(:created_at) || updated_at
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
217
  end
218

219
  def wiki_notes
Nihad Abbasov's avatar
Nihad Abbasov committed
220
    Note.where(noteable_id: wikis.pluck(:id), noteable_type: 'Wiki', project_id: self.id)
221 222
  end

223 224 225
  def project_id
    self.id
  end
randx's avatar
randx committed
226 227 228 229

  def issues_labels
    issues.tag_counts_on(:labels)
  end
230 231 232 233

  def services
    [gitlab_ci_service].compact
  end
234 235 236 237

  def gitlab_ci?
    gitlab_ci_service && gitlab_ci_service.active
  end
238 239 240

  def path_with_namespace
    if namespace
241
      namespace.path + '/' + path
242 243 244 245
    else
      path
    end
  end
246

247 248 249
  # For compatibility with old code
  def code
    path
250
  end
251 252 253 254 255 256 257 258 259

  def transfer(new_namespace)
    Project.transaction do
      old_namespace = namespace
      self.namespace = new_namespace

      old_dir = old_namespace.try(:path) || ''
      new_dir = new_namespace.try(:path) || ''

260 261 262 263 264
      old_repo = if old_dir.present?
                   File.join(old_dir, self.path)
                 else
                   self.path
                 end
265

266 267
      Gitlab::ProjectMover.new(self, old_dir, new_dir).execute

268
      git_host.move_repository(old_repo, self)
269

270 271 272
      save!
    end
  end
273 274 275 276 277 278 279 280 281 282

  def name_with_namespace
    @name_with_namespace ||= begin
                               if namespace
                                 namespace.human_name + " / " + name
                               else
                                 name
                               end
                             end
  end
283 284 285 286 287 288 289 290 291

  def items_for entity
    case entity
    when 'issue' then
      issues
    when 'merge_request' then
      merge_requests
    end
  end
gitlabhq's avatar
gitlabhq committed
292
end