class Project < ActiveRecord::Base VISIBILITIES = ['open', 'hidden'] belongs_to :category, :counter_cache => true belongs_to :owner, :polymorphic => true has_many :build_lists, :dependent => :destroy has_many :auto_build_lists, :dependent => :destroy has_many :project_to_repositories, :dependent => :destroy has_many :repositories, :through => :project_to_repositories has_many :relations, :as => :target, :dependent => :destroy has_many :collaborators, :through => :relations, :source => :object, :source_type => 'User' has_many :groups, :through => :relations, :source => :object, :source_type => 'Group' validates :name, :uniqueness => {:scope => [:owner_id, :owner_type]}, :presence => true, :format => { :with => /^[a-zA-Z0-9_\-\+\.]+$/ } validates :owner, :presence => true # validate {errors.add(:base, I18n.t('flash.project.save_warning_ssh_key')) if owner.ssh_key.blank?} #attr_accessible :category_id, :name, :description, :visibility attr_readonly :name scope :recent, order("name ASC") scope :by_name, lambda { |name| where('name like ?', '%' + name + '%') } scope :by_visibilities, lambda {|v| {:conditions => ['visibility in (?)', v.join(',')]}} scope :addable_to_repository, lambda { |repository_id| where("projects.id NOT IN (SELECT project_to_repositories.project_id FROM project_to_repositories WHERE (project_to_repositories.repository_id = #{ repository_id }))") } scope :automateable, where("projects.id NOT IN (SELECT auto_build_lists.project_id FROM auto_build_lists)") after_create :attach_to_personal_repository after_create :create_git_repo after_destroy :destroy_git_repo # after_rollback lambda { destroy_git_repo rescue true if new_record? } has_ancestry include Modules::Models::Owner def auto_build auto_build_lists.each do |auto_build_list| build_lists.create( :pl => auto_build_list.pl, :bpl => auto_build_list.bpl, :arch => auto_build_list.arch, :project_version => collected_project_versions.last, :build_requires => true, :update_type => 'bugfix') unless build_lists.for_creation_date_period(Time.current - 15.seconds, Time.current).present? end end # TODO deprecate and remove project_versions and collected_project_versions ? def project_versions res = tags.select{|tag| tag.name =~ /^v\./} return res if res and res.size > 0 tags end def collected_project_versions project_versions.collect{|tag| tag.name.gsub(/^\w+\./, "")} end def tags self.git_repository.tags #.sort_by{|t| t.name.gsub(/[a-zA-Z.]+/, '').to_i} end def branches self.git_repository.branches end def versions tags.map(&:name) + branches.map{|b| "latest_#{b.name}"} end def members collaborators + groups.map(&:members).flatten end def git_repository @git_repository ||= Git::Repository.new(path) end def git_repo_name File.join owner.uname, name end def public? visibility == 'open' end def fork(new_owner) clone.tap do |c| c.parent_id = id c.owner = new_owner c.updated_at = nil; c.created_at = nil # :id = nil c.save end end def path build_path(git_repo_name) end def xml_rpc_create(repository) result = BuildServer.create_project name, repository.platform.name, repository.name, path if result == BuildServer::SUCCESS return true else raise "Failed to create project #{name} (repo #{repository.name}) inside platform #{repository.platform.name} in path #{path} with code #{result}." end end def xml_rpc_destroy(repository) result = BuildServer.delete_project name, repository.platform.name if result == BuildServer::SUCCESS return true else raise "Failed to delete repository #{name} (repo main) inside platform #{owner.uname}_personal with code #{result}." end end protected def build_path(dir) File.join(APP_CONFIG['root_path'], 'git_projects', "#{dir}.git") end def attach_to_personal_repository repositories << self.owner.personal_repository if !repositories.exists?(:id => self.owner.personal_repository) end def create_git_repo is_root? ? Grit::Repo.init_bare(path) : parent.git_repository.repo.delay.fork_bare(path) end def destroy_git_repo FileUtils.rm_rf path end end