rosa-build/app/models/platform.rb

287 lines
9.2 KiB
Ruby
Raw Normal View History

class Platform < ActiveRecord::Base
2013-09-01 16:33:09 +01:00
extend FriendlyId
friendly_id :name
2013-08-27 17:27:22 +01:00
include Modules::Models::FileStoreClean
include Modules::Models::RegenerationStatus
include Modules::Models::Owner
include EventLoggable
AUTOMATIC_METADATA_REGENERATIONS = %w(day week)
VISIBILITIES = %w(open hidden)
2013-07-08 15:44:07 +01:00
NAME_PATTERN = /[\w\-\.]+/
HUMAN_STATUSES = HUMAN_STATUSES.clone.freeze
2011-10-23 22:39:44 +01:00
2014-01-21 04:51:49 +00:00
belongs_to :parent, class_name: 'Platform', foreign_key: 'parent_platform_id'
belongs_to :owner, polymorphic: true
2011-10-18 16:00:06 +01:00
2014-01-21 04:51:49 +00:00
has_many :repositories, dependent: :destroy
has_many :products, dependent: :destroy
has_many :tokens, as: :subject, dependent: :destroy
has_many :platform_arch_settings, dependent: :destroy
has_many :repository_statuses
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'
2012-05-04 18:12:51 +01:00
has_and_belongs_to_many :advisories
2014-01-21 04:51:49 +00:00
has_many :packages, class_name: "BuildList::Package", dependent: :destroy
2014-01-21 04:51:49 +00:00
has_many :mass_builds, foreign_key: :save_to_platform_id
2014-01-21 04:51:49 +00:00
validates :description, presence: true
validates :visibility, presence: true, inclusion: {in: VISIBILITIES}
validates :automatic_metadata_regeneration, inclusion: {in: AUTOMATIC_METADATA_REGENERATIONS}, allow_blank: true
2014-01-21 04:51:49 +00:00
validates :name, uniqueness: {case_sensitive: false}, presence: true, format: { with: /\A#{NAME_PATTERN}\z/ }
validates :distrib_type, presence: true, inclusion: {in: APP_CONFIG['distr_types']}
validate -> {
if released_was && !released
errors.add(:released, I18n.t('flash.platform.released_status_can_not_be_changed'))
end
}
validate -> {
if personal? && (owner_id_changed? || owner_type_changed?)
errors.add :owner, I18n.t('flash.platform.owner_can_not_be_changed')
end
2014-01-21 04:51:49 +00:00
}, on: :update
2011-03-09 17:38:21 +00:00
before_create :create_directory
2012-12-14 17:36:35 +00:00
before_destroy :detele_directory
after_update :freeze_platform_and_update_repos
after_update :update_owner_relation
after_create -> { symlink_directory unless hidden? }
after_destroy -> { remove_symlink_directory unless hidden? }
scope :search_order, -> { order(:name) }
scope :search, ->(q) { where("#{table_name}.name ILIKE ?", "%#{q.to_s.strip}%") }
scope :by_visibilities, ->(v) { where(visibility: v) }
scope :opened, -> { where(visibility: 'open') }
scope :hidden, -> { where(visibility: 'hidden') }
scope :by_type, ->(type) { where(platform_type: type) if type.present? }
scope :main, -> { by_type('main') }
scope :personal, -> { by_type('personal') }
scope :waiting_for_regeneration, -> { where(status: WAITING_FOR_REGENERATION) }
2011-10-23 22:39:44 +01:00
2014-01-21 04:51:49 +00:00
accepts_nested_attributes_for :platform_arch_settings, allow_destroy: true
attr_accessible :name, :distrib_type, :parent_platform_id, :platform_type, :owner, :visibility, :description, :released, :platform_arch_settings_attributes, :automatic_metadata_regeneration
attr_readonly :name, :distrib_type, :parent_platform_id, :platform_type
2014-01-21 04:51:49 +00:00
state_machine :status, initial: :ready do
after_transition on: :ready, do: :notify_users
event :ready do
2014-01-21 04:51:49 +00:00
transition regenerating: :ready
end
event :regenerate do
transition ready: :waiting_for_regeneration, if: -> { |p| p.main? }
end
event :start_regeneration do
2014-01-21 04:51:49 +00:00
transition waiting_for_regeneration: :regenerating
end
HUMAN_STATUSES.each do |code,name|
2014-01-21 04:51:49 +00:00
state name, value: code
end
end
def clear
system("rm -Rf #{ APP_CONFIG['root_path'] }/platforms/#{ self.name }/repository/*")
end
def urpmi_list(host = nil, pair = nil, add_commands = true, repository_name = 'main')
host ||= default_host
urpmi_commands = ActiveSupport::OrderedHash.new
2012-12-25 10:44:40 +00:00
# TODO: rename method or create separate methods for mdv and rhel
2014-01-21 04:51:49 +00:00
# Platform.main.opened.where(distrib_type: APP_CONFIG['distr_types'].first).each do |pl|
2012-12-25 10:44:40 +00:00
Platform.main.opened.each do |pl|
urpmi_commands[pl.name] = {}
# FIXME should support restricting access to the hidden platform
Arch.all.each do |arch|
tail = "/#{arch.name}/#{repository_name}/release"
command = add_commands ? "urpmi.addmedia #{name} " : ''
2013-05-06 20:13:08 +01:00
command << "#{APP_CONFIG['downloads_url']}/#{name}/repository/#{pl.name}#{tail}"
urpmi_commands[pl.name][arch.name] = command
end
end
return urpmi_commands
end
2011-03-11 09:39:34 +00:00
def path
build_path(name)
2011-03-11 09:39:34 +00:00
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
def symlink_path
Rails.root.join("public", "downloads", name)
end
2013-05-16 18:44:56 +01:00
# Returns URL to repository, for example:
# - http://abf-downloads.rosalinux.ru/rosa-server2012/repository/x86_64/base/
# - http://abf-downloads.rosalinux.ru/uname_personal/repository/rosa-server2012/x86_64/base/
def public_downloads_url(subplatform_name = nil, arch = nil, repo = nil)
"#{APP_CONFIG['downloads_url']}/#{name}/repository/".tap do |url|
url << "#{subplatform_name}/" if subplatform_name.present?
url << "#{arch}/" if arch.present?
url << "#{repo}/" if repo.present?
end
end
def hidden?
visibility == 'hidden'
end
def personal?
platform_type == 'personal'
end
def main?
platform_type == 'main'
end
def base_clone(attrs = {}) # :description, :name, :owner
2012-06-08 18:37:16 +01:00
dup.tap do |c|
attrs.each {|k,v| c.send("#{k}=", v)} # c.attributes = attrs
c.updated_at = nil; c.created_at = nil
c.parent = self; c.released = false
2011-11-01 14:20:53 +00:00
end
2011-11-03 00:32:01 +00:00
end
def clone_relations(from = parent)
2014-01-21 04:51:49 +00:00
self.repositories = from.repositories.map{|r| r.full_clone(platform_id: id)}
2013-02-27 21:55:30 +00:00
self.products = from.products.map(&:full_clone)
end
def full_clone(attrs = {})
base_clone(attrs).tap do |c|
with_skip {c.save} and c.clone_relations(self) and c.fs_clone # later with resque
end
2011-03-11 17:38:28 +00:00
end
def change_visibility
if !hidden?
2014-01-21 04:51:49 +00:00
update_attributes(visibility: 'hidden')
else
2014-01-21 04:51:49 +00:00
update_attributes(visibility: 'open')
end
end
def symlink_directory
# umount_directory_for_rsync # TODO ignore errors
system("ln -s #{path} #{symlink_path}")
Arch.all.each do |arch|
str = "country=Russian Federation,city=Moscow,latitude=52.18,longitude=48.88,bw=1GB,version=2011,arch=#{arch.name},type=distrib,url=#{public_downloads_url}\n"
File.open(File.join(symlink_path, "#{name}.#{arch.name}.list"), 'w') {|f| f.write(str) }
end
end
def remove_symlink_directory
system("rm -Rf #{symlink_path}")
2011-12-01 14:20:24 +00:00
end
def update_owner_relation
if owner_id_was != owner_id
2014-01-21 04:51:49 +00:00
r = relations.where(actor_id: owner_id_was, actor_type: owner_type_was).first
r.update_attributes(actor_id: owner_id, actor_type: owner_type)
end
end
def destroy
with_skip {super} # avoid cascade XML RPC requests
end
2014-01-21 04:51:49 +00:00
later :destroy, queue: :clone_build
def default_host
EventLog.current_controller.request.host_with_port rescue ::Rosa::Application.config.action_mailer.default_url_options[:host]
end
# Checks access rights to platform and caching for 1 day.
def self.allowed?(path, request)
platform_name = path.gsub(/^[\/]+/, '')
.match(/^(#{NAME_PATTERN}\/|#{NAME_PATTERN}$)/)
return true unless platform_name
platform_name = platform_name[0].gsub(/\//, '')
if request.authorization.present?
token, pass = *ActionController::HttpAuthentication::Basic::user_name_and_password(request)
end
2014-01-21 04:51:49 +00:00
Rails.cache.fetch([platform_name, token, :platform_allowed], expires_in: 2.minutes) do
platform = Platform.find_by_name platform_name
next false unless platform
next true unless platform.hidden?
next false unless token
2014-01-21 04:51:49 +00:00
next true if platform.tokens.by_active.where(authentication_token: token).exists?
user = User.find_by_authentication_token token
current_ability = Ability.new(user)
if user && current_ability.can?(:show, platform)
true
else
false
end
end
end
2014-02-11 20:37:14 +00:00
def self.autostart_metadata_regeneration(value)
Platform.main.where(automatic_metadata_regeneration: value).each(&:regenerate)
end
def self.availables_main_platforms(user, ability = nil)
p_ids = Rails.cache.fetch([:availables_main_platforms, user], expires_in: 10.minutes) do
ability ||= Ability.new user
Platform.main.accessible_by(ability, :show).joins(:repositories).
2014-02-17 21:39:49 +00:00
where('repositories.id IS NOT NULL').uniq.pluck(:id)
end
Platform.preload(:repositories).where(id: p_ids).order(:name)
end
2011-03-09 17:38:21 +00:00
protected
def create_directory
system("mkdir -p -m 0777 #{build_path([name, 'repository'])}")
end
def build_path(dir)
2011-10-18 16:00:06 +01:00
File.join(APP_CONFIG['root_path'], 'platforms', dir)
end
def detele_directory
FileUtils.rm_rf path
2011-04-07 10:10:46 +01:00
end
def fs_clone(old_name = parent.name, new_name = name)
FileUtils.cp_r "#{parent.path}/repository", path
2011-05-30 10:04:32 +01:00
end
2014-01-21 04:51:49 +00:00
later :fs_clone, queue: :clone_build
2011-05-30 10:04:32 +01:00
def freeze_platform_and_update_repos
if released_changed? && released == true
2014-01-21 04:51:49 +00:00
repositories.update_all(publish_without_qa: false)
end
2011-04-11 17:55:52 +01:00
end
def notify_users
users = members.includes(:notifier).select{ |u| u.notifier.can_notify? }
users.each{ |u| UserMailer.metadata_regeneration_notification(self, u).deliver }
end
2011-03-09 17:38:21 +00:00
end