foreman_rh_cloud 11.1.0 → 11.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (47) hide show
  1. checksums.yaml +4 -4
  2. data/app/assets/javascripts/foreman_rh_cloud/locale/fr/foreman_rh_cloud.js +400 -313
  3. data/app/assets/javascripts/foreman_rh_cloud/locale/ja/foreman_rh_cloud.js +400 -313
  4. data/app/assets/javascripts/foreman_rh_cloud/locale/ka/foreman_rh_cloud.js +393 -306
  5. data/app/assets/javascripts/foreman_rh_cloud/locale/ko/foreman_rh_cloud.js +589 -0
  6. data/app/assets/javascripts/foreman_rh_cloud/locale/zh_CN/foreman_rh_cloud.js +398 -311
  7. data/app/controllers/insights_cloud/api/machine_telemetries_controller.rb +11 -1
  8. data/app/services/foreman_rh_cloud/cloud_request_forwarder.rb +5 -3
  9. data/app/services/foreman_rh_cloud/hits_uploader.rb +3 -8
  10. data/app/services/foreman_rh_cloud/rules_ingester.rb +61 -0
  11. data/app/services/foreman_rh_cloud/template_renderer_helper.rb +1 -1
  12. data/config/routes.rb +10 -4
  13. data/db/seeds.d/189_add_host_inventory_param.rb +7 -0
  14. data/lib/foreman_inventory_upload/async/upload_report_job.rb +1 -1
  15. data/lib/foreman_inventory_upload/generators/queries.rb +1 -1
  16. data/lib/foreman_rh_cloud/engine.rb +1 -1
  17. data/lib/foreman_rh_cloud/version.rb +1 -1
  18. data/lib/insights_cloud/async/insights_full_sync.rb +14 -9
  19. data/lib/insights_cloud/async/insights_resolutions_sync.rb +5 -1
  20. data/lib/insights_cloud.rb +4 -0
  21. data/lib/tasks/insights.rake +10 -7
  22. data/lib/tasks/rh_cloud_inventory.rake +8 -3
  23. data/locale/Makefile +12 -2
  24. data/locale/foreman_rh_cloud.pot +217 -96
  25. data/locale/fr/LC_MESSAGES/foreman_rh_cloud.mo +0 -0
  26. data/locale/fr/foreman_rh_cloud.po +123 -36
  27. data/locale/ja/LC_MESSAGES/foreman_rh_cloud.mo +0 -0
  28. data/locale/ja/foreman_rh_cloud.po +124 -36
  29. data/locale/ka/LC_MESSAGES/foreman_rh_cloud.mo +0 -0
  30. data/locale/ka/foreman_rh_cloud.po +107 -19
  31. data/locale/ko/LC_MESSAGES/foreman_rh_cloud.mo +0 -0
  32. data/locale/ko/foreman_rh_cloud.po +592 -0
  33. data/locale/zh_CN/LC_MESSAGES/foreman_rh_cloud.mo +0 -0
  34. data/locale/zh_CN/foreman_rh_cloud.po +123 -35
  35. data/package.json +1 -1
  36. data/test/controllers/insights_cloud/api/machine_telemetries_controller_test.rb +13 -0
  37. data/test/jobs/insights_full_sync_test.rb +28 -0
  38. data/test/jobs/insights_resolutions_sync_test.rb +22 -0
  39. data/test/unit/services/foreman_rh_cloud/cloud_request_forwarder_test.rb +24 -0
  40. data/test/unit/services/foreman_rh_cloud/hits_uploader_test.rb +13 -2
  41. data/test/unit/services/foreman_rh_cloud/rules_ingester_test.rb +60 -0
  42. data/test/unit/slice_generator_test.rb +24 -2
  43. data/webpack/InsightsCloudSync/Components/RemediationModal/RemediationModal.js +5 -1
  44. metadata +10 -6
  45. data/app/assets/javascripts/foreman_rh_cloud/locale/en/foreman_rh_cloud.js +0 -502
  46. data/locale/en/LC_MESSAGES/foreman_rh_cloud.mo +0 -0
  47. data/locale/en/foreman_rh_cloud.po +0 -496
@@ -16,7 +16,17 @@ module InsightsCloud::Api
16
16
  # The method that "proxies" requests over to Cloud
17
17
  def forward_request
18
18
  certs = candlepin_id_cert @organization
19
- @cloud_response = ::ForemanRhCloud::CloudRequestForwarder.new.forward_request(request, controller_name, @branch_id, certs)
19
+ begin
20
+ @cloud_response = ::ForemanRhCloud::CloudRequestForwarder.new.forward_request(request, controller_name, @branch_id, certs)
21
+ rescue RestClient::Exception => e
22
+ logger.info("Forwarding request failed with exception: #{e}")
23
+ return render json: { error: e }, status: :bad_gateway
24
+ rescue RestClient::Timeout => e
25
+ logger.info("Forwarding request failed with timeout: #{e}")
26
+ return render json: { error: e }, status: :gateway_timeout
27
+ end
28
+
29
+ return render json: { message: @cloud_response.to_s }, status: :gateway_timeout if @cloud_response.is_a?(RestClient::Exceptions::OpenTimeout)
20
30
 
21
31
  if @cloud_response.code == 401
22
32
  return render json: {
@@ -17,8 +17,8 @@ module ForemanRhCloud
17
17
  logger.debug("Sending request to: #{request_opts[:url]}")
18
18
 
19
19
  execute_cloud_request(request_opts)
20
- rescue RestClient::Exception => error_response
21
- error_response.response
20
+ rescue RestClient::ExceptionWithResponse => error_response
21
+ error_response.response.presence || error_response.exception
22
22
  end
23
23
 
24
24
  def prepare_request_opts(original_request, forward_payload, forward_params, certs)
@@ -52,7 +52,9 @@ module ForemanRhCloud
52
52
 
53
53
  def prepare_forward_params(original_request, branch_id)
54
54
  forward_params = original_request.query_parameters
55
- if original_request.user_agent && !original_request.user_agent.include?('redhat_access_cfme')
55
+ compliance_request = original_request.path.match?(/compliance\/v2(\/.*)?/)
56
+ user_agent = original_request.user_agent.present? && !original_request.user_agent.include?('redhat_access_cfme')
57
+ if user_agent && !compliance_request
56
58
  forward_params = forward_params.merge(:branch_id => branch_id)
57
59
  end
58
60
 
@@ -35,14 +35,9 @@ module ForemanRhCloud
35
35
 
36
36
  def update_rules_and_resolutions
37
37
  return if @payload[:rules].blank?
38
- # rubocop:disable Rails/SkipsModelValidations
39
- ::InsightsRule.upsert_all(@payload[:rules], unique_by: :rule_id)
40
- rules = @payload[:rules].map { |rule| rule[:rule_id] }
41
-
42
- return if @payload[:resolutions].blank?
43
- ::InsightsResolution.where(rule_id: rules).delete_all
44
- ::InsightsResolution.insert_all(@payload[:resolutions])
45
- # rubocop:enable Rails/SkipsModelValidations
38
+ rule_ids = @payload[:rules].map { |rule| rule[:rule_id] }
39
+ has_missing_rules = InsightsRule.where(rule_id: rule_ids).count != rule_ids.count
40
+ RulesIngester.new.ingest_rules_and_resolutions! if has_missing_rules
46
41
  end
47
42
 
48
43
  def update_details
@@ -0,0 +1,61 @@
1
+ module ForemanRhCloud
2
+ class RulesIngester
3
+ def ingest_rules_and_resolutions!
4
+ rules, resolutions = fetch_rules_and_resolutions
5
+ # rubocop:disable Rails/SkipsModelValidations
6
+ ::InsightsRule.upsert_all(rules, unique_by: :rule_id)
7
+ ::InsightsResolution.delete_all
8
+ ::InsightsResolution.insert_all(resolutions)
9
+ # rubocop:enable Rails/SkipsModelValidations
10
+ end
11
+
12
+ private
13
+
14
+ def fetch_rules_data
15
+ advisor_url = "#{ForemanRhCloud.on_premise_url}/r/insights/v1/static/release/content.json"
16
+ JSON.parse(Net::HTTP.get(URI.parse(advisor_url)), symbolize_names: true)
17
+ end
18
+
19
+ def fetch_rules_and_resolutions
20
+ rules = fetch_rules_data.values.map do |rule|
21
+ next unless rule[:active]
22
+ next if rule[:playbooks].blank?
23
+ rule.slice(:description, :category,
24
+ :impact_name, :summary, :generic, :reason,
25
+ :rec_likelihood, :rec_impact, :reboot_required,
26
+ :more_info, :rule_id, :playbooks)
27
+ end
28
+ rules.compact!
29
+
30
+ resolutions = rules.flat_map do |rule|
31
+ rule[:playbooks].map do |fix_type, playbook|
32
+ {
33
+ rule_id: rule[:rule_id],
34
+ description: playbook[:name],
35
+ needs_reboot: playbook[:reboot_required] == true,
36
+ resolution_risk: rule[:resolution_risk],
37
+ resolution_type: fix_type,
38
+ }
39
+ end
40
+ end
41
+
42
+ rules.map! do |rule|
43
+ {
44
+ rule_id: rule[:rule_id],
45
+ description: rule[:description],
46
+ category_name: rule[:category],
47
+ impact_name: rule[:impact_name],
48
+ summary: rule[:summary],
49
+ generic: rule[:generic],
50
+ reason: rule[:reason],
51
+ total_risk: ((rule[:rec_likelihood] + rule[:rec_impact]) / 2).to_i,
52
+ reboot_required: rule[:reboot_required] == true,
53
+ more_info: rule[:more_info],
54
+ rating: 0,
55
+ }
56
+ end
57
+
58
+ [rules, resolutions]
59
+ end
60
+ end
61
+ end
@@ -25,7 +25,7 @@ module ForemanRhCloud
25
25
  returns String, desc: 'Playbook downloaded from the cloud'
26
26
  end
27
27
  def download_rh_playbook(playbook_url, organization_id)
28
- retriever = ForemanRhCloud::UrlRemediationsRetriever.new(url: playbook_url, organization_id: organization_id, logger: template_logger)
28
+ retriever = ForemanRhCloud::URLRemediationsRetriever.new(url: playbook_url, organization_id: organization_id, logger: template_logger)
29
29
 
30
30
  cached("rh_playbook_#{playbook_url}") do
31
31
  retriever.create_playbook
data/config/routes.rb CHANGED
@@ -9,7 +9,9 @@ Rails.application.routes.draw do
9
9
  get 'settings', to: 'uploads_settings#index'
10
10
  post 'setting', to: 'uploads_settings#set_advanced_setting'
11
11
 
12
- post 'cloud_connector', to: 'uploads#enable_cloud_connector'
12
+ unless ForemanRhCloud.with_local_advisor_engine?
13
+ post 'cloud_connector', to: 'uploads#enable_cloud_connector'
14
+ end
13
15
 
14
16
  resources :tasks, only: [:create, :show]
15
17
 
@@ -31,7 +33,9 @@ Rails.application.routes.draw do
31
33
  end
32
34
 
33
35
  namespace :foreman_rh_cloud do
34
- get 'inventory_upload', to: '/react#index'
36
+ unless ForemanRhCloud.with_local_advisor_engine?
37
+ get 'inventory_upload', to: '/react#index'
38
+ end
35
39
  get 'insights_cloud', to: '/react#index' # Uses foreman's react controller
36
40
  end
37
41
 
@@ -61,8 +65,10 @@ Rails.application.routes.draw do
61
65
  end
62
66
 
63
67
  namespace 'rh_cloud' do
64
- post 'enable_connector', to: 'inventory#enable_cloud_connector'
65
- post 'cloud_request', to: 'cloud_request#update'
68
+ unless ForemanRhCloud.with_local_advisor_engine?
69
+ post 'enable_connector', to: 'inventory#enable_cloud_connector'
70
+ post 'cloud_request', to: 'cloud_request#update'
71
+ end
66
72
  get 'advisor_engine_config', to: 'advisor_engine_config#show'
67
73
  end
68
74
 
@@ -0,0 +1,7 @@
1
+ CommonParameter.without_auditing do
2
+ params = [
3
+ { name: "host_registration_insights_inventory", key_type: "boolean", value: true },
4
+ ]
5
+
6
+ params.each { |param| CommonParameter.find_or_create_by(param) }
7
+ end
@@ -78,7 +78,7 @@ module ForemanInventoryUpload
78
78
  end
79
79
 
80
80
  def content_disconnected?
81
- Setting[:content_disconnected]
81
+ !Setting[:subscription_connection_enabled]
82
82
  end
83
83
  end
84
84
  end
@@ -44,7 +44,7 @@ module ForemanInventoryUpload
44
44
 
45
45
  def self.for_slice(base)
46
46
  base
47
- .search_for("not params.#{InsightsCloud.enable_client_param} = f")
47
+ .search_for("not params.#{InsightsCloud.enable_client_param_inventory} = f")
48
48
  .joins(:subscription_facet)
49
49
  .preload(
50
50
  :interfaces,
@@ -40,7 +40,7 @@ module ForemanRhCloud
40
40
  apipie_documented_controllers ["#{ForemanRhCloud::Engine.root}/app/controllers/api/v2/**/*.rb"]
41
41
 
42
42
  settings do
43
- category(:rh_cloud, N_('RHCloud')) do
43
+ category(:rh_cloud, N_('Insights')) do
44
44
  setting('allow_auto_inventory_upload', type: :boolean, description: N_('Enable automatic upload of your host inventory to the Red Hat cloud'), default: true, full_name: N_('Automatic inventory upload'))
45
45
  setting('allow_auto_insights_sync', type: :boolean, description: N_('Enable automatic synchronization of Insights recommendations from the Red Hat cloud'), default: true, full_name: N_('Synchronize recommendations Automatically'))
46
46
  setting('allow_auto_insights_mismatch_delete', type: :boolean, description: N_('Enable automatic deletion of mismatched host records from the Red Hat cloud'), default: false, full_name: N_('Automatic mismatch deletion'))
@@ -1,3 +1,3 @@
1
1
  module ForemanRhCloud
2
- VERSION = '11.1.0'.freeze
2
+ VERSION = '11.3.0'.freeze
3
3
  end
@@ -9,10 +9,11 @@ module InsightsCloud
9
9
 
10
10
  def plan(organizations)
11
11
  organizations = organizations.select do |organization|
12
- if cert_auth_available?(organization)
12
+ checker = ::Katello::UpstreamConnectionChecker.new(organization)
13
+ if cert_auth_available?(organization) && !organization.manifest_expired? && checker.can_connect?
13
14
  true
14
15
  else
15
- logger.debug("Certificate is not available for org: #{organization.name}, skipping insights sync")
16
+ logger.info("A manifest is not available for org: #{organization.name}, or it has expired, or been deleted. skipping insights sync")
16
17
  false
17
18
  end
18
19
  end
@@ -20,19 +21,23 @@ module InsightsCloud
20
21
  sequence do
21
22
  # This can be turned off when we enable automatic status syncs
22
23
  # This step will query cloud inventory to retrieve inventory uuids for each host
23
- plan_hosts_sync(organizations)
24
- plan_self(organization_ids: organizations.map(&:id))
25
- concurrence do
26
- plan_rules_sync(organizations)
27
- plan_notifications
24
+ valid_organizations = organizations.reject(&:manifest_expired?)
25
+ if valid_organizations.any?
26
+ plan_hosts_sync(organizations)
27
+ plan_self(organization_ids: organizations.map(&:id))
28
+ concurrence do
29
+ plan_rules_sync(organizations)
30
+ plan_notifications
31
+ end
28
32
  end
29
33
  end
30
34
  end
31
35
 
32
36
  def try_execute
33
37
  organizations.each do |organization|
34
- unless cert_auth_available?(organization)
35
- logger.debug("Certificate is not available for org: #{organization.name}, skipping insights sync")
38
+ checker = ::Katello::UpstreamConnectionChecker.new(organization)
39
+ if !cert_auth_available?(organization) && organization.manifest_expired && !checker.can_connect?
40
+ logger.info("A manifest is not available for org: #{organization.name}, or it has expired, or been deleted. skipping insights sync")
36
41
  next
37
42
  end
38
43
 
@@ -13,7 +13,11 @@ module InsightsCloud
13
13
  InsightsResolution.delete_all
14
14
  rule_ids = relevant_rules
15
15
  Organization.all.each do |organization|
16
- next if !cert_auth_available?(organization) || organization.manifest_expired?
16
+ checker = ::Katello::UpstreamConnectionChecker.new(organization)
17
+ if !cert_auth_available?(organization) || organization.manifest_expired? || !checker.can_connect?
18
+ logger.info("A manifest is not available for org: #{organization.name}, or it has expired, or been deleted. skipping resolutions sync")
19
+ next
20
+ end
17
21
  api_response = query_insights_resolutions(rule_ids, organization) unless rule_ids.empty?
18
22
  written_rules = write_resolutions(api_response) if api_response
19
23
  rule_ids -= Array(written_rules)
@@ -29,6 +29,10 @@ module InsightsCloud
29
29
  'host_registration_insights'
30
30
  end
31
31
 
32
+ def self.enable_client_param_inventory
33
+ 'host_registration_insights_inventory'
34
+ end
35
+
32
36
  def self.enable_cloud_remediations_param
33
37
  'enable_cloud_remediations'
34
38
  end
@@ -24,13 +24,16 @@ namespace :rh_cloud_insights do
24
24
  desc "Re-announce all organizations into Sources on RH cloud."
25
25
  task announce_to_sources: [:environment] do
26
26
  logger = Logging::Logger.new(STDOUT)
27
- Organization.unscoped.each do |org|
28
- presence = ForemanRhCloud::CloudPresence.new(org, logger)
29
- presence.announce_to_sources
30
- rescue StandardError => ex
31
- logger.warn(ex)
27
+ if ForemanRhCloud.with_local_advisor_engine?
28
+ logger.warn('Task announce_to_sources is not available when using local advisor engine')
29
+ else
30
+ Organization.unscoped.each do |org|
31
+ presence = ForemanRhCloud::CloudPresence.new(org, logger)
32
+ presence.announce_to_sources
33
+ rescue StandardError => ex
34
+ logger.warn(ex)
35
+ end
36
+ logger.info('Reannounced all organizations')
32
37
  end
33
-
34
- logger.info('Reannounced all organizations')
35
38
  end
36
39
  end
@@ -9,10 +9,15 @@ namespace :rh_cloud_inventory do
9
9
  else
10
10
  organizations = [Organization.where(:id => ENV['organization_id']).first]
11
11
  end
12
-
12
+ disconnected = ForemanRhCloud.with_local_advisor_engine?
13
13
  User.as_anonymous_admin do
14
14
  organizations.each do |organization|
15
- ForemanTasks.async_task(ForemanInventoryUpload::Async::GenerateReportJob, ForemanInventoryUpload.generated_reports_folder, organization.id, false)
15
+ ForemanTasks.async_task(
16
+ ForemanInventoryUpload::Async::GenerateReportJob,
17
+ ForemanInventoryUpload.generated_reports_folder,
18
+ organization.id,
19
+ disconnected
20
+ )
16
21
  puts "Generated and uploaded inventory report for organization '#{organization.name}'"
17
22
  end
18
23
  end
@@ -47,7 +52,7 @@ namespace :rh_cloud_inventory do
47
52
  base_folder = ENV['target'] || ForemanInventoryUpload.generated_reports_folder
48
53
  organization_id = ENV['organization_id']
49
54
  report_file = ForemanInventoryUpload.facts_archive_name(organization_id)
50
- disconnected = false
55
+ disconnected = ForemanRhCloud.with_local_advisor_engine?
51
56
  ForemanTasks.sync_task(ForemanInventoryUpload::Async::QueueForUploadJob, base_folder, report_file, organization_id, disconnected)
52
57
  puts "Uploaded #{report_file}"
53
58
  end
data/locale/Makefile CHANGED
@@ -31,9 +31,16 @@ all-mo: $(MOFILES)
31
31
  cat $@
32
32
  ! grep -q msgid $@
33
33
 
34
- %.edit.po:
34
+ %.edit.po: %.po.time_stamp
35
35
  touch $@
36
36
 
37
+ # gettext will trash the .edit.po file if the time stamp doesn't exist or is older than the po file
38
+ %.po.time_stamp: %.po
39
+ touch --reference $< $@
40
+
41
+ # Prevent make from treating this as an intermediate file to be cleaned up
42
+ .PRECIOUS: %.po.time_stamp
43
+
37
44
  check: $(POXFILES)
38
45
 
39
46
  # Unify duplicate translations
@@ -43,7 +50,10 @@ uniq-po:
43
50
  done
44
51
 
45
52
  tx-pull: $(EDITFILES)
46
- cd .. && tx pull -f --all
53
+ # Initialize new languages
54
+ cd .. && tx pull -f --all --minimum-perc 50
55
+ # Force update all existing languages
56
+ cd .. && tx pull -f --minimum-perc 0
47
57
  for f in $(EDITFILES) ; do \
48
58
  sed -i 's/^\("Project-Id-Version: \).*$$/\1$(DOMAIN) $(VERSION)\\n"/' $$f; \
49
59
  done