rosa-build/app/models/project.rb

401 lines
15 KiB
Ruby
Raw Normal View History

2011-03-09 17:38:21 +00:00
class Project < ActiveRecord::Base
2014-03-13 16:32:36 +00:00
has_ancestry orphan_strategy: :adopt # we replace a 'path' method in the Git module
include Autostart
include Owner
include Git
include Wiki
include UrlHelper
include EventLoggable
include Project::DefaultBranch
2011-10-23 22:39:44 +01:00
VISIBILITIES = ['open', 'hidden']
MAX_OWN_PROJECTS = 32000
2013-07-08 15:44:07 +01:00
NAME_REGEXP = /[\w\-\+\.]+/
2014-03-13 16:35:00 +00:00
OWNER_AND_NAME_REGEXP = /#{User::NAME_REGEXP.source}\/#{NAME_REGEXP.source}/
self.per_page = 25
2011-10-23 22:39:44 +01:00
2014-01-21 04:51:49 +00:00
belongs_to :owner, polymorphic: true, counter_cache: :own_projects_count
belongs_to :maintainer, class_name: 'User'
2014-01-21 04:51:49 +00:00
belongs_to :alias_from, class_name: 'Project'
has_many :aliases, class_name: 'Project', foreign_key: 'alias_from_id'
2014-05-02 21:38:28 +01:00
has_many :issues, dependent: :destroy
has_many :pull_requests, dependent: :destroy, foreign_key: 'to_project_id'
has_many :labels, dependent: :destroy
has_many :build_scripts, dependent: :destroy
2014-01-21 04:51:49 +00:00
has_many :project_imports, dependent: :destroy
has_many :project_to_repositories, dependent: :destroy
has_many :repositories, through: :project_to_repositories
has_many :project_tags, dependent: :destroy
has_many :project_statistics, dependent: :destroy
has_many :build_lists, dependent: :destroy
has_many :hooks, dependent: :destroy
has_many :relations, as: :target, dependent: :destroy
has_many :collaborators, through: :relations, source: :actor, source_type: 'User'
has_many :groups, through: :relations, source: :actor, source_type: 'Group'
has_many :packages, class_name: 'BuildList::Package', dependent: :destroy
2014-01-21 04:51:49 +00:00
has_and_belongs_to_many :advisories # should be without dependent: :destroy
2014-03-11 12:40:44 +00:00
validates :name, uniqueness: { scope: [:owner_id, :owner_type], case_sensitive: false },
2014-01-21 04:51:49 +00:00
presence: true,
2014-03-13 16:35:00 +00:00
format: { with: /\A#{NAME_REGEXP.source}\z/,
message: I18n.t("activerecord.errors.project.uname") },
length: { maximum: 100 }
2014-11-25 16:42:02 +00:00
validates :maintainer, presence: true, unless: :new_record?
2014-03-13 10:23:14 +00:00
validates :url, presence: true, format: { with: /\Ahttps?:\/\/[\S]+\z/ }, if: :mass_import
2014-11-25 16:42:02 +00:00
validates :add_to_repository, presence: true, if: :mass_import
2014-03-11 12:40:44 +00:00
validates :visibility, presence: true, inclusion: { in: VISIBILITIES }
2014-01-21 04:51:49 +00:00
validate { errors.add(:base, :can_have_less_or_equal, count: MAX_OWN_PROJECTS) if owner.projects.size >= MAX_OWN_PROJECTS }
2013-04-09 11:21:17 +01:00
# throws validation error message from ProjectToRepository model into Project model
validate do |project|
project.project_to_repositories.each do |p_to_r|
next if p_to_r.valid?
p_to_r.errors.full_messages.each{ |msg| errors[:base] << msg }
end
errors.delete :project_to_repositories
end
2015-02-18 23:34:07 +00:00
attr_accessible :name, :description, :visibility, :srpm, :is_package,
2013-11-13 22:01:12 +00:00
:has_issues, :has_wiki, :maintainer_id, :publish_i686_into_x86_64,
:url, :srpms_list, :mass_import, :add_to_repository_id, :architecture_dependent,
:autostart_status
attr_readonly :owner_id, :owner_type
scope :recent, -> { order(:name) }
scope :search_order, -> { order('CHAR_LENGTH(projects.name) ASC') }
scope :search, ->(q) {
q = q.to_s.strip
by_name("%#{q}%").search_order if q.present?
}
2014-05-15 20:52:43 +01:00
scope :by_name, ->(name) { where('projects.name ILIKE ?', name) if name.present? }
scope :by_owner, ->(name) { where('projects.owner_uname ILIKE ?', "%#{name}%") if name.present? }
scope :by_owner_and_name, ->(*params) {
2013-04-04 14:33:11 +01:00
term = params.map(&:strip).join('/').downcase
2014-03-12 13:14:19 +00:00
where("lower(concat(owner_uname, '/', name)) ILIKE ?", "%#{term}%") if term.present?
}
scope :by_visibilities, ->(v) { where(visibility: v) }
scope :opened, -> { where(visibility: 'open') }
scope :package, -> { where(is_package: true) }
scope :addable_to_repository, ->(repository_id) {
where('projects.id NOT IN (
SELECT ptr.project_id
FROM project_to_repositories AS ptr
WHERE ptr.repository_id = ?)', repository_id)
2013-04-04 14:33:11 +01:00
}
scope :by_owners, ->(group_owner_ids, user_owner_ids) {
2014-03-25 17:57:40 +00:00
where("(projects.owner_id in (?) AND projects.owner_type = 'Group') OR
(projects.owner_id in (?) AND projects.owner_type = 'User')", group_owner_ids, user_owner_ids)
2012-09-23 17:22:49 +01:00
}
scope :project_aliases, ->(project) {
where.not(id: project.id).
where('alias_from_id IN (:ids) OR id IN (:ids)', { ids: [project.alias_from_id, project.id].compact })
}
2014-01-21 04:51:49 +00:00
before_validation :truncate_name, on: :create
before_save -> { self.owner_uname = owner.uname if owner_uname.blank? || owner_id_changed? || owner_type_changed? }
before_create :set_maintainer
after_save :attach_to_personal_repository
after_update -> { update_path_to_project(name_was) }, if: :name_changed?
attr_accessor :url, :srpms_list, :mass_import, :add_to_repository_id
2013-11-13 22:01:12 +00:00
class << self
2014-03-13 16:35:00 +00:00
def find_by_owner_and_name(first, last = nil)
arr = first.try(:split, '/') || []
arr = (arr << last).compact
return nil if arr.length != 2
where(owner_uname: arr.first, name: arr.last).first || by_owner_and_name(*arr).first
end
2014-03-13 16:35:00 +00:00
def find_by_owner_and_name!(first, last = nil)
find_by_owner_and_name(first, last) or raise ActiveRecord::RecordNotFound
end
end
2013-11-14 21:26:05 +00:00
def init_mass_import
2014-04-15 22:26:12 +01:00
Project.perform_later :low, :run_mass_import, url, srpms_list, visibility, owner, add_to_repository_id
end
2013-04-03 22:06:13 +01:00
def name_with_owner
2013-04-03 22:16:20 +01:00
"#{owner_uname || owner.uname}/#{name}"
2013-04-03 22:06:13 +01:00
end
def to_param
2014-03-13 16:46:21 +00:00
name_with_owner
end
2014-03-19 22:01:42 +00:00
def all_members(*includes)
members(includes) | (owner_type == 'User' ? [owner] : owner.members.includes(includes))
end
2014-03-19 22:01:42 +00:00
def members(*includes)
collaborators.includes(includes) | groups.map{ |g| g.members.includes(includes) }.flatten
end
2012-10-17 14:46:16 +01:00
def add_member(member, role = 'admin')
Relation.add_member(member, self, role)
end
def remove_member(member)
Relation.remove_member(member, self)
end
def platforms
@platforms ||= repositories.map(&:platform).uniq
end
def admins
admins = self.collaborators.where("relations.role = 'admin'")
grs = self.groups.where("relations.role = 'admin'")
if self.owner.is_a? Group
grs = grs.where("relations.actor_id != ?", self.owner.id)
admins = admins | owner.members.where("relations.role = 'admin'")
end
admins = admins | grs.map(&:members).flatten # member of the admin group is admin
end
def public?
visibility == 'open'
end
def owner?(user)
owner == user
end
2012-12-14 11:25:10 +00:00
def git_project_address auth_user
opts = default_url_options
2014-01-21 04:51:49 +00:00
opts.merge!({user: auth_user.authentication_token, password: ''}) unless self.public?
Rails.application.routes.url_helpers.project_url(self.name_with_owner, opts) + '.git'
2012-11-08 09:41:07 +00:00
#path #share by NFS
end
def build_for(mass_build, repository_id, arch = Arch.find_by(name: 'i586'), priority = 0, increase_rt = false)
build_for_platform = mass_build.build_for_platform
save_to_platform = mass_build.save_to_platform
user = mass_build.user
# Select main and project platform repository(contrib, non-free and etc)
# If main does not exist, will connect only project platform repository
# If project platform repository is main, only main will be connect
main_rep_id = build_for_platform.repositories.main.first.try(:id)
2013-06-03 16:44:55 +01:00
include_repos = ([main_rep_id] << (save_to_platform.main? ? repository_id : nil)).compact.uniq
project_version = project_version_for save_to_platform, build_for_platform
2014-01-21 04:51:49 +00:00
increase_release_tag(project_version, user, "MassBuild##{mass_build.id}: Increase release tag") if increase_rt
2012-12-25 16:01:44 +00:00
build_list = build_lists.build do |bl|
bl.save_to_platform = save_to_platform
bl.build_for_platform = build_for_platform
bl.update_type = 'newpackage'
bl.arch = arch
bl.project_version = project_version
bl.user = user
bl.auto_publish_status = mass_build.auto_publish_status
bl.auto_create_container = mass_build.auto_create_container
bl.include_repos = include_repos
bl.extra_repositories = mass_build.extra_repositories
bl.extra_build_lists = mass_build.extra_build_lists
bl.priority = priority
bl.mass_build_id = mass_build.id
bl.save_to_repository_id = repository_id
bl.include_testing_subrepository = mass_build.include_testing_subrepository?
bl.use_cached_chroot = mass_build.use_cached_chroot?
bl.use_extra_tests = mass_build.use_extra_tests?
bl.external_nodes = mass_build.external_nodes
end
2012-12-25 16:01:44 +00:00
build_list.save
end
def fork(new_owner, new_name: nil, is_alias: false)
new_name = new_name.presence || name
dup.tap do |c|
c.name = new_name
c.parent_id = id
c.alias_from_id = is_alias ? (alias_from_id || id) : nil
c.owner = new_owner
c.updated_at = nil; c.created_at = nil # :id = nil
# Hack to call protected method :)
c.send :set_maintainer
c.save
end
2011-03-11 17:38:28 +00:00
end
def get_project_tag_sha1(tag, format)
2013-02-14 13:20:08 +00:00
format_id = ProjectTag::FORMATS["#{tag_file_format(format)}"]
2014-01-21 04:51:49 +00:00
project_tag = project_tags.where(tag_name: tag.name, format_id: format_id).first
2014-06-20 19:40:18 +01:00
return project_tag.sha1 if project_tag && project_tag.commit_id == tag.commit.id && FileStoreService::File.new(sha1: project_tag.sha1).exist?
archive = archive_by_treeish_and_format tag.name, format
2014-06-20 19:40:18 +01:00
sha1 = FileStoreService::File.new(data: archive).save
2014-05-05 16:27:09 +01:00
return nil if sha1.blank?
if project_tag
project_tag.destroy_files_from_file_store(project_tag.sha1)
2014-01-21 04:51:49 +00:00
project_tag.update_attributes(sha1: sha1)
else
project_tags.create(
2014-01-21 04:51:49 +00:00
tag_name: tag.name,
format_id: format_id,
commit_id: tag.commit.id,
sha1: sha1
)
end
return sha1
end
def archive_by_treeish_and_format(treeish, format)
@archive ||= create_archive treeish, format
end
2013-09-19 13:07:49 +01:00
# Finds release tag and increase its:
# 'Release: %mkrel 4mdk' => 'Release: 5mdk'
# 'Release: 4' => 'Release: 5'
# Finds release macros and increase it:
# '%define release %mkrel 4mdk' => '%define release 5mdk'
# '%define release 4' => '%define release 5'
def self.replace_release_tag(content)
build_new_release = Proc.new do |release, combine_release|
if combine_release.present?
r = combine_release.split('.').last.to_i
release << combine_release.gsub(/.[\d]+$/, '') << ".#{r + 1}"
else
release = release.to_i + 1
end
2014-01-21 04:51:49 +00:00
release
2013-09-19 13:07:49 +01:00
end
content.gsub(/^Release:(\s+)(%mkrel\s+)?(\d+)([.\d]+)?(mdk)?$/) do |line|
tab, mkrel, mdk = $1, $2, $5
"Release:#{tab}#{build_new_release.call($3, $4)}#{mdk}"
end.gsub(/^%define\s+release:?(\s+)(%mkrel\s+)?(\d+)([.\d]+)?(mdk)?$/) do |line|
tab, mkrel, mdk = $1, $2, $5
"%define release#{tab}#{build_new_release.call($3, $4)}#{mdk}"
end
end
class << self
Autostart::HUMAN_AUTOSTART_STATUSES.each do |autostart_status, human_autostart_status|
define_method "autostart_build_lists_#{human_autostart_status}" do
autostart_build_lists autostart_status
end
end
end
def self.autostart_build_lists(autostart_status)
Project.where(autostart_status: autostart_status).find_each do |p|
p.project_to_repositories.autostart_enabled.includes(repository: :platform).each do |p_to_r|
repository = p_to_r.repository
user = User.find(p_to_r.user_id)
if repository.platform.personal?
platforms = Platform.availables_main_platforms(user)
else
platforms = [repository.platform]
end
platforms.each do |platform|
platform.platform_arch_settings.by_default.pluck(:arch_id).each do |arch_id|
build_list = p.build_lists.build do |bl|
bl.save_to_platform = repository.platform
bl.build_for_platform = platform
bl.update_type = BuildList::UPDATE_TYPE_NEWPACKAGE
bl.arch_id = arch_id
2014-03-12 20:49:36 +00:00
bl.project_version = p.project_version_for(repository.platform, platform)
bl.user = user
bl.auto_publish_status = p_to_r.auto_publish? ? BuildList::AUTO_PUBLISH_STATUS_DEFAULT : BuildList::AUTO_PUBLISH_STATUS_NONE
bl.save_to_repository = repository
2014-03-12 20:49:36 +00:00
bl.include_repos = [platform.repositories.main.first.try(:id)].compact
if repository.platform.personal?
bl.extra_repositories = [repository.id]
else
bl.include_repos |= [repository.id]
end
end
2014-03-12 20:49:36 +00:00
build_list.save
end
end
end
end
end
def increase_release_tag(project_version, user, message)
blob, raw = find_blob_and_raw_of_spec_file(project_version)
return unless blob
2013-09-19 13:07:49 +01:00
content = self.class.replace_release_tag raw.content
return if content == raw.content
update_file(blob.name, content.gsub("\r", ''),
2014-01-21 04:51:49 +00:00
message: message,
actor: user,
head: project_version
)
end
protected
2013-09-19 13:07:49 +01:00
def create_archive(treeish, format)
file_name = "#{name}-#{treeish}"
fullname = "#{file_name}.#{tag_file_format(format)}"
2014-01-17 17:27:44 +00:00
file = Tempfile.new fullname, File.join(Rails.root, 'tmp')
system("cd #{path}; git archive --format=#{format == 'zip' ? 'zip' : 'tar'} --prefix=#{file_name}/ #{treeish} #{format == 'zip' ? '' : ' | gzip -9'} > #{file.path}")
file.close
{
2014-01-21 04:51:49 +00:00
path: file.path,
fullname: fullname
}
end
def tag_file_format(format)
format == 'zip' ? 'zip' : 'tar.gz'
end
def truncate_name
self.name = name.strip if name
end
def attach_to_personal_repository
owner_repos = self.owner.personal_platform.repositories
if is_package
2014-01-21 04:51:49 +00:00
repositories << self.owner.personal_repository unless repositories.exists?(id: owner_repos.pluck(:id))
else
repositories.delete owner_repos
end
end
def set_maintainer
if maintainer_id.blank?
self.maintainer_id = (owner_type == 'User') ? self.owner_id : self.owner.owner_id
end
end
def update_path_to_project(old_name)
new_name, new_path = name, path
self.name = old_name
old_path = path
self.name = new_name
2014-01-21 04:51:49 +00:00
FileUtils.mv old_path, new_path, force: true if Dir.exists?(old_path)
pull_requests_old_path = File.join(APP_CONFIG['git_path'], 'pull_requests', owner.uname, old_name)
if Dir.exists?(pull_requests_old_path)
FileUtils.mv pull_requests_old_path,
File.join(APP_CONFIG['git_path'], 'pull_requests', owner.uname, new_name),
2014-01-21 04:51:49 +00:00
force: true
end
2014-01-21 04:51:49 +00:00
PullRequest.where(from_project_id: id).update_all(from_project_name: new_name)
2014-01-21 04:51:49 +00:00
PullRequest.where(from_project_id: id).each{ |p| p.update_relations(old_name) }
pull_requests.where('from_project_id != to_project_id').each(&:update_relations)
end
2014-04-15 19:41:06 +01:00
later :update_path_to_project, queue: :middle
2011-03-09 17:38:21 +00:00
end