Project

General

Profile

Download (20.5 KB) Statistics
| Branch: | Tag: | Revision:
module Host
class Base < ApplicationRecord
prepend Foreman::STI
include Authorizable
include Parameterizable::ByName
include DestroyFlag
include InterfaceCloning
include Hostext::Ownership
include Foreman::TelemetryHelper

self.table_name = :hosts
extend FriendlyId
friendly_id :name

validates_lengths_from_database
belongs_to :model, :name_accessor => 'hardware_model_name'
belongs_to :owner, :polymorphic => true
has_many :fact_values, :dependent => :destroy, :foreign_key => :host_id
has_many :fact_names, :through => :fact_values
has_many :interfaces, -> { order(:identifier) }, :dependent => :destroy, :inverse_of => :host, :class_name => 'Nic::Base',
:foreign_key => :host_id
has_one :primary_interface, -> { where(:primary => true) }, :class_name => 'Nic::Base', :foreign_key => 'host_id'
has_one :provision_interface, -> { where(:provision => true) }, :class_name => 'Nic::Base', :foreign_key => 'host_id'
has_one :domain, :through => :primary_interface
has_one :subnet, :through => :primary_interface
has_one :subnet6, :through => :primary_interface
accepts_nested_attributes_for :interfaces, :allow_destroy => true

belongs_to :location
belongs_to :organization
belongs_to :hostgroup

alias_attribute :hostname, :name

validates :name, :presence => true, :uniqueness => true, :format => {:with => Net::Validations::HOST_REGEXP, :message => _(Net::Validations::HOST_REGEXP_ERR_MSG)}
validate :host_has_required_interfaces
validate :uniq_interfaces_identifiers
validate :build_managed_only

include PxeLoaderSuggestion

default_scope -> { where(taxonomy_conditions) }

def self.taxonomy_conditions
org = Organization.expand(Organization.current) if SETTINGS[:organizations_enabled]
loc = Location.expand(Location.current) if SETTINGS[:locations_enabled]
conditions = {}
conditions[:organization_id] = Array(org).map { |o| o.subtree_ids }.flatten.uniq unless org.nil?
conditions[:location_id] = Array(loc).map { |l| l.subtree_ids }.flatten.uniq unless loc.nil?
conditions
end

scope :no_location, -> { rewhere(:location_id => nil) }
scope :no_organization, -> { rewhere(:organization_id => nil) }

delegate :ssh_authorized_keys, :to => :owner, :allow_nil => true
delegate :notification_recipients_ids, :to => :owner, :allow_nil => true

PRIMARY_INTERFACE_ATTRIBUTES = [:name, :ip, :ip6, :mac,
:subnet, :subnet_id, :subnet_name,
:subnet6, :subnet6_id, :subnet6_name,
:domain, :domain_id, :domain_name,
:lookup_values_attributes].freeze

# primary interface is mandatory because of delegated methods so we build it if it's missing
# similar for provision interface
# we can't set name attribute until we have primary interface so we don't pass it to super
# initializer and we set name when we are sure that we have primary interface
# we can't create primary interface before calling super because args may contain nested
# interface attributes
def initialize(*args)
values_for_primary_interface = {}
build_values_for_primary_interface!(values_for_primary_interface, args)

super(*args)

build_required_interfaces
update_primary_interface_attributes(values_for_primary_interface)
end

def dup
super.tap do |host|
host.interfaces << self.primary_interface.dup if self.primary_interface.present?
end
end

delegate :ip, :ip6, :mac,
:subnet, :subnet_id, :subnet_name,
:subnet6, :subnet6_id, :subnet6_name,
:domain, :domain_id, :domain_name,
:hostname, :fqdn, :shortname,
:to => :primary_interface, :allow_nil => true
delegate :name=, :ip=, :ip6=, :mac=,
:subnet=, :subnet_id=, :subnet_name=,
:subnet6=, :subnet6_id=, :subnet6_name=,
:domain=, :domain_id=, :domain_name=, :to => :primary_interface

attr_writer :updated_virtuals
def updated_virtuals
@updated_virtuals ||= []
end

def self.attributes_protected_by_default
super - [ inheritance_column ]
end

def self.import_host(hostname, certname = nil, deprecated_proxy = nil)
raise(::Foreman::Exception.new("Invalid Hostname, must be a String")) unless hostname.is_a?(String)
Foreman::Deprecation.deprecation_warning("1.19", "proxy parameter is deprecated, please use import_facts to set it") if deprecated_proxy

# downcase everything
hostname.try(:downcase!)
certname.try(:downcase!)

host = Host.find_by_certname(certname) if certname.present?
host ||= Host.find_by_name(hostname)
host ||= self.new(:name => hostname) # if no host was found, build a new one

# if we were given a certname but found the Host by hostname we should update the certname
# this also sets certname for newly created hosts
host.certname = certname if certname.present?

host
end

def create_new_host_when_facts_are_uploaded?
Setting[:create_new_host_when_facts_are_uploaded]
end

# expect a facts hash
def import_facts(facts, source_proxy = nil)
return false if !create_new_host_when_facts_are_uploaded? && new_record?

# we are not importing facts for hosts in build state (e.g. waiting for a re-installation)
raise ::Foreman::Exception.new('Host is pending for Build') if build?
facts = facts.with_indifferent_access

facts[:domain] = facts[:domain].downcase if facts[:domain].present?

type = facts.delete(:_type)
importer = FactImporter.importer_for(type).new(self, facts)
telemetry_observe_histogram(:importer_facts_import_duration, facts.size, type: type)
telemetry_duration_histogram(:importer_facts_import_duration, 1000, type: type) do
importer.import!
end

save(:validate => false)

parse_facts facts, type, source_proxy
end

def parse_facts(facts, type, source_proxy)
time = facts[:_timestamp]
time = time.to_time if time.is_a?(String)
self.last_compile = time if time

unless build?
parser = FactParser.parser_for(type).new(facts)

telemetry_duration_histogram(:importer_facts_import_duration, 1000, type: type) do
populate_fields_from_facts(parser, type, source_proxy)
end
end

set_taxonomies(facts)

# we are saving here with no validations, as we want this process to be as fast
# as possible, assuming we already have all the right settings in Foreman.
# If we don't (e.g. we never install the server via Foreman, we populate the fields from facts
# TODO: if it was installed by Foreman and there is a mismatch,
# we should probably send out an alert.
save(:validate => false)
end

def attributes_to_import_from_facts
[ :model ]
end

def populate_fields_from_facts(parser, type, source_proxy)
# we must create interface if it's missing so we can store domain
build_required_interfaces(:managed => false)
set_non_empty_values(parser, attributes_to_import_from_facts)
set_interfaces(parser) if parser.parse_interfaces?
end

def set_non_empty_values(parser, methods)
methods.each do |attr|
value = parser.send(attr)
self.send("#{attr}=", value) if value.present?
end
end

def set_interfaces(parser)
# if host has no information in primary interface we try to match it and update it
# instead of creating new interface, suggested primary interface mac and identifier
# is saved to primary interface so we match it in updating code below
if !self.managed? && self.primary_interface.mac.blank? && self.primary_interface.identifier.blank?
identifier, values = parser.suggested_primary_interface(self)
self.primary_interface.mac = Net::Validations.normalize_mac(values[:macaddress]) if values.present?
self.primary_interface.update_attribute(:identifier, identifier)
self.primary_interface.save!
end

changed_count = 0
parser.interfaces.each do |name, attributes|
iface = get_interface_scope(name, attributes).try(:first) || interface_class(name).new(:managed => false)
# create or update existing interface
changed_count += 1 if set_interface(attributes, name, iface)
end

ipmi = parser.ipmi_interface
if ipmi.present?
existing = self.interfaces.find_by(:mac => ipmi[:macaddress], :type => Nic::BMC.name)
iface = existing || Nic::BMC.new(:managed => false)
iface.provider ||= 'IPMI'
changed_count += 1 if set_interface(ipmi, 'ipmi', iface)
end
telemetry_increment_counter(:importer_facts_count_interfaces, changed_count, type: parser.class_name_humanized)

self.interfaces.reload
end

def facts_hash
hash = {}
fact_values.includes(:fact_name).collect do |fact|
hash[fact.fact_name.name] = fact.value
end
hash
end
alias_method :facts, :facts_hash

def ==(comparison_object)
super ||
comparison_object.is_a?(Host::Base) &&
id.present? &&
comparison_object.id == id
end

def set_taxonomies(facts)
['location', 'organization'].each do |taxonomy|
next unless SETTINGS["#{taxonomy.pluralize}_enabled".to_sym]
taxonomy_class = taxonomy.classify.constantize
taxonomy_fact = Setting["#{taxonomy}_fact"]

if taxonomy_fact.present? && facts.key?(taxonomy_fact)
taxonomy_from_fact = taxonomy_class.find_by_title(facts[taxonomy_fact].to_s)
else
default_taxonomy = taxonomy_class.find_by_title(Setting["default_#{taxonomy}"])
end

if self.send(taxonomy.to_s).present?
# Change taxonomy to fact taxonomy if set, otherwise leave it as is
self.send("#{taxonomy}=", taxonomy_from_fact) unless taxonomy_from_fact.nil?
else
# No taxonomy was set, set to fact taxonomy or default taxonomy
self.send "#{taxonomy}=", (taxonomy_from_fact || default_taxonomy)
end
end
end

def overwrite?
@overwrite ||= false
end

# We have to coerce the value back to boolean. It is not done for us by the framework.
def overwrite=(value)
@overwrite = value.to_s == "true"
end

def primary_interface
get_interface_by_flag(:primary)
end

def provision_interface
get_interface_by_flag(:provision)
end

def managed_interfaces
self.interfaces.managed.is_managed.all
end

def bond_interfaces
self.interfaces.bonds.is_managed.all
end

def bridge_interfaces
self.interfaces.bridges.is_managed.all
end

def interfaces_with_identifier(identifiers)
self.interfaces.is_managed.where(:identifier => identifiers).all
end

def reload(*args)
drop_primary_interface_cache
drop_provision_interface_cache
super
end

def becomes(*args)
became = super
became.drop_primary_interface_cache
became.drop_provision_interface_cache
became.interfaces = self.interfaces
became
end

def drop_primary_interface_cache
@primary_interface = nil
end

def drop_provision_interface_cache
@provision_interface = nil
end

def matching?
missing_ids.empty?
end

def missing_ids
Array.wrap(tax_location.try(:missing_ids)) + Array.wrap(tax_organization.try(:missing_ids))
end

def import_missing_ids
tax_location.import_missing_ids if location
tax_organization.import_missing_ids if organization
end

# Provide _id aliases for consistency with the _name methods
alias_attribute :hardware_model_id, :model_id

def lookup_value_match
"fqdn=#{fqdn || name}"
end

# we must also clone interfaces objects so we can detect their attribute changes
# method is public because it's used when we run orchestration from interface side
def setup_clone
return if new_record?
@old = super { |clone| clone.interfaces = self.interfaces.map {|i| setup_object_clone(i) } }
end

def skip_orchestration?
false
end

def orchestrated?
self.class.included_modules.include?(Orchestration)
end

private

def build_values_for_primary_interface!(values_for_primary_interface, args)
new_attrs = args.shift
unless new_attrs.nil?
new_attrs = new_attrs.with_indifferent_access
values_for_primary_interface[:name] = NameGenerator.new.next_random_name unless new_attrs.has_key?(:name)
PRIMARY_INTERFACE_ATTRIBUTES.each do |attr|
values_for_primary_interface[attr] = new_attrs.delete(attr) if new_attrs.has_key?(attr)
end

model_name = new_attrs.delete(:model_name)
new_attrs[:hardware_model_name] = model_name if model_name.present?

args.unshift(new_attrs)
end
end

def update_primary_interface_attributes(attrs)
attrs.each do |name, value|
self.send "#{name}=", value
end
end

def tax_location
return nil unless location_id
@tax_location ||= TaxHost.new(location, self)
end

def tax_organization
return nil unless organization_id
@tax_organization ||= TaxHost.new(organization, self)
end

def build_required_interfaces(attrs = {})
if self.primary_interface.nil?
if self.interfaces.empty?
self.interfaces.build(attrs.merge(:primary => true, :type => 'Nic::Managed'))
else
interface = self.interfaces.first
interface.attributes = attrs
interface.primary = true
end
end
self.primary_interface.provision = true if self.provision_interface.nil?
end

def get_interface_scope(name, attributes, base = self.interfaces)
case interface_class(name).to_s
# we search bonds based on identifiers, e.g. ubuntu sets random MAC after each reboot se we can't
# rely on mac
when 'Nic::Bond', 'Nic::Bridge'
base.virtual.where(:identifier => name)
# for other interfaces we distinguish between virtual and physical interfaces
# for virtual devices we don't check only mac address since it's not unique,
# if we want to update the device it must have same identifier
else
begin
macaddress = Net::Validations.normalize_mac(attributes[:macaddress])
rescue Net::Validations::Error
logger.debug "invalid mac during parsing: #{attributes[:macaddress]}"
end

mac_based = base.where(:mac => macaddress)
if attributes[:virtual]
mac_based.virtual.where(:identifier => name)
elsif mac_based.physical.any?
mac_based.physical
elsif !self.managed
# Unmanaged host's interfaces are just used for reporting, so overwrite based on identifier first
base.where(:identifier => name)
end
end
end

def update_bonds(iface, name, attributes)
bond_interfaces.each do |bond|
next unless bond.children_mac_addresses.include?(attributes['macaddress'])
next if bond.attached_devices_identifiers.include? name
update_bond bond, iface, name
end
end

def update_bond(bond, iface, name)
if iface&.identifier
bond.remove_device(iface.identifier)
bond.add_device(name)
logger.debug "Updating bond #{bond.identifier}, id #{bond.id}: removing #{iface.identifier}, adding #{name} to attached interfaces"
save_updated_bond bond
end
end

def save_updated_bond(bond)
bond.save!
rescue StandardError => e
logger.warn "Saving #{bond.identifier} NIC for host #{self.name} failed, skipping because #{e.message}:"
bond.errors.full_messages.each { |e| logger.warn " #{e}" }
end

def set_interface(attributes, name, iface)
# update bond.attached_interfaces when interface is in the list and identifier has changed
update_bonds(iface, name, attributes) if iface.identifier != name && !iface.virtual? && iface.persisted?
attributes = attributes.clone
iface.mac = attributes.delete(:macaddress)
iface.ip = attributes.delete(:ipaddress)
iface.ip6 = attributes.delete(:ipaddress6)
iface.ip6 = nil if (IPAddr.new('fe80::/10').include?(iface.ip6) rescue false)

if Setting[:update_subnets_from_facts]
iface.subnet = Subnet.subnet_for(iface.ip) if iface.ip_changed? && !iface.matches_subnet?(:ip, :subnet)
iface.subnet6 = Subnet.subnet_for(iface.ip6) if iface.ip6_changed? && !iface.matches_subnet?(:ip6, :subnet6)
end

iface.virtual = attributes.delete(:virtual) || false
iface.tag = attributes.delete(:tag) || ''
iface.attached_to = attributes.delete(:attached_to) if attributes[:attached_to].present?
iface.link = attributes.delete(:link) if attributes.has_key?(:link)
iface.identifier = name
iface.host = self
update_virtuals(iface.identifier_was, name) if iface.identifier_changed? && !iface.virtual? && iface.persisted? && iface.identifier_was.present?
iface.attrs = attributes

if iface.new_record? || iface.changed?
logger.debug "Saving #{name} NIC for host #{self.name}"
result = iface.save

unless result
logger.warn "Saving #{name} NIC for host #{self.name} failed, skipping because:"
iface.errors.full_messages.each { |e| logger.warn " #{e}" }
end

result
end
end

def update_virtuals(old, new)
self.updated_virtuals ||= []

self.interfaces.where(:attached_to => old).virtual.each do |virtual_interface|
next if self.updated_virtuals.include?(virtual_interface.id) # may have been already renamed by another physical

virtual_interface.attached_to = new
virtual_interface.identifier = virtual_interface.identifier.sub(old, new)
virtual_interface.save!
self.updated_virtuals.push(virtual_interface.id)
end
end

def interface_class(name)
case name
when FactParser::BONDS
Nic::Bond
when FactParser::BRIDGES
Nic::Bridge
else
Nic::Managed
end
end

# we can't use SQL query for new records, because interfaces may not exist yet
def get_interface_by_flag(flag)
if self.new_record?
self.interfaces.detect(&flag)
else
cache = "@#{flag}_interface"
if (result = instance_variable_get(cache))
result
else
# we can't use SQL, we need to get even unsaved objects
interface = self.interfaces.detect(&flag)

interface.host = self if interface && !interface.destroyed? # inverse_of does not help (STI), but ignore this on deletion
instance_variable_set(cache, interface)
end
end
end

# we require primary interface so have know the name of host
# provision is required only for managed host and defaults to primary
def host_has_required_interfaces
check_primary_interface
check_provision_interface if self.managed?
end

def check_primary_interface
if self.primary_interface.nil?
errors.add :interfaces, _("host must have one primary interface")
end
end

def check_provision_interface
if self.provision_interface.nil?
errors.add :interfaces, _("managed host must have one provision interface")
end
end

# we can't use standard unique validation on interface since we can't properly handle :scope => :host_id
# for new hosts host_id does not exist at that moment, validation would work only for persisted records
def uniq_interfaces_identifiers
success = true
identifiers = []
relevant_interfaces = self.interfaces.select { |i| !i.marked_for_destruction? }
relevant_interfaces.each do |interface|
next if interface.identifier.blank?
if identifiers.include?(interface.identifier)
interface.errors.add :identifier, :taken
success = false
end
identifiers.push(interface.identifier)
end

errors.add(:interfaces, _('some interfaces are invalid')) unless success
success
end

def build_managed_only
if !managed? && build?
errors.add(:build, _('cannot be enabled for an unmanaged host'))
end
end

def password_base64_encrypted?
if root_pass_changed?
root_pass == hostgroup.try(:read_attribute, :root_pass)
else
true
end
end
end
end

require_dependency 'host/managed'
(1-1/3)