foreman_rh_cloud 3.0.18 → 3.0.21.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (100) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +20 -5
  3. data/app/controllers/foreman_inventory_upload/tasks_controller.rb +14 -3
  4. data/app/controllers/foreman_inventory_upload/uploads_settings_controller.rb +8 -0
  5. data/app/controllers/insights_cloud/hits_controller.rb +37 -0
  6. data/app/controllers/insights_cloud/settings_controller.rb +1 -1
  7. data/app/controllers/insights_cloud/tasks_controller.rb +1 -2
  8. data/app/models/insights_resolution.rb +1 -1
  9. data/app/models/inventory_sync/inventory_status.rb +6 -0
  10. data/app/models/setting/rh_cloud.rb +5 -5
  11. data/app/services/foreman_rh_cloud/cloud_connector.rb +1 -1
  12. data/app/services/foreman_rh_cloud/cloud_request_forwarder.rb +1 -0
  13. data/app/services/foreman_rh_cloud/remediations_retriever.rb +78 -0
  14. data/app/services/foreman_rh_cloud/template_renderer_helper.rb +22 -0
  15. data/app/views/job_templates/rh_cloud_remediations.erb +14 -0
  16. data/config/routes.rb +2 -1
  17. data/db/migrate/20210404000001_change_resolutions.foreman_rh_cloud.rb +10 -0
  18. data/db/seeds.d/179_ui_notifications.rb +11 -0
  19. data/db/seeds.d/50_job_templates.rb +14 -0
  20. data/lib/foreman_inventory_upload.rb +9 -0
  21. data/lib/foreman_inventory_upload/async/generate_all_reports_job.rb +8 -2
  22. data/lib/foreman_inventory_upload/generators/queries.rb +3 -2
  23. data/lib/foreman_inventory_upload/generators/tags.rb +8 -6
  24. data/lib/foreman_inventory_upload/scripts/uploader.sh.erb +5 -1
  25. data/lib/foreman_rh_cloud.rb +18 -0
  26. data/lib/foreman_rh_cloud/engine.rb +32 -1
  27. data/lib/foreman_rh_cloud/version.rb +1 -1
  28. data/lib/insights_cloud.rb +12 -0
  29. data/lib/insights_cloud/async/insights_full_sync.rb +39 -24
  30. data/lib/insights_cloud/async/insights_generate_notifications.rb +58 -0
  31. data/lib/insights_cloud/async/insights_resolutions_sync.rb +69 -0
  32. data/lib/insights_cloud/async/insights_rules_sync.rb +13 -17
  33. data/lib/insights_cloud/async/insights_scheduled_sync.rb +1 -1
  34. data/lib/inventory_sync/async/host_result.rb +11 -6
  35. data/lib/inventory_sync/async/inventory_full_sync.rb +24 -41
  36. data/lib/inventory_sync/async/inventory_hosts_sync.rb +34 -0
  37. data/lib/inventory_sync/async/inventory_scheduled_sync.rb +17 -0
  38. data/lib/inventory_sync/async/query_inventory_job.rb +54 -0
  39. data/lib/tasks/insights.rake +4 -12
  40. data/lib/tasks/rh_cloud_inventory.rake +12 -4
  41. data/package.json +1 -1
  42. data/test/controllers/insights_cloud/api/machine_telemetries_controller_test.rb +12 -0
  43. data/test/factories/insights_factories.rb +22 -0
  44. data/test/jobs/insights_full_sync_test.rb +25 -15
  45. data/test/jobs/insights_resolutions_sync_test.rb +74 -0
  46. data/test/jobs/insights_rules_sync_test.rb +5 -3
  47. data/test/jobs/inventory_full_sync_test.rb +182 -12
  48. data/test/unit/rh_cloud_http_proxy_test.rb +4 -4
  49. data/test/unit/services/foreman_rh_cloud/remediations_retriever_test.rb +49 -0
  50. data/test/unit/services/foreman_rh_cloud/template_renderer_helper_test.rb +28 -0
  51. data/test/unit/tags_generator_test.rb +10 -0
  52. data/webpack/ForemanInventoryUpload/Components/InventoryFilter/InventoryFilter.fixtures.js +2 -1
  53. data/webpack/ForemanInventoryUpload/Components/InventoryFilter/InventoryFilter.js +29 -16
  54. data/webpack/ForemanInventoryUpload/Components/InventoryFilter/InventoryFilterReducer.js +1 -16
  55. data/webpack/ForemanInventoryUpload/Components/InventoryFilter/__tests__/InventoryFilter.test.js +7 -1
  56. data/webpack/ForemanInventoryUpload/Components/InventoryFilter/__tests__/InventoryFilterReducer.test.js +1 -17
  57. data/webpack/ForemanInventoryUpload/Components/InventoryFilter/__tests__/__snapshots__/InventoryFilter.test.js.snap +4 -9
  58. data/webpack/ForemanInventoryUpload/Components/InventoryFilter/__tests__/__snapshots__/InventoryFilterReducer.test.js.snap +0 -12
  59. data/webpack/ForemanInventoryUpload/Components/InventoryFilter/__tests__/__snapshots__/integration.test.js.snap +2 -3
  60. data/webpack/ForemanInventoryUpload/Components/InventoryFilter/__tests__/integration.test.js +1 -1
  61. data/webpack/ForemanInventoryUpload/Components/InventoryFilter/index.js +2 -0
  62. data/webpack/ForemanInventoryUpload/Components/InventoryFilter/inventoryFilter.scss +1 -1
  63. data/webpack/ForemanInventoryUpload/Components/InventorySettings/AdvancedSetting/AdvancedSettingsConstants.js +5 -3
  64. data/webpack/ForemanInventoryUpload/Components/PageHeader/components/PageDescription/PageDescription.js +15 -2
  65. data/webpack/ForemanInventoryUpload/Components/PageHeader/components/PageDescription/__tests__/__snapshots__/PageDescription.test.js.snap +13 -2
  66. data/webpack/ForemanInventoryUpload/Components/PageHeader/components/SyncButton/SyncButtonActions.js +81 -46
  67. data/webpack/ForemanInventoryUpload/Components/PageHeader/components/SyncButton/SyncButtonConstants.js +3 -3
  68. data/webpack/ForemanInventoryUpload/Components/PageHeader/components/SyncButton/SyncButtonSelectors.js +6 -12
  69. data/webpack/ForemanInventoryUpload/Components/PageHeader/components/SyncButton/__tests__/SyncButtonFixtures.js +1 -9
  70. data/webpack/ForemanInventoryUpload/Components/PageHeader/components/SyncButton/__tests__/SyncButtonSelectors.test.js +18 -27
  71. data/webpack/ForemanInventoryUpload/Components/PageHeader/components/SyncButton/__tests__/__snapshots__/SyncButtonSelectors.test.js.snap +1 -16
  72. data/webpack/ForemanInventoryUpload/Components/PageHeader/components/SyncButton/__tests__/__snapshots__/integrations.test.js.snap +58 -0
  73. data/webpack/ForemanInventoryUpload/Components/PageHeader/components/SyncButton/__tests__/integrations.test.js +51 -0
  74. data/webpack/ForemanInventoryUpload/Components/PageHeader/components/SyncButton/index.js +2 -5
  75. data/webpack/ForemanInventoryUpload/ForemanInventoryUploadReducers.js +0 -2
  76. data/webpack/InsightsCloudSync/Components/InsightsTable/InsightsTable.js +1 -1
  77. data/webpack/InsightsCloudSync/Components/InsightsTable/InsightsTableSelectors.js +3 -0
  78. data/webpack/InsightsCloudSync/Components/InsightsTable/__tests__/__snapshots__/InsightsTable.test.js.snap +2 -0
  79. data/webpack/InsightsCloudSync/Components/RemediationModal/RemediateButton.js +59 -0
  80. data/webpack/InsightsCloudSync/Components/RemediationModal/RemediationActions.js +12 -0
  81. data/webpack/InsightsCloudSync/Components/RemediationModal/RemediationHelpers.js +43 -0
  82. data/webpack/InsightsCloudSync/Components/RemediationModal/RemediationModal.js +101 -0
  83. data/webpack/InsightsCloudSync/Components/RemediationModal/RemediationModal.scss +9 -0
  84. data/webpack/InsightsCloudSync/Components/RemediationModal/RemediationModalFooter.js +43 -0
  85. data/webpack/InsightsCloudSync/Components/RemediationModal/RemediationTableConstants.js +38 -0
  86. data/webpack/InsightsCloudSync/Components/RemediationModal/Resolutions.js +55 -0
  87. data/webpack/InsightsCloudSync/Components/RemediationModal/index.js +34 -0
  88. data/webpack/InsightsCloudSync/InsightsCloudSync.js +8 -3
  89. data/webpack/InsightsCloudSync/InsightsCloudSync.scss +5 -0
  90. data/webpack/InsightsCloudSync/__snapshots__/InsightsCloudSync.test.js.snap +9 -6
  91. data/webpack/__mocks__/foremanReact/components/Layout/LayoutSelectors.js +1 -0
  92. data/webpack/__mocks__/foremanReact/redux/middlewares/IntervalMiddleware.js +4 -0
  93. data/webpack/{InsightsCloudSync/Components/InsightsTable/components → common/table}/EmptyState.js +0 -0
  94. data/webpack/common/table/helpers.js +7 -0
  95. metadata +37 -16
  96. data/webpack/ForemanInventoryUpload/Components/PageHeader/components/SyncButton/SyncButtonReducer.js +0 -36
  97. data/webpack/ForemanInventoryUpload/Components/PageHeader/components/SyncButton/__tests__/SyncButtonActions.test.js +0 -31
  98. data/webpack/ForemanInventoryUpload/Components/PageHeader/components/SyncButton/__tests__/SyncButtonReducer.test.js +0 -26
  99. data/webpack/ForemanInventoryUpload/Components/PageHeader/components/SyncButton/__tests__/__snapshots__/SyncButtonActions.test.js.snap +0 -98
  100. data/webpack/ForemanInventoryUpload/Components/PageHeader/components/SyncButton/__tests__/__snapshots__/SyncButtonReducer.test.js.snap +0 -18
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 50f4cac8e85d4aa6a6f5311e0276277ee3de729dfe29954d2fb8e907c2996e65
4
- data.tar.gz: 96a40dac10cabdcf362add97a29d841a88a16d96a9cfb2a59c10eab0a64b8b5d
3
+ metadata.gz: d87599d8ef67eeeaa3e17e720dc990b58d4085caf7fb6491dab093c54770c8fd
4
+ data.tar.gz: 04b52a7412dddf8e6829d61555ec9315c7f3856fde32e976eb8c50638082a3bc
5
5
  SHA512:
6
- metadata.gz: 9dee0e2da14903584e433cf39e5f21c0e6e913ee54a6c579d929dee1db8163d60990502d7404285d7df8fed49f2b2910e923d47adc529dbda0c5f9aada23174b
7
- data.tar.gz: 9c468f95609d7d844ae0b0c1b8bbe697c4b43e55529a6dfc5030b74089b05c2d0fae0843e415cb98bec95dacd45c056d25b2130c8df141238059552303a62573
6
+ metadata.gz: 4ca381e6bcbf74d21a23ae825ebb6556be1afbe98c193fb33e58ef7fee6027142a809ed79b8cbcb3c8dce95200b0bd4528d497597a1f0411f703172ce1f9c123
7
+ data.tar.gz: 945faf0852d9ebce259e4da1bcebd079ef5a4fad87f503fc9ee694f4eae7e3ab36a7dc2390d54879a7da57d176aa22d8030a168d2715fe1f54a521b2258fbb37
data/README.md CHANGED
@@ -42,7 +42,7 @@ In UI: Configure -> Insights -> Sync now
42
42
 
43
43
  From command-line:
44
44
 
45
- /usr/sbin/foreman-rake rh_cloud_inventory:sync
45
+ /usr/sbin/foreman-rake rh_cloud_insights:sync
46
46
 
47
47
  #### Synchronize inventory status
48
48
 
@@ -51,16 +51,32 @@ In UI: Configure -> Inventory Upload -> Sync inventory status
51
51
  From command-line:
52
52
 
53
53
  # all organizations
54
- /usr/sbin/foreman-rake rh_cloud_insights:sync
55
-
54
+ /usr/sbin/foreman-rake rh_cloud_inventory:sync
55
+
56
56
  # specific organization with id 1
57
57
  export organization_id=1
58
- /usr/sbin/foreman-rake rh_cloud_insights:sync
58
+ /usr/sbin/foreman-rake rh_cloud_inventory:sync
59
59
 
60
60
  ## TODO
61
61
 
62
62
  *Todo list here*
63
63
 
64
+ ## Design
65
+
66
+ ### Endpoints
67
+
68
+ | purpose | url | ENV setting for the **bold** part
69
+ | ------------------------------| -------| -----------
70
+ | Inventory uploads | **https://cert.cloud.redhat.com/api/ingress/v1/upload** | SATELLITE_INVENTORY_UPLOAD_URL
71
+ | Query inventory hosts list | **https://cloud.redhat.com** /api/inventory/v1/hosts?tags= | SATELLITE_RH_CLOUD_URL
72
+ | Query insights hits | **https://cloud.redhat.com** /api/insights/v1/export/hits/ | SATELLITE_RH_CLOUD_URL
73
+ | Query insights rules | **https://cloud.redhat.com** /api/insights/v1/rule/?impacting=true&rule_status=enabled&has_playbook=true&limit=&offset= | SATELLITE_RH_CLOUD_URL
74
+ | Query insights resolutions | **https://cloud.redhat.com** /api/remediations/v1/resolutions| SATELLITE_RH_CLOUD_URL
75
+ | Forward insights-client `/static` requests | **https://cloud.redhat.com** /api/static | SATELLITE_RH_CLOUD_URL
76
+ | Forward insights-client legacy `/platform` requests | **https://cert.cloud.redhat.com** /api | SATELLITE_CERT_RH_CLOUD_URL
77
+ | Forward insights-client legacy `/redhat_access/r/insights` requests | **https://cert-api.access.redhat.com** /r/insights | SATELLITE_LEGACY_INSIGHTS_URL
78
+
79
+
64
80
  ## Contributing
65
81
 
66
82
  Fork and send a Pull Request. Thanks!
@@ -81,4 +97,3 @@ GNU General Public License for more details.
81
97
 
82
98
  You should have received a copy of the GNU General Public License
83
99
  along with this program. If not, see <http://www.gnu.org/licenses/>.
84
-
@@ -13,12 +13,23 @@ module ForemanInventoryUpload
13
13
  message: N_('Nothing to sync, there are no hosts with subscription for this organization.'),
14
14
  }, status: :method_not_allowed
15
15
  else
16
- host_statuses = InventorySync::Async::InventoryFullSync.perform_now(selected_org)
16
+ task = ForemanTasks.async_task(InventorySync::Async::InventoryFullSync, selected_org)
17
17
  end
18
+ return render json: { message: N_('there was an issue triggering the task') }, status: :internal_server_error unless task
18
19
 
19
20
  render json: {
20
- syncHosts: host_statuses[:sync],
21
- disconnectHosts: host_statuses[:disconnect],
21
+ task: task,
22
+ }, status: :ok
23
+ end
24
+
25
+ def show
26
+ task = ForemanTasks::Task.find_by(id: params[:id])
27
+ return render json: { message: N_('No task was found') }, status: :internal_server_error unless task
28
+
29
+ render json: {
30
+ endedAt: task.ended_at,
31
+ output: task.output,
32
+ result: task.result,
22
33
  }, status: :ok
23
34
  end
24
35
  end
@@ -8,6 +8,7 @@ module ForemanInventoryUpload
8
8
  excludePackagesEnabled: Setting[:exclude_installed_packages],
9
9
  CloudConnectorStatus: ForemanInventoryUpload::UploadsSettingsController.cloud_connector_status,
10
10
  cloudToken: !Setting[:rh_cloud_token].empty?,
11
+ lastSyncTask: last_successful_inventory_sync_task,
11
12
  }, status: :ok
12
13
  end
13
14
 
@@ -22,5 +23,12 @@ module ForemanInventoryUpload
22
23
  return nil unless job
23
24
  { id: job.id, task: ForemanTasks::Task.where(:id => job.task_id).first }
24
25
  end
26
+
27
+ def last_successful_inventory_sync_task
28
+ task = ForemanTasks::Task.where(label: 'InventorySync::Async::InventoryFullSync', result: 'success')
29
+ .reorder('ended_at desc').first
30
+ return nil unless task
31
+ { output: task.output, endedAt: task.ended_at }
32
+ end
25
33
  end
26
34
  end
@@ -9,6 +9,7 @@ module InsightsCloud
9
9
  hasToken: !Setting[:rh_cloud_token].empty?,
10
10
  hits: hits.map { |hit| hit.attributes.merge(hostname: hit.host&.name, has_playbook: hit.has_playbook?) },
11
11
  itemCount: hits.count,
12
+ isExperimentalMode: Setting[:lab_features],
12
13
  }, status: :ok
13
14
  end
14
15
 
@@ -20,6 +21,21 @@ module InsightsCloud
20
21
  }, status: :ok
21
22
  end
22
23
 
24
+ def resolutions
25
+ if remediation_all_selected_param
26
+ hits = InsightsHit.with_playbook.search_for(params[:query])
27
+ else
28
+ hits = resource_base_search_and_page.where(id: remediation_ids_param)
29
+ end
30
+
31
+ hits.preload(:host, rule: :resolutions)
32
+
33
+ render json: {
34
+ hits: hits.map { |hit| hit.attributes.merge(hostname: hit.host&.name, resolutions: hit.rule.resolutions.map(&:attributes), reboot: hit.rule.reboot_required) },
35
+ itemCount: hits.count,
36
+ }, status: :ok
37
+ end
38
+
23
39
  def model_of_controller
24
40
  ::InsightsHit
25
41
  end
@@ -32,10 +48,31 @@ module InsightsCloud
32
48
  :insights_hits
33
49
  end
34
50
 
51
+ def action_permission
52
+ case params[:action]
53
+ when 'resolutions'
54
+ 'view'
55
+ else
56
+ super
57
+ end
58
+ end
59
+
35
60
  private
36
61
 
37
62
  def host_id_param
38
63
  params.require(:host_id)
39
64
  end
65
+
66
+ def remediation_request_params
67
+ params.permit(remediations: [:hit_id, :remediation_id]).require(:remediations)
68
+ end
69
+
70
+ def remediation_ids_param
71
+ params.require(:ids).map(&:to_i)
72
+ end
73
+
74
+ def remediation_all_selected_param
75
+ ActiveModel::Type::Boolean.new.cast(params[:isAllSelected])
76
+ end
40
77
  end
41
78
  end
@@ -13,7 +13,7 @@ module InsightsCloud
13
13
  token = Setting::RhCloud.find_by_name("rh_cloud_token")
14
14
  token.value = params.require(:value)
15
15
  token.save!
16
- InsightsCloud::Async::InsightsFullSync.perform_now
16
+ ForemanTasks.sync_task(InsightsCloud::Async::InsightsFullSync)
17
17
  redirect_to(:controller => "hits", :action => 'index')
18
18
  end
19
19
 
@@ -1,8 +1,7 @@
1
1
  module InsightsCloud
2
2
  class TasksController < ::ApplicationController
3
3
  def create
4
- job = InsightsCloud::Async::InsightsFullSync.perform_later
5
- task = ForemanTasks::Task.find_by(external_id: job.provider_job_id)
4
+ task = ForemanTasks.async_task(InsightsCloud::Async::InsightsFullSync)
6
5
 
7
6
  render json: {
8
7
  task: task,
@@ -1,3 +1,3 @@
1
1
  class InsightsResolution < ApplicationRecord
2
- belongs_to :rule, class_name: 'InsightsRule', foreign_key: 'rule_id', primary_key: 'rule_id', inverse_of: :resolutions
2
+ belongs_to :rule, class_name: 'InsightsRule', primary_key: 'rule_id', inverse_of: :resolutions
3
3
  end
@@ -26,5 +26,11 @@ module InventorySync
26
26
  N_('Successfully uploaded to your RH cloud inventory')
27
27
  end
28
28
  end
29
+
30
+ def to_status(options = {})
31
+ # this method used to calculate status.
32
+ # Since the calculation is done externally we should return the previously calculated status
33
+ status
34
+ end
29
35
  end
30
36
  end
@@ -14,12 +14,12 @@ class Setting::RhCloud < Setting
14
14
  def self.default_settings
15
15
  return unless ActiveRecord::Base.connection.table_exists?('settings')
16
16
  [
17
- set('allow_auto_inventory_upload', N_('Allow automatic upload of the host inventory to the Red Hat cloud'), true, N_('Allow automatic inventory uploads')),
18
- set('allow_auto_insights_sync', N_('Allow recommendations synchronization from Red Hat cloud'), false, N_('Allow recommendations synchronization')),
19
- set('obfuscate_inventory_hostnames', N_('Obfuscate host names sent to Red Hat cloud'), false, N_('Obfuscate host names')),
20
- set('obfuscate_inventory_ips', N_('Obfuscate ip addresses sent to Red Hat cloud'), false, N_('Obfuscate IPs')),
17
+ set('allow_auto_inventory_upload', N_('Enable automatic upload of your host inventory to the Red Hat cloud'), true, N_('Automatic inventory upload')),
18
+ set('allow_auto_insights_sync', N_('Enable automatic synchronization of Insights recommendations from the Red Hat cloud'), false, N_('Synchronize recommendations Automatically')),
19
+ set('obfuscate_inventory_hostnames', N_('Obfuscate host names sent to the Red Hat cloud'), false, N_('Obfuscate host names')),
20
+ set('obfuscate_inventory_ips', N_('Obfuscate ipv4 addresses sent to the Red Hat cloud'), false, N_('Obfuscate host ipv4 addresses')),
21
21
  set('rh_cloud_token', N_('Authentication token to Red Hat cloud services. Used to authenticate requests to cloud APIs'), nil, N_('Red Hat Cloud token'), nil, encrypted: true),
22
- set('exclude_installed_packages', N_('Exclude installed packages from Red Hat cloud inventory report'), false, N_("Don't upload installed packages")),
22
+ set('exclude_installed_packages', N_('Exclude installed packages from being uploaded to the Red Hat cloud'), false, N_("Exclude installed Packages")),
23
23
  set('include_parameter_tags', N_('Should import include parameter tags from Foreman?'), false, N_('Include parameters in insights-client reports')),
24
24
  ]
25
25
  end
@@ -59,7 +59,7 @@ module ForemanRhCloud
59
59
  end
60
60
 
61
61
  def foreman_host
62
- ::Host.unscoped.friendly.find(::SmartProxy.default_capsule.name)
62
+ ForemanRhCloud.foreman_host
63
63
  end
64
64
  end
65
65
  end
@@ -27,6 +27,7 @@ module ForemanRhCloud
27
27
  headers: {
28
28
  params: forward_params,
29
29
  user_agent: http_user_agent(original_request),
30
+ content_type: original_request.media_type,
30
31
  },
31
32
  }
32
33
  base_params.merge(path_params(original_request.path, certs))
@@ -0,0 +1,78 @@
1
+ module ForemanRhCloud
2
+ class RemediationsRetriever
3
+ include CloudAuth
4
+
5
+ attr_reader :logger
6
+
7
+ def initialize(hit_remediation_pairs, logger: Logger.new(IO::NULL))
8
+ @hit_remediation_pairs = hit_remediation_pairs
9
+ @logger = logger
10
+
11
+ logger.debug("Querying playbook for #{hit_remediation_pairs}")
12
+ end
13
+
14
+ def create_playbook
15
+ response = query_playbook
16
+
17
+ logger.debug("Got playbook response: #{response.body}")
18
+
19
+ response.body
20
+ end
21
+
22
+ private
23
+
24
+ def hit_ids
25
+ @hit_remediation_pairs.map { |pair| pair["hit_id"] }
26
+ end
27
+
28
+ def remediation_ids
29
+ @hit_remediation_pairs.map { |pair| pair["resolution_id"] }
30
+ end
31
+
32
+ def hits
33
+ @hits ||= Hash[
34
+ InsightsHit.joins(:insights_facet).where(id: hit_ids).pluck(:id, 'insights_facets.uuid')
35
+ ]
36
+ end
37
+
38
+ def pairs_by_remediation_id
39
+ @hit_remediation_pairs.group_by { |pair| pair["resolution_id"] }
40
+ end
41
+
42
+ def remediations
43
+ @remediations ||= Hash[
44
+ InsightsResolution.where(id: remediation_ids).pluck(:id, :resolution_type, :rule_id).map do |id, resolution_type, rule_id|
45
+ [id, {resolution_type: resolution_type, rule_id: rule_id}]
46
+ end
47
+ ]
48
+ end
49
+
50
+ def playbook_request
51
+ {
52
+ issues: pairs_by_remediation_id.map do |remediation_id, pairs|
53
+ {
54
+ resolution: remediations[remediation_id][:resolution_type],
55
+ id: InsightsCloud.remediation_rule_id(remediations[remediation_id][:rule_id]),
56
+ systems: pairs.map do |pair|
57
+ hits[pair["hit_id"]]
58
+ end,
59
+ }
60
+ end,
61
+ }
62
+ end
63
+
64
+ def query_playbook
65
+ RestClient::Request.execute(
66
+ method: :post,
67
+ url: InsightsCloud.playbook_url,
68
+ verify_ssl: ForemanRhCloud.verify_ssl_method,
69
+ proxy: ForemanRhCloud.transformed_http_proxy_string(logger: logger),
70
+ headers: {
71
+ content_type: :json,
72
+ Authorization: "Bearer #{rh_credentials}",
73
+ },
74
+ payload: playbook_request.to_json
75
+ )
76
+ end
77
+ end
78
+ end
@@ -0,0 +1,22 @@
1
+ module ForemanRhCloud
2
+ # Macro to fetch remediation playbook from cloud.redhat.com
3
+ module TemplateRendererHelper
4
+ extend ActiveSupport::Concern
5
+ extend ApipieDSL::Module
6
+
7
+ apipie :class, 'Macros related to Red Hat cloud interface' do
8
+ name 'RHCloud'
9
+ sections only: %w[all jobs]
10
+ end
11
+
12
+ apipie :method, 'Returns a playbook generated from Red Hat cloud recommendations' do
13
+ required :hit_remediation_pairs, String, desc: 'JSON-encoded array of hashes in the form of [{hit_id: 1, remediation_id: 2}, ...]'
14
+ returns String, desc: 'Playbook generated for the specific recommendations and hosts'
15
+ end
16
+ def remediations_playbook(hit_remediation_pairs)
17
+ hit_remediation_pairs = JSON.parse(hit_remediation_pairs)
18
+ retriever = ForemanRhCloud::RemediationsRetriever.new(hit_remediation_pairs, logger: template_logger)
19
+ retriever.create_playbook
20
+ end
21
+ end
22
+ end
@@ -0,0 +1,14 @@
1
+ <%#
2
+ kind: job_template
3
+ name: Run remediations based on Insights recommendations
4
+ job_category: Red Hat Insights
5
+ description_format: 'Insights remediations for selected issues'
6
+ feature: rh_cloud_remediate_hosts
7
+ template_inputs:
8
+ - name: hit_remediation_pairs
9
+ description: Pairs of issues and selected remediation
10
+ input_type: user
11
+ required: true
12
+ provider_type: Ansible
13
+ %>
14
+ <%= remediations_playbook(input('hit_remediation_pairs')) %>
data/config/routes.rb CHANGED
@@ -10,7 +10,7 @@ Rails.application.routes.draw do
10
10
 
11
11
  post 'cloud_connector', to: 'uploads#enable_cloud_connector'
12
12
 
13
- resources :tasks, only: [:create]
13
+ resources :tasks, only: [:create, :show]
14
14
  end
15
15
 
16
16
  namespace :insights_cloud do
@@ -19,6 +19,7 @@ Rails.application.routes.draw do
19
19
  resources :hits, except: %i[show] do
20
20
  collection do
21
21
  get 'auto_complete_search'
22
+ get 'resolutions', to: 'hits#resolutions'
22
23
  end
23
24
  end
24
25
  match 'hits/:host_id', to: 'hits#show', via: :get
@@ -0,0 +1,10 @@
1
+ class ChangeResolutions < ActiveRecord::Migration[5.2]
2
+ def change
3
+ remove_column :insights_resolutions, :system_type, :integer
4
+ remove_column :insights_resolutions, :has_playbook, :boolean
5
+ rename_column :insights_resolutions, :resolution, :description
6
+ add_column :insights_resolutions, :needs_reboot, :boolean
7
+ add_column :insights_resolutions, :resolution_risk, :integer
8
+ add_column :insights_resolutions, :resolution_type, :string
9
+ end
10
+ end
@@ -0,0 +1,11 @@
1
+ blueprints = [
2
+ {
3
+ group: N_('Red Hat Insights'),
4
+ name: 'insights_satellite_hits',
5
+ message: N_('Satellite server has %{hits_count} recommendations by Red Hat'),
6
+ level: 'warning',
7
+ },
8
+ ]
9
+
10
+ blueprints.each { |blueprint| UINotifications::Seed.new(blueprint).configure }
11
+
@@ -0,0 +1,14 @@
1
+ organizations = Organization.unscoped.all
2
+ locations = Location.unscoped.all
3
+ User.as_anonymous_admin do
4
+ JobTemplate.without_auditing do
5
+ Dir[File.join("#{ForemanRhCloud::Engine.root}/app/views/job_templates/**/*.erb")].each do |template|
6
+ template = JobTemplate.import_raw!(File.read(template),
7
+ :default => true,
8
+ :lock => true,
9
+ :update => true)
10
+ template.organizations = organizations if template.present?
11
+ template.locations = locations if template.present?
12
+ end
13
+ end
14
+ end
@@ -56,6 +56,15 @@ module ForemanInventoryUpload
56
56
  @upload_url ||= ENV['SATELLITE_INVENTORY_UPLOAD_URL'] || 'https://cert.cloud.redhat.com/api/ingress/v1/upload'
57
57
  end
58
58
 
59
+ def self.slice_size
60
+ @slice_size ||= (ENV['SATELLITE_INVENTORY_SLICE_SIZE'] || '1000').to_i
61
+ end
62
+
63
+ def self.max_org_size
64
+ # Set max amount of hosts per organization for automatic uploads
65
+ @max_org_size ||= (ENV['SATELLITE_INVENTORY_MAX_ORG_SIZE'] || 150_000).to_i
66
+ end
67
+
59
68
  def self.ensure_folder(folder)
60
69
  FileUtils.mkdir_p(folder)
61
70
  folder