require 'rack/utils' module ForemanOpenscap class Policy < ApplicationRecord audited include Authorizable include Taxonomix attr_writer :current_step, :wizard_initiated belongs_to :scap_content belongs_to :scap_content_profile belongs_to :tailoring_file belongs_to :tailoring_file_profile, :class_name => 'ForemanOpenscap::ScapContentProfile' has_many :policy_arf_reports has_many :arf_reports, :through => :policy_arf_reports, :dependent => :destroy has_many :asset_policies has_many :assets, :through => :asset_policies, :as => :assetable, :dependent => :destroy scoped_search :on => :name, :complete_value => true scoped_search :relation => :scap_content, :on => :title, :rename => 'content', :complete_value => true scoped_search :relation => :scap_content_profile, :on => :title, :rename => 'profile', :complete_value => true scoped_search :relation => :tailoring_file, :on => :name, :rename => 'tailoring_file', :complete_value => true scoped_search :relation => :tailoring_file_profile, :on => :title, :rename => 'tailoring_file_profile', :complete_value => true before_validation :update_period_attrs def self.deploy_by_variants %w[puppet ansible manual] end validates :name, :presence => true, :uniqueness => true, :length => { :maximum => 255 }, :if => Proc.new { |policy| policy.should_validate?('Policy Attributes') } validates :period, :inclusion => { :in => %w[weekly monthly custom], :message => _('is not a valid value') }, :if => Proc.new { |policy| policy.should_validate?('Schedule') } validates :deploy_by, :inclusion => { :in => Policy.deploy_by_variants }, :if => Proc.new { |policy| policy.should_validate?('Deployment Options') } validates :scap_content_id, presence: true, if: Proc.new { |policy| policy.should_validate?('SCAP Content') } validate :matching_content_profile, if: Proc.new { |policy| policy.should_validate?('SCAP Content') } validate :valid_cron_line, :valid_weekday, :valid_day_of_month, :valid_tailoring, :valid_tailoring_profile, :no_mixed_deployments after_save :assign_policy_to_hostgroups # before_destroy - ensure that the policy has no hostgroups, or classes default_scope do with_taxonomy_scope do order("foreman_openscap_policies.name") end end def to_html if scap_content.nil? return html_error_message(_('Cannot generate HTML guide, scap content is missing.')) end if (proxy = scap_content.proxy_url) api = ProxyAPI::Openscap.new(:url => proxy) else return html_error_message(_('Cannot generate HTML guide, no valid OpenSCAP proxy server found.')) end api.policy_html_guide(scap_content.scap_file, scap_content_profile.try(:profile_id)) end def hostgroup_ids assets.where(:assetable_type => 'Hostgroup').pluck(:assetable_id) end def hostgroup_ids=(ids) assign_ids ids, 'Hostgroup' end def hostgroups Hostgroup.find(hostgroup_ids) end def hostgroups=(hostgroups) hostgroup_ids = hostgroups.map(&:id).map(&:to_s) end def host_ids assets.where(:assetable_type => 'Host::Base').pluck(:assetable_id) end def host_ids=(ids) assign_ids ids, 'Host::Base' end def hosts Host.where(:id => host_ids) end def hosts=(hosts) host_ids = hosts.map(&:id).map(&:to_s) end def step_to_i(step_name) steps.index(step_name) + 1 end def steps base_steps = [N_('Deployment Options'), N_('Policy Attributes'), N_('SCAP Content'), N_('Schedule')] base_steps << N_('Locations') if SETTINGS[:locations_enabled] base_steps << N_('Organizations') if SETTINGS[:organizations_enabled] base_steps << N_('Hostgroups') # always be last. end def current_step @current_step || steps.first end def previous_step return steps.last if current_step.empty? steps[steps.index(current_step) - 1] end def next_step steps[steps.index(current_step) + 1] end def rewind_step @current_step = previous_step end def first_step? current_step == steps.first end def current_step?(step_name) current_step == step_name end def last_step? current_step == steps.last end def wizard_completed? new_record? && current_step.blank? end def step_index wizard_completed? ? steps.index(steps.last) : steps.index(current_step) + 1 end def scan_name name end def used_location_ids Location.joins(:taxable_taxonomies).where( 'taxable_taxonomies.taxable_type' => 'ForemanOpenscap::Policy', 'taxable_taxonomies.taxable_id' => id ).pluck("#{Location.arel_table.name}.id") end def used_organization_ids Organization.joins(:taxable_taxonomies).where( 'taxable_taxonomies.taxable_type' => 'ForemanOpenscap::Policy', 'taxable_taxonomies.taxable_id' => id ).pluck("#{Location.arel_table.name}.id") end def used_hostgroup_ids [] end def unassign_hosts(hosts) host_asset_ids = ForemanOpenscap::Asset.where(:assetable_type => 'Host::Base', :assetable_id => hosts.map(&:id)).pluck(:id) self.asset_ids = self.asset_ids - host_asset_ids end def to_enc { 'id' => self.id, 'profile_id' => profile_for_scan, 'content_path' => "/var/lib/openscap/content/#{self.scap_content.digest}.xml", 'tailoring_path' => tailoring_file ? "/var/lib/openscap/tailoring/#{self.tailoring_file.digest}.xml" : '', 'download_path' => "/compliance/policies/#{self.id}/content/#{scap_content.digest}", 'tailoring_download_path' => tailoring_file ? "/compliance/policies/#{self.id}/tailoring/#{tailoring_file.digest}" : '' }.merge(period_enc) end def to_enc_legacy to_enc.tap { |hash| hash['download_path'] = "/compliance/policies/#{self.id}/content" } end def should_validate?(step_name) if new_record? && wizard_initiated? step_index > step_to_i(step_name) elsif new_record? && !wizard_initiated? true else persisted? end end def wizard_initiated? @wizard_initiated end def update_period_attrs case period when 'monthly' erase_period_attrs(%w[cron_line weekday]) when 'weekly' erase_period_attrs(%w[cron_line day_of_month]) when 'custom' erase_period_attrs(%w[weekday day_of_month]) end end private def html_error_message(message) error_message = '