module Katello module Host class SubscriptionFacet < Katello::Model audited :associations => [:pools] self.table_name = 'katello_subscription_facets' include Facets::Base belongs_to :user, :inverse_of => :subscription_facets, :class_name => "::User" belongs_to :hypervisor_host, :class_name => "::Host::Managed", :foreign_key => "hypervisor_host_id" has_many :subscription_facet_activation_keys, :class_name => "Katello::SubscriptionFacetActivationKey", :dependent => :destroy, :inverse_of => :subscription_facet has_many :activation_keys, :through => :subscription_facet_activation_keys, :class_name => "Katello::ActivationKey" has_many :subscription_facet_pools, :class_name => "Katello::SubscriptionFacetPool", :dependent => :delete_all, :inverse_of => :subscription_facet has_many :pools, :through => :subscription_facet_pools, :class_name => "Katello::Pool" has_many :subscription_facet_installed_products, :class_name => "Katello::SubscriptionFacetInstalledProduct", :dependent => :destroy, :inverse_of => :subscription_facet has_many :installed_products, :through => :subscription_facet_installed_products, :class_name => "Katello::InstalledProduct" has_many :compliance_reasons, :class_name => "Katello::ComplianceReason", :dependent => :destroy, :inverse_of => :subscription_facet validates :host, :presence => true, :allow_blank => false DEFAULT_TYPE = 'system'.freeze accepts_nested_attributes_for :installed_products attr_accessor :facts def update_from_consumer_attributes(consumer_params) import_database_attributes(consumer_params) self.hypervisor_guest_uuids = consumer_params['guestIds'] unless consumer_params['hypervisor_guest_uuids'].blank? self.facts = consumer_params['facts'] unless consumer_params['facts'].blank? end def import_database_attributes(consumer_params = candlepin_consumer.consumer_attributes) update_subscription_status(consumer_params[:entitlementStatus]) unless consumer_params[:entitlementStatus].blank? update_hypervisor(consumer_params) update_guests(consumer_params) self.autoheal = consumer_params['autoheal'] unless consumer_params['autoheal'].blank? self.service_level = consumer_params['serviceLevel'] unless consumer_params['serviceLevel'].nil? self.registered_at = consumer_params['created'] unless consumer_params['created'].blank? self.last_checkin = consumer_params['lastCheckin'] unless consumer_params['lastCheckin'].blank? self.update_installed_products(consumer_params['installedProducts']) if consumer_params.key?('installedProducts') unless consumer_params['releaseVer'].blank? release = consumer_params['releaseVer'] release = release['releaseVer'] if release.is_a?(Hash) self.release_version = release end end def update_installed_products(consumer_installed_product_list) self.installed_products = consumer_installed_product_list.map do |consumer_installed_product| InstalledProduct.find_or_create_from_consumer(consumer_installed_product) end end def update_compliance_reasons(reasons) reasons = Katello::Candlepin::Consumer.friendly_compliance_reasons(reasons) existing = self.compliance_reasons.pluck(:reason) to_delete = existing - reasons to_create = reasons - existing self.compliance_reasons.where(:reason => to_delete).destroy_all if to_delete.any? to_create.each { |reason| self.compliance_reasons.create(:reason => reason) } end def virtual_guests ::Host.joins(:subscription_facet).where("#{self.class.table_name}.hypervisor_host_id" => self.host_id) end def virtual_guest_uuids virtual_guests.pluck("#{Katello::Host::SubscriptionFacet.table_name}.uuid") end def update_hypervisor(consumer_params = candlepin_consumer.consumer_attributes) if consumer_params.try(:[], 'type').try(:[], 'label') == 'hypervisor' self.hypervisor = true elsif !consumer_params.try(:[], 'guestIds').empty? self.hypervisor = true elsif !candlepin_consumer.virtual_guests.empty? # Check by calling out to Candlepin last for efficiency self.hypervisor = true end end def update_guests(consumer_params) if self.hypervisor if !consumer_params.try(:[], 'guestIds').empty? guest_ids = consumer_params['guestIds'].map do |id| if id.is_a?(Hash) id['guestId'].downcase elsif id.is_a?(String) id.downcase end end guest_ids = FactValue.where("lower(value) IN (?)", guest_ids). where(:fact_name_id => FactName.where(:name => 'virt::uuid')). pluck(:host_id) else guest_ids = self.candlepin_consumer.virtual_guests.pluck(:id) end subscription_facets = SubscriptionFacet.where(:host_id => guest_ids) subscription_facets.update_all(:hypervisor_host_id => self.host.id) elsif (virtual_host = self.candlepin_consumer.virtual_host) self.hypervisor_host = virtual_host end end def consumer_attributes attrs = { :autoheal => autoheal, :serviceLevel => service_level, :releaseVer => release_version, :environment => {:id => self.candlepin_environment_id}, :installedProducts => self.installed_products.map(&:consumer_attributes), :guestIds => virtual_guest_uuids } attrs[:facts] = facts if facts HashWithIndifferentAccess.new(attrs) end def candlepin_environment_id if self.host.content_facet self.host.content_facet.content_view.cp_environment_id(self.host.content_facet.lifecycle_environment) else self.host.organization.default_content_view.cp_environment_id(self.host.organization.library) end end def content_view self.host.content_facet.try(:content_view) || self.organization.default_content_view end def lifecycle_environment self.host.content_facet.try(:lifecycle_environment) || self.organization.library end def organization self.host.organization end def update_subscription_status(status_override = nil) status = host.get_status(::Katello::SubscriptionStatus) if status_override status.status = status.to_status(:status_override => status_override) status.save! else host.get_status(::Katello::SubscriptionStatus).refresh! end host.refresh_global_status! end def self.new_host_from_facts(facts, org, location) name = propose_name_from_facts(facts) ::Host::Managed.new(:name => name, :organization => org, :location => location, :managed => false) end def self.update_facts(host, rhsm_facts) return if host.build? || rhsm_facts.nil? rhsm_facts[:_type] = RhsmFactName::FACT_TYPE rhsm_facts[:_timestamp] = Time.now.to_s host.import_facts(rhsm_facts) end def self.find_or_create_host(organization, rhsm_params) host = find_host(rhsm_params[:facts], organization) host ||= Katello::Host::SubscriptionFacet.new_host_from_facts( rhsm_params[:facts], organization, Location.default_host_subscribe_location! ) host.organization = organization unless host.organization host end def self.propose_name_from_facts(facts) setting_fact = Setting[:register_hostname_fact] if !setting_fact.blank? && facts[setting_fact] && facts[setting_fact] != 'localhost' facts[setting_fact] else Rails.logger.warn(_("register_hostname_fact set for %s, but no fact found, or was localhost.") % setting_fact) unless setting_fact.blank? [facts['network.fqdn'], facts['network.hostname-override'], facts['network.hostname']].find { |name| !name.blank? && name != 'localhost' } end end def self.propose_custom_fact(facts) setting_fact = Setting[:register_hostname_fact] only_use_custom_fact = Setting[:register_hostname_fact_strict_match] if !setting_fact.blank? && !facts[setting_fact].blank? if only_use_custom_fact || ::Host.where(:name => setting_fact.downcase).any? facts[setting_fact] end end end def self.propose_existing_hostname(facts) if propose_custom_fact(facts) name = propose_custom_fact(facts) elsif ::Host.where(:name => facts['network.hostname'].downcase).any? name = facts['network.hostname'] elsif !facts['network.fqdn'].blank? && ::Host.where(:name => facts['network.fqdn'].downcase).any? name = facts['network.fqdn'] elsif !facts['network.hostname-override'].blank? && ::Host.where(:name => facts['network.hostname-override'].downcase).any? name = facts['network.hostname-override'] else name = facts['network.hostname'] #fallback to default, even if it doesn't exist end name.downcase end def products Katello::Product.joins(:subscriptions => {:pools => :subscription_facets}).where("#{Katello::Host::SubscriptionFacet.table_name}.id" => self.id).enabled.uniq end def remove_subscriptions(pools_with_quantities) ForemanTasks.sync_task(Actions::Katello::Host::RemoveSubscriptions, self.host, pools_with_quantities) end def self.find_host(facts, organization) host_name = propose_existing_hostname(facts) hosts = ::Host.unscoped.where(:name => host_name) return nil if hosts.empty? #no host exists if hosts.where("organization_id = #{organization.id} OR organization_id is NULL").empty? #not in the correct org #TODO: http://projects.theforeman.org/issues/11532 fail _("Host with name %{host_name} is currently registered to a different org, please migrate host to %{org_name}.") % {:org_name => organization.name, :host_name => host_name } end hosts.first end def self.sanitize_name(name) name.gsub('_', '-').chomp('.').downcase end def candlepin_consumer @candlepin_consumer ||= Katello::Candlepin::Consumer.new(self.uuid, self.host.organization.label) end def backend_update_needed? %w(release_version service_level autoheal).each do |method| return true if self.send("#{method}_changed?") end if self.host.content_facet return true if (self.host.content_facet.content_view_id_changed? || self.host.content_facet.lifecycle_environment_id_changed?) end false end end end end