project.rb 8.5 KB
Newer Older
D
Dmitriy Zaporozhets 已提交
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
D
Dmitriy Zaporozhets 已提交
18
#  namespace_id           :integer
D
Dmitriy Zaporozhets 已提交
19 20
#

G
gitlabhq 已提交
21 22 23
require "grit"

class Project < ActiveRecord::Base
24
  include Repository
R
randx 已提交
25
  include PushObserver
26 27 28
  include Authority
  include Team

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

32
  attr_accessible :namespace_id, :owner_id, as: :admin
33

N
Nihad Abbasov 已提交
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

  # TODO: replace owner with creator.
  # With namespaces a project owner will be a namespace owner
  # so this field makes sense only for global projects
43 44 45 46 47 48 49 50 51 52 53 54 55
  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
56
  has_one :last_event, class_name: 'Event', order: 'events.created_at DESC', foreign_key: 'project_id'
D
Dmitriy Zaporozhets 已提交
57
  has_one :gitlab_ci_service, dependent: :destroy
A
Aleksei Kvitinskii 已提交
58

59 60
  delegate :name, to: :owner, allow_nil: true, prefix: true

A
Andrey Kumanyaev 已提交
61 62 63
  # Validations
  validates :owner, presence: true
  validates :description, length: { within: 0..2000 }
64 65
  validates :name, presence: true, length: { within: 0..255 }
  validates :path, presence: true, length: { within: 0..255 },
66
            format: { with: Gitlab::Regex.path_regex,
A
Andrey Kumanyaev 已提交
67 68 69
                      message: "only letters, digits & '_' '-' '.' allowed. Letter should be first" }
  validates :issues_enabled, :wall_enabled, :merge_requests_enabled,
            :wiki_enabled, inclusion: { in: [true, false] }
70

71 72 73
  validates_uniqueness_of :name, scope: :namespace_id
  validates_uniqueness_of :path, scope: :namespace_id

A
Andrey Kumanyaev 已提交
74 75
  validate :check_limit, :repo_name

76
  # Scopes
77
  scope :public_only, where(private_flag: false)
D
Dmitriy Zaporozhets 已提交
78 79
  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 ) }
80
  scope :sorted_by_activity, ->() { order("(SELECT max(events.created_at) FROM events WHERE events.project_id = projects.id) DESC") }
D
Dmitriy Zaporozhets 已提交
81 82
  scope :personal, ->(user) { where(namespace_id: user.namespace_id) }
  scope :joined, ->(user) { where("namespace_id != ?", user.namespace_id) }
G
gitlabhq 已提交
83

A
Andrey Kumanyaev 已提交
84
  class << self
85 86 87 88 89
    def authorized_for user
      projects = includes(:users_projects, :namespace)
      projects = projects.where("users_projects.user_id = :user_id or projects.owner_id = :user_id or namespaces.owner_id = :user_id", user_id: user.id)
    end

A
Andrey Kumanyaev 已提交
90 91 92
    def active
      joins(:issues, :notes, :merge_requests).order("issues.created_at, notes.created_at, merge_requests.created_at DESC")
    end
93

A
Andrey Kumanyaev 已提交
94
    def search query
95
      where("projects.name LIKE :query OR projects.path LIKE :query", query: "%#{query}%")
A
Andrey Kumanyaev 已提交
96
    end
97

98 99 100 101 102 103 104 105 106 107
    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

A
Andrey Kumanyaev 已提交
108
    def create_by_user(params, user)
109
      namespace_id = params.delete(:namespace_id)
110

A
Andrey Kumanyaev 已提交
111 112 113
      project = Project.new params

      Project.transaction do
114

115
        # Parametrize path for project
116
        #
117 118 119 120
        # Ex.
        #  'GitLab HQ'.parameterize => "gitlab-hq"
        #
        project.path = project.name.dup.parameterize
121

A
Andrey Kumanyaev 已提交
122
        project.owner = user
123 124 125

        # Apply namespace if user has access to it
        # else fallback to user namespace
126 127 128 129 130 131 132 133
        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
134 135 136
          end
        end

A
Andrey Kumanyaev 已提交
137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154
        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
155 156
    end

A
Andrey Kumanyaev 已提交
157 158 159
    def access_options
      UsersProject.access_roles
    end
160 161 162 163 164 165 166 167
  end

  def git_error?
    error_code == :gitolite
  end

  def saved?
    id && valid?
168 169
  end

170 171 172 173 174
  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
R
Robert Speicher 已提交
175
    errors[:base] << ("Can't check your ability to create project")
G
gitlabhq 已提交
176 177
  end

178
  def repo_name
179 180 181 182
    denied_paths = %w(gitolite-admin groups projects dashboard)

    if denied_paths.include?(path)
      errors.add(:path, "like #{path} is not allowed")
183 184
    end
  end
V
Valeriy Sizov 已提交
185

186
  def to_param
187
    if namespace
188
      namespace.path + "/" + path
189
    else
190
      path
191
    end
192 193
  end

194
  def web_url
195
    [Gitlab.config.url, path_with_namespace].join("/")
196 197
  end

G
gitlabhq 已提交
198
  def common_notes
199
    notes.where(noteable_type: ["", nil]).inc_author_project
G
gitlabhq 已提交
200 201
  end

202
  def build_commit_note(commit)
203
    notes.new(noteable_id: commit.id, noteable_type: "Commit")
G
gitlabhq 已提交
204
  end
N
Nihad Abbasov 已提交
205

206
  def commit_notes(commit)
207
    notes.where(noteable_id: commit.id, noteable_type: "Commit", line_code: nil)
D
Dmitriy Zaporozhets 已提交
208 209 210
  end

  def commit_line_notes(commit)
V
Valeriy Sizov 已提交
211
    notes.where(noteable_id: commit.id, noteable_type: "Commit").where("line_code IS NOT NULL")
G
gitlabhq 已提交
212
  end
N
Nihad Abbasov 已提交
213

G
gitlabhq 已提交
214 215 216 217 218 219 220 221
  def public?
    !private_flag
  end

  def private?
    private_flag
  end

N
Nihad Abbasov 已提交
222
  def last_activity
223
    last_event
G
gitlabhq 已提交
224 225 226
  end

  def last_activity_date
227
    last_event.try(:created_at) || updated_at
D
Dmitriy Zaporozhets 已提交
228
  end
229

D
Dmitriy Zaporozhets 已提交
230 231 232
  def project_id
    self.id
  end
R
randx 已提交
233 234 235 236

  def issues_labels
    issues.tag_counts_on(:labels)
  end
237 238 239 240

  def services
    [gitlab_ci_service].compact
  end
D
Dmitriy Zaporozhets 已提交
241 242 243 244

  def gitlab_ci?
    gitlab_ci_service && gitlab_ci_service.active
  end
245 246 247

  def path_with_namespace
    if namespace
248
      namespace.path + '/' + path
249 250 251 252
    else
      path
    end
  end
253

254 255 256
  # For compatibility with old code
  def code
    path
257
  end
258 259 260 261 262 263 264 265 266

  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) || ''

267 268 269 270 271
      old_repo = if old_dir.present?
                   File.join(old_dir, self.path)
                 else
                   self.path
                 end
272

273 274
      Gitlab::ProjectMover.new(self, old_dir, new_dir).execute

275
      git_host.move_repository(old_repo, self)
276

277 278 279
      save!
    end
  end
280 281 282 283 284 285 286 287 288 289

  def name_with_namespace
    @name_with_namespace ||= begin
                               if namespace
                                 namespace.human_name + " / " + name
                               else
                                 name
                               end
                             end
  end
D
Dmitriy Zaporozhets 已提交
290 291 292 293 294 295 296 297 298

  def items_for entity
    case entity
    when 'issue' then
      issues
    when 'merge_request' then
      merge_requests
    end
  end
299 300 301 302

  def namespace_owner
    namespace.try(:owner)
  end
303 304 305 306 307 308 309 310

  def chief
    if namespace
      namespace_owner
    else
      owner
    end
  end
G
gitlabhq 已提交
311
end