rosa-build/app/models/repository.rb

198 lines
5.3 KiB
Ruby
Raw Normal View History

class Repository < ActiveRecord::Base
2013-09-01 16:33:09 +01:00
extend FriendlyId
2014-03-14 19:36:32 +00:00
friendly_id :name, use: [:finders]
2013-09-01 16:33:09 +01:00
include EventLoggable
2014-09-30 21:05:53 +01:00
include EmptyMetadata
2014-03-11 12:40:44 +00:00
LOCK_FILE_NAMES = { sync: '.sync.lock', repo: '.repo.lock' }
SORT = { 'base' => 1, 'main' => 2, 'contrib' => 3, 'non-free' => 4, 'restricted' => 5 }
belongs_to :platform
2014-01-21 04:51:49 +00:00
has_many :relations, as: :target, dependent: :destroy
has_many :actors, as: :target, class_name: 'Relation', dependent: :destroy
has_many :members, through: :actors, source: :actor, source_type: 'User'
2014-01-21 04:51:49 +00:00
has_many :project_to_repositories, dependent: :destroy, validate: true
has_many :projects, through: :project_to_repositories
has_many :repository_statuses, dependent: :destroy
has_one :key_pair, dependent: :destroy
2014-01-21 04:51:49 +00:00
has_many :build_lists, foreign_key: :save_to_repository_id, dependent: :destroy
validates :description, presence: true,
length: { maximum: 100 }
validates :name, uniqueness: { scope: :platform_id, case_sensitive: false },
presence: true,
format: { with: /\A[a-z0-9_\-]+\z/ },
length: { maximum: 100 }
validates :publish_builds_only_from_branch, length: { maximum: 255 }
scope :recent, -> { order(:name) }
scope :main, -> { where(name: %w(main base)) }
2014-09-30 21:05:53 +01:00
before_destroy :detele_directory
attr_readonly :name, :platform_id
attr_accessor :projects_list, :build_for_platform_id
def regenerate(build_for_platform_id = nil)
build_for_platform = Platform.main.find build_for_platform_id if platform.personal?
status = repository_statuses.find_or_create_by(platform_id: build_for_platform.try(:id) || platform_id)
status.regenerate
end
def resign
if platform.main?
status = repository_statuses.find_or_create_by(platform_id: platform_id)
status.resign
end
end
def base_clone(attrs = {})
2012-06-08 18:37:16 +01:00
dup.tap do |c|
c.platform_id = nil
attrs.each {|k,v| c.send("#{k}=", v)}
2012-06-08 18:37:16 +01:00
c.updated_at = nil; c.created_at = nil
end
end
def clone_relations(from)
with_skip do
from.projects.find_each {|p| self.projects << p}
end
end
2014-04-15 19:41:06 +01:00
later :clone_relations, loner: true, queue: :low
2014-09-30 21:05:53 +01:00
def add_projects(list, user)
list.lines.each do |line|
begin
line.chomp!; line.strip!
owner, name = line.split('/')
next if owner.blank? || name.blank?
2015-03-26 00:26:24 +00:00
project = ProjectPolicy::Scope.new(user, Project).read.where(owner_uname: owner, name: name).first
projects << project if project
rescue RuntimeError, Exception
end
end
end
2014-04-15 19:41:06 +01:00
later :add_projects, queue: :middle
def remove_projects(list)
list.lines.each do |name|
begin
name.chomp!; name.strip!
next if name.blank?
2014-03-18 21:12:39 +00:00
project_to_repositories.where(projects: { name: name }).joins(:project).readonly(false).destroy_all
rescue RuntimeError, Exception
end
end
end
2014-04-15 19:41:06 +01:00
later :remove_projects, queue: :middle
def full_clone(attrs = {})
base_clone(attrs).tap do |c|
with_skip {c.save} and c.clone_relations(self) # later with resque
2011-11-03 00:32:01 +00:00
end
2011-03-11 17:38:28 +00:00
end
# Checks locking of sync
def sync_lock_file_exists?
lock_file_actions :check, :sync
2013-07-26 15:41:39 +01:00
end
# Uses for locking sync
# Calls from UI
def add_sync_lock_file
lock_file_actions :add, :sync
2013-07-26 15:41:39 +01:00
end
# Uses for unlocking sync
# Calls from UI
def remove_sync_lock_file
lock_file_actions :remove, :sync
2013-07-26 15:41:39 +01:00
end
# Uses for locking publishing
# Calls from API
def add_repo_lock_file
lock_file_actions :add, :repo
end
# Uses for unlocking publishing
# Calls from API
def remove_repo_lock_file
lock_file_actions :remove, :repo
end
# Presence of `.repo.lock` file means that mirror is currently synchronising the repository state.
def repo_lock_file_exists?
lock_file_actions :check, :repo
end
2012-09-12 21:47:39 +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
2011-12-02 15:25:59 +00:00
class << self
def build_stub(platform)
rep = Repository.new
rep.platform = platform
rep
end
end
def destroy
with_skip {super} # avoid cascade XML RPC requests
end
2014-04-15 19:41:06 +01:00
later :destroy, queue: :low
2013-07-26 09:44:15 +01:00
def self.custom_sort(repos)
repos.select{ |r| SORT.keys.include?(r.name) }.sort{ |a,b| SORT[a.name] <=> SORT[b.name] } | repos.sort_by(&:name)
end
protected
2011-10-18 16:00:06 +01:00
def lock_file_actions(action, lock_file)
2013-07-26 15:41:39 +01:00
result = false
(['SRPMS'] << Arch.pluck(:name)).each do |arch|
path = "#{platform.path}/repository/#{arch}/#{name}/#{LOCK_FILE_NAMES[lock_file]}"
2013-07-26 15:41:39 +01:00
case action
when :add
result ||= FileUtils.touch(path) rescue nil
when :remove
result ||= FileUtils.rm_f(path)
2013-07-26 15:41:39 +01:00
when :check
return true if File.exist?(path)
end
end
return result
end
def detele_directory
2013-04-25 17:37:07 +01:00
return unless platform
2013-01-17 20:51:12 +00:00
repository_path = platform.path << '/repository'
if platform.personal?
Platform.main.pluck(:name).each do |main_platform_name|
detele_repositories_directory "#{repository_path}/#{main_platform_name}"
end
else
detele_repositories_directory repository_path
2011-04-07 10:27:50 +01:00
end
end
2011-12-01 09:29:04 +00:00
def detele_repositories_directory(repository_path)
srpm_and_arches = (['SRPM'] << Arch.pluck(:name)).join(',')
`bash -c 'rm -rf #{repository_path}/{#{srpm_and_arches}}/#{name}'`
end
end