rosa-build/app/models/platform.rb

207 lines
6.6 KiB
Ruby

#require 'lib/build_server.rb'
class Platform < ActiveRecord::Base
DOWNLOADS_PATH = RAILS_ROOT + '/public/downloads'
VISIBILITIES = ['open', 'hidden']
relationable :as => :target
belongs_to :parent, :class_name => 'Platform', :foreign_key => 'parent_platform_id'
belongs_to :owner, :polymorphic => true
has_many :repositories, :dependent => :destroy
has_many :products, :dependent => :destroy
has_many :objects, :as => :target, :class_name => 'Relation', :dependent => :destroy
has_many :members, :through => :objects, :source => :object, :source_type => 'User'
has_many :groups, :through => :objects, :source => :object, :source_type => 'Group'
validates :name, :presence => true, :uniqueness => true
validates :unixname, :uniqueness => true, :presence => true, :format => { :with => /^[a-zA-Z0-9_]+$/ }, :allow_nil => false, :allow_blank => false
validates :distrib_type, :presence => true, :allow_nil => :false, :allow_blank => false, :inclusion => {:in => APP_CONFIG['distr_types']}
after_create :make_owner_rel
before_save :check_owner_rel
# before_save :create_directory
# after_destroy :remove_directory
before_create :xml_rpc_create
before_destroy :xml_rpc_destroy
# before_update :check_freezing
after_create lambda { add_downloads_symlink unless self.hidden? }
scope :by_visibilities, lambda {|v| {:conditions => ['visibility in (?)', v.join(',')]}}
scope :main, where(:platform_type => 'main')
scope :personal, where(:platform_type => 'personal')
#attr_accessible :visibility
def urpmi_list(host, pair = nil)
pair = {:login => 'login', :pass => 'password'} if pair.blank?
urpmi_commands = []
if self.hidden?
Platform.main.each do |pl|
urpmi_commands << "urpmi.addmedia http://#{ pair[:login] }@#{ pair[:pass] }:#{ host }/private/#{ self.name }/repository/#{ pl.name }/$ARCH/main/release"
end
else
Platform.main.each do |pl|
urpmi_commands << "urpmi.addmedia http://#{ host }/downloads/#{ self.name }/repository/#{ pl.name }/$ARCH/main/release"
end
end
return urpmi_commands
end
def path
build_path(unixname)
end
def hidden?
self.visibility == 'hidden'
end
def personal?
platform_type == 'personal'
end
def clone(new_name, new_unixname, cowner)
# TODO * make it Delayed Job *
p = Platform.new
p.name = new_name
p.unixname = new_unixname
p.parent = self
p.owner = cowner
p.repositories = repositories.map{|r| r.clone(cowner)}
result = p.save
p.products = products.map do |pr|
pr_cloned = Product.new
pr_cloned.attributes = pr.attributes
pr_cloned.id = nil
pr_cloned.platform = p
pr_cloned.save
pr_cloned
end
return (result && xml_rpc_clone(new_unixname) && p)
end
def name
released? ? "#{self[:name]} #{I18n.t("layout.platforms.released_suffix")}" : self[:name]
end
def roles_of(user)
objects.where(:object_id => user.id, :object_type => user.class).map {|rel| rel.role}.reject {|r| r.nil?}
end
def add_role(user, role)
roles = objects.where(:object_id => user.id, :object_type => user.class).map {|rel| rel.role}.reject {|r| r.nil?}
unless roles.include? role
rel = Relation.create(:object_type => user.class.to_s, :object_id => user.id,
:target_type => self.class.to_s, :target_id => id)
rel.role = role
rel.save
end
end
def change_visibility
if !self.hidden?
self.update_attribute(:visibility, 'hidden')
remove_downloads_symlink
else
self.update_attribute(:visibility, 'open')
add_downloads_symlink
end
# Because observer is not invoked...
ActiveSupport::Notifications.instrument "event_log.observer", :object => self,
:message => I18n.t("activerecord.attributes.platform.visibility_types.#{visibility}")
end
def add_downloads_symlink
#raise "Personal platform path #{ symlink_downloads_path } already exists!" if File.exists?(symlink_downloads_path) && File.directory?(symlink_downloads_path)
return true if File.exists?(symlink_downloads_path) && File.directory?(symlink_downloads_path)
FileUtils.symlink path, symlink_downloads_path
end
def remove_downloads_symlink
#raise "Personal platform path #{ symlink_downloads_path } does not exists!" if !(File.exists?(symlink_downloads_path) && File.directory?(symlink_downloads_path))
return true if !(File.exists?(symlink_downloads_path) && File.directory?(symlink_downloads_path))
FileUtils.rm_rf symlink_downloads_path
end
protected
def build_path(dir)
File.join(APP_CONFIG['root_path'], 'platforms', dir)
end
def git_path(dir)
File.join(build_path(dir), 'git')
end
def create_directory
exists = File.exists?(path) && File.directory?(path)
raise "Directory #{path} already exists" if exists
if new_record?
FileUtils.mkdir_p(path)
elsif unixname_changed?
FileUtils.mv(build_path(unixname_was), build_path(unixname))
end
end
def remove_directory
exists = File.exists?(path) && File.directory?(path)
raise "Directory #{path} didn't exists" unless exists
FileUtils.rm_rf(path)
end
def xml_rpc_create
# return true
result = BuildServer.add_platform unixname, APP_CONFIG['root_path'] + '/platforms' , distrib_type
if result == BuildServer::SUCCESS
return true
else
raise "Failed to create platform #{name} with code #{result}. Path: #{build_path(unixname)}"
end
end
def xml_rpc_destroy
# return true
result = BuildServer.delete_platform unixname
if result == BuildServer::SUCCESS
return true
else
raise "Failed to delete platform #{unixname} with code #{result}."
end
end
def xml_rpc_clone(new_unixname)
# return true
result = BuildServer.clone_platform new_unixname, self.unixname, APP_CONFIG['root_path'] + '/platforms'
if result == BuildServer::SUCCESS
return true
else
raise "Failed to clone platform #{name} with code #{result}. Path: #{build_path(unixname)} to platform #{new_unixname}"
end
end
def check_freezing
if released_changed?
BuildServer.freeze_platform self.unixname
end
end
def symlink_downloads_path
"#{ DOWNLOADS_PATH }/#{ self.unixname }"
end
def make_owner_rel
add_owner owner
end
def check_owner_rel
if !new_record? and owner_id_changed?
remove_owner owner_type_was.classify.find(owner_id_was) if owner_type_was
add_owner owner if owner
end
end
end