gitlabhq/app/models/project.rb

364 lines
9.6 KiB
Ruby
Raw Normal View History

2011-10-09 00:36:38 +03:00
require "grit"
class Project < ActiveRecord::Base
2011-10-09 22:36:57 +03:00
belongs_to :owner, :class_name => "User"
2011-11-28 09:39:43 +02:00
has_many :merge_requests, :dependent => :destroy
2011-10-15 19:56:53 +03:00
has_many :issues, :dependent => :destroy, :order => "position"
2011-10-09 00:36:38 +03:00
has_many :users_projects, :dependent => :destroy
has_many :users, :through => :users_projects
has_many :notes, :dependent => :destroy
2011-10-17 00:07:10 +03:00
has_many :snippets, :dependent => :destroy
has_many :deploy_keys, :dependent => :destroy, :foreign_key => "project_id", :class_name => "Key"
has_many :web_hooks, :dependent => :destroy
has_many :protected_branches, :dependent => :destroy
2011-10-09 00:36:38 +03:00
2011-11-04 09:42:36 +02:00
acts_as_taggable
2011-10-09 00:36:38 +03:00
validates :name,
:uniqueness => true,
:presence => true,
:length => { :within => 0..255 }
validates :path,
:uniqueness => true,
:presence => true,
:format => { :with => /^[a-zA-Z0-9_\-\.]*$/,
:message => "only letters, digits & '_' '-' '.' allowed" },
2011-10-09 00:36:38 +03:00
:length => { :within => 0..255 }
2011-10-09 00:36:38 +03:00
validates :description,
:length => { :within => 0..2000 }
validates :code,
:presence => true,
:uniqueness => true,
:format => { :with => /^[a-zA-Z0-9_\-\.]*$/,
:message => "only letters, digits & '_' '-' '.' allowed" },
2011-10-18 11:21:44 +03:00
:length => { :within => 3..255 }
2011-10-09 00:36:38 +03:00
2011-10-09 22:36:57 +03:00
validates :owner,
:presence => true
2011-10-09 11:15:01 -07:00
validate :check_limit
2011-10-21 21:40:36 +03:00
validate :repo_name
2011-12-05 09:43:53 +02:00
after_destroy :destroy_repository
after_save :update_repository
2011-10-09 00:36:38 +03:00
2011-10-09 04:05:31 -07:00
attr_protected :private_flag, :owner_id
2011-10-09 00:36:38 +03:00
scope :public_only, where(:private_flag => false)
2012-02-11 19:56:18 +02:00
scope :without_user, lambda { |user| where("id not in (:ids)", :ids => user.projects.map(&:id) ) }
2011-10-09 00:36:38 +03:00
def self.active
joins(:issues, :notes, :merge_requests).order("issues.created_at, notes.created_at, merge_requests.created_at DESC")
end
def self.access_options
2012-02-16 09:03:55 +02:00
UsersProject.access_roles
end
2011-11-11 00:51:19 +02:00
def repository
@repository ||= Repository.new(self)
end
delegate :repo,
:url_to_repo,
:path_to_repo,
2011-12-05 09:43:53 +02:00
:update_repository,
:destroy_repository,
2011-11-11 00:51:19 +02:00
:tags,
:repo_exists?,
:commit,
:commits,
:commits_with_refs,
2011-11-11 00:51:19 +02:00
:tree,
:heads,
:commits_since,
:fresh_commits,
:commits_between,
2011-11-11 00:51:19 +02:00
:to => :repository, :prefix => nil
2011-10-09 00:36:38 +03:00
def to_param
code
end
def web_url
[GIT_HOST['host'], code].join("/")
end
def execute_web_hooks(oldrev, newrev, ref)
ref_parts = ref.split('/')
# Return if this is not a push to a branch (e.g. new commits)
return if ref_parts[1] !~ /heads/ || oldrev == "00000000000000000000000000000000"
data = web_hook_data(oldrev, newrev, ref)
web_hooks.each { |web_hook| web_hook.execute(data) }
end
def web_hook_data(oldrev, newrev, ref)
data = {
before: oldrev,
after: newrev,
ref: ref,
repository: {
name: name,
url: web_url,
description: description,
homepage: web_url,
private: private?
},
commits: []
}
commits_between(oldrev, newrev).each do |commit|
data[:commits] << {
id: commit.id,
message: commit.safe_message,
timestamp: commit.date.xmlschema,
url: "http://#{GIT_HOST['host']}/#{code}/commits/#{commit.id}",
author: {
name: commit.author_name,
email: commit.author_email
}
}
end
data
end
def open_branches
if protected_branches.empty?
self.repo.heads
else
pnames = protected_branches.map(&:name)
self.repo.heads.reject { |h| pnames.include?(h.name) }
end.sort_by(&:name)
end
2011-11-06 22:38:08 +02:00
def team_member_by_name_or_email(email = nil, name = nil)
user = users.where("email like ? or name like ?", email, name).first
users_projects.find_by_user_id(user.id) if user
end
2011-12-13 23:24:31 +02:00
def team_member_by_id(user_id)
users_projects.find_by_user_id(user_id)
end
def fresh_merge_requests(n)
merge_requests.includes(:project, :author).order("created_at desc").first(n)
end
2011-11-15 04:09:07 -05:00
def fresh_issues(n)
issues.includes(:project, :author).order("created_at desc").first(n)
end
def fresh_notes(n)
notes.inc_author_project.order("created_at desc").first(n)
end
2011-10-09 00:36:38 +03:00
def common_notes
2011-11-15 04:09:07 -05:00
notes.where(:noteable_type => ["", nil]).inc_author_project
2011-10-09 00:36:38 +03:00
end
def build_commit_note(commit)
notes.new(:noteable_id => commit.id, :noteable_type => "Commit")
2011-10-09 00:36:38 +03:00
end
def commit_notes(commit)
2012-01-10 22:08:46 +02:00
notes.where(:noteable_id => commit.id, :noteable_type => "Commit", :line_code => nil)
end
def commit_line_notes(commit)
2012-01-19 21:44:10 +02:00
notes.where(:noteable_id => commit.id, :noteable_type => "Commit").where("line_code is not null")
2011-10-09 00:36:38 +03:00
end
2011-12-05 09:23:53 +02:00
def has_commits?
!!commit
end
2011-12-07 09:48:44 +02:00
# Compatible with all access rights
# Should be rewrited for new access rights
2011-10-09 00:36:38 +03:00
def add_access(user, *access)
2011-12-07 09:48:44 +02:00
access = if access.include?(:admin)
2012-02-16 09:03:55 +02:00
{ :project_access => UsersProject::MASTER }
2011-12-07 09:48:44 +02:00
elsif access.include?(:write)
2012-02-16 09:03:55 +02:00
{ :project_access => UsersProject::DEVELOPER }
2011-12-07 09:48:44 +02:00
else
2012-02-16 09:03:55 +02:00
{ :project_access => UsersProject::GUEST }
2011-12-07 09:48:44 +02:00
end
2011-10-09 00:36:38 +03:00
opts = { :user => user }
2011-12-07 09:48:44 +02:00
opts.merge!(access)
2011-10-09 00:36:38 +03:00
users_projects.create(opts)
end
def reset_access(user)
users_projects.where(:project_id => self.id, :user_id => user.id).destroy if self.id
end
def repository_readers
keys = Key.joins({:user => :users_projects}).
2012-02-16 09:03:55 +02:00
where("users_projects.project_id = ? AND users_projects.project_access = ?", id, UsersProject::REPORTER)
keys.map(&:identifier) + deploy_keys.map(&:identifier)
2011-10-09 00:36:38 +03:00
end
2011-12-05 09:43:53 +02:00
def repository_writers
keys = Key.joins({:user => :users_projects}).
2012-02-16 09:03:55 +02:00
where("users_projects.project_id = ? AND users_projects.project_access = ?", id, UsersProject::DEVELOPER)
2011-10-09 00:36:38 +03:00
keys.map(&:identifier)
end
def repository_masters
keys = Key.joins({:user => :users_projects}).
2012-02-16 09:03:55 +02:00
where("users_projects.project_id = ? AND users_projects.project_access = ?", id, UsersProject::MASTER)
keys.map(&:identifier)
end
2011-10-09 00:36:38 +03:00
def readers
2012-02-16 09:03:55 +02:00
@readers ||= users_projects.includes(:user).map(&:user)
end
def writers
2012-02-16 09:03:55 +02:00
@writers ||= users_projects.includes(:user).map(&:user)
2011-10-09 00:36:38 +03:00
end
def admins
2012-02-16 09:03:55 +02:00
@admins ||= users_projects.includes(:user).where(:project_access => UsersProject::MASTER).map(&:user)
2011-10-09 00:36:38 +03:00
end
2011-12-15 23:57:46 +02:00
def allow_read_for?(user)
2012-02-16 09:03:55 +02:00
!users_projects.where(:user_id => user.id).empty?
2011-12-15 23:57:46 +02:00
end
def allow_write_for?(user)
2012-02-16 09:03:55 +02:00
!users_projects.where(:user_id => user.id).empty?
2011-12-15 23:57:46 +02:00
end
def allow_admin_for?(user)
2012-02-16 09:03:55 +02:00
!users_projects.where(:user_id => user.id, :project_access => [UsersProject::MASTER]).empty? || owner_id == user.id
2011-12-15 23:57:46 +02:00
end
def allow_pull_for?(user)
2012-02-16 09:03:55 +02:00
!users_projects.where(:user_id => user.id, :project_access => [UsersProject::REPORTER, UsersProject::DEVELOPER, UsersProject::MASTER]).empty?
end
2011-11-16 08:38:53 +03:00
def root_ref
default_branch || "master"
2011-11-16 08:38:53 +03:00
end
2011-10-09 00:36:38 +03:00
def public?
!private_flag
end
def private?
private_flag
end
2011-11-15 12:34:30 +04:00
def last_activity
2011-10-31 22:57:16 +02:00
updates(1).first
2011-11-15 12:34:30 +04:00
rescue
2011-10-31 22:57:16 +02:00
nil
end
def last_activity_date
last_activity.try(:created_at)
end
def last_activity_date_cached(expire = 1.hour)
activity_date_key = "project_#{id}_activity_date"
cached_activities = Rails.cache.read(activity_date_key)
if cached_activities
activity_date = if cached_activities == "Never"
nil
else
cached_activities
end
else
activity_date = last_activity_date
Rails.cache.write(activity_date_key, activity_date || "Never", :expires_in => expire)
end
activity_date
end
2011-11-27 14:53:12 +02:00
# Get project updates from cache
# or calculate.
def cached_updates(limit, expire = 2.minutes)
activities_key = "project_#{id}_activities"
cached_activities = Rails.cache.read(activities_key)
if cached_activities
activities = cached_activities
else
activities = updates(limit)
Rails.cache.write(activities_key, activities, :expires_in => expire)
2011-11-27 14:53:12 +02:00
end
activities
end
# Get 20 events for project like
# commits, issues or notes
2011-10-31 22:57:16 +02:00
def updates(n = 3)
2011-11-15 12:34:30 +04:00
[
2011-10-31 22:57:16 +02:00
fresh_commits(n),
2011-11-15 04:09:07 -05:00
fresh_issues(n),
fresh_notes(n)
2011-10-31 22:57:16 +02:00
].compact.flatten.sort do |x, y|
y.created_at <=> x.created_at
end[0...n]
end
2012-01-14 23:46:06 +02:00
def activities(n=3)
[
fresh_issues(n),
fresh_merge_requests(n),
2012-01-14 23:46:06 +02:00
notes.inc_author_project.where("noteable_type is not null").order("created_at desc").first(n)
].compact.flatten.sort do |x, y|
2011-10-31 22:57:16 +02:00
y.created_at <=> x.created_at
2011-11-02 22:14:03 +02:00
end[0...n]
2011-10-31 22:57:16 +02:00
end
2011-10-09 11:15:01 -07:00
def check_limit
unless owner.can_create_project?
2011-10-09 22:36:57 +03:00
errors[:base] << ("Your own projects limit is #{owner.projects_limit}! Please contact administrator to increase it")
2011-10-09 11:15:01 -07:00
end
rescue
2011-10-09 22:36:57 +03:00
errors[:base] << ("Cant check your ability to create project")
2011-10-09 11:15:01 -07:00
end
2011-10-21 21:40:36 +03:00
def repo_name
2011-12-05 09:43:53 +02:00
if path == "gitolite-admin"
errors.add(:path, " like 'gitolite-admin' is not allowed")
2011-10-21 21:40:36 +03:00
end
end
2011-10-09 00:36:38 +03:00
def valid_repo?
repo
rescue
errors.add(:path, "Invalid repository path")
false
end
end
# == Schema Information
#
# Table name: projects
#
# id :integer not null, primary key
# name :string(255)
# path :string(255)
# description :text
# created_at :datetime
# updated_at :datetime
# private_flag :boolean default(TRUE), not null
# code :string(255)
# owner_id :integer
# default_branch :string(255) default("master"), not null
# issues_enabled :boolean default(TRUE), not null
# wall_enabled :boolean default(TRUE), not null
# merge_requests_enabled :boolean default(TRUE), not null
2011-10-09 00:36:38 +03:00
#