project.rb 8.17 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
                  :wall_enabled, :merge_requests_enabled, :wiki_enabled, as: [:default, :admin]

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

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 :sorted_by_activity, ->() { order("(SELECT max(events.created_at) FROM events WHERE events.project_id = projects.id) DESC") }
77 78
  scope :personal, ->(user) { where(namespace_id: user.namespace_id) }
  scope :joined, ->(user) { where("namespace_id != ?", user.namespace_id) }
gitlabhq's avatar
gitlabhq committed
79

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
80
  class << self
81 82 83 84 85
    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

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
86 87 88
    def active
      joins(:issues, :notes, :merge_requests).order("issues.created_at, notes.created_at, merge_requests.created_at DESC")
    end
89

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
90
    def search query
91
      where("projects.name LIKE :query OR projects.path LIKE :query", query: "%#{query}%")
Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
92
    end
93

94 95 96 97 98 99 100 101 102 103
    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
104
    def create_by_user(params, user)
105
      namespace_id = params.delete(:namespace_id)
106

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
107 108 109
      project = Project.new params

      Project.transaction do
110

111
        # Parametrize path for project
112
        #
113 114 115 116
        # Ex.
        #  'GitLab HQ'.parameterize => "gitlab-hq"
        #
        project.path = project.name.dup.parameterize
117

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
118
        project.owner = user
119 120 121

        # Apply namespace if user has access to it
        # else fallback to user namespace
122 123 124 125 126 127 128 129
        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
130 131 132
          end
        end

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150
        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
151 152
    end

Andrey Kumanyaev's avatar
Andrey Kumanyaev committed
153 154 155
    def access_options
      UsersProject.access_roles
    end
156 157 158 159 160 161 162 163
  end

  def git_error?
    error_code == :gitolite
  end

  def saved?
    id && valid?
164 165
  end

166 167 168 169 170
  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
171
    errors[:base] << ("Can't check your ability to create project")
gitlabhq's avatar
gitlabhq committed
172 173
  end

174
  def repo_name
175 176 177 178
    denied_paths = %w(gitolite-admin groups projects dashboard)

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

182
  def to_param
183
    if namespace
184
      namespace.path + "/" + path
185
    else
186
      path
187
    end
188 189
  end

190
  def web_url
191
    [Gitlab.config.url, path].join("/")
192 193
  end

gitlabhq's avatar
gitlabhq committed
194
  def common_notes
195
    notes.where(noteable_type: ["", nil]).inc_author_project
gitlabhq's avatar
gitlabhq committed
196 197
  end

198
  def build_commit_note(commit)
199
    notes.new(noteable_id: commit.id, noteable_type: "Commit")
gitlabhq's avatar
gitlabhq committed
200
  end
Nihad Abbasov's avatar
Nihad Abbasov committed
201

202
  def commit_notes(commit)
203
    notes.where(noteable_id: commit.id, noteable_type: "Commit", line_code: nil)
204 205 206
  end

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

gitlabhq's avatar
gitlabhq committed
210 211 212 213 214 215 216 217
  def public?
    !private_flag
  end

  def private?
    private_flag
  end

218
  def last_activity
219
    last_event
gitlabhq's avatar
gitlabhq committed
220 221 222
  end

  def last_activity_date
223
    last_event.try(:created_at) || updated_at
Dmitriy Zaporozhets's avatar
Dmitriy Zaporozhets committed
224
  end
225

226 227 228
  def project_id
    self.id
  end
randx's avatar
randx committed
229 230 231 232

  def issues_labels
    issues.tag_counts_on(:labels)
  end
233 234 235 236

  def services
    [gitlab_ci_service].compact
  end
237 238 239 240

  def gitlab_ci?
    gitlab_ci_service && gitlab_ci_service.active
  end
241 242 243

  def path_with_namespace
    if namespace
244
      namespace.path + '/' + path
245 246 247 248
    else
      path
    end
  end
249

250 251 252
  # For compatibility with old code
  def code
    path
253
  end
254 255 256 257 258 259 260 261 262

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

263 264 265 266 267
      old_repo = if old_dir.present?
                   File.join(old_dir, self.path)
                 else
                   self.path
                 end
268

269 270
      Gitlab::ProjectMover.new(self, old_dir, new_dir).execute

271
      git_host.move_repository(old_repo, self)
272

273 274 275
      save!
    end
  end
276 277 278 279 280 281 282 283 284 285

  def name_with_namespace
    @name_with_namespace ||= begin
                               if namespace
                                 namespace.human_name + " / " + name
                               else
                                 name
                               end
                             end
  end
286 287 288 289 290 291 292 293 294

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