katello 4.3.0 → 4.4.0.rc1

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of katello might be problematic. Click here for more details.

Files changed (337) hide show
  1. checksums.yaml +4 -4
  2. data/app/assets/javascripts/katello/common/vendor.js +0 -1
  3. data/app/controllers/katello/api/v2/api_controller.rb +13 -4
  4. data/app/controllers/katello/api/v2/content_view_versions_controller.rb +3 -0
  5. data/app/controllers/katello/api/v2/content_views_controller.rb +46 -0
  6. data/app/controllers/katello/api/v2/host_packages_controller.rb +21 -1
  7. data/app/controllers/katello/api/v2/hosts_bulk_actions_controller.rb +30 -1
  8. data/app/controllers/katello/api/v2/organizations_controller.rb +10 -6
  9. data/app/controllers/katello/api/v2/packages_controller.rb +4 -0
  10. data/app/controllers/katello/api/v2/repositories_controller.rb +17 -7
  11. data/app/controllers/katello/api/v2/repository_sets_controller.rb +7 -0
  12. data/app/controllers/katello/concerns/api/v2/repository_content_controller.rb +3 -0
  13. data/app/controllers/katello/concerns/hosts_controller_extensions.rb +32 -0
  14. data/app/controllers/katello/remote_execution_controller.rb +5 -4
  15. data/app/helpers/katello/content_source_helper.rb +43 -0
  16. data/app/helpers/katello/hosts_and_hostgroups_helper.rb +4 -0
  17. data/app/lib/actions/katello/cdn_configuration/update.rb +5 -7
  18. data/app/lib/actions/katello/content_view/publish.rb +5 -0
  19. data/app/lib/actions/katello/content_view_version/incremental_update.rb +17 -3
  20. data/app/lib/actions/katello/repository/discover.rb +1 -14
  21. data/app/lib/actions/katello/repository/filtered_index_content.rb +1 -1
  22. data/app/lib/actions/pulp3/content_view_version/import.rb +7 -0
  23. data/app/lib/actions/pulp3/orchestration/content_view_version/import.rb +7 -5
  24. data/app/lib/actions/pulp3/repository/copy_content.rb +1 -1
  25. data/app/lib/actions/pulp3/repository/save_artifact.rb +1 -0
  26. data/app/lib/katello/concerns/base_template_scope_extensions.rb +11 -0
  27. data/app/lib/katello/errors.rb +3 -3
  28. data/app/lib/katello/logging.rb +6 -1
  29. data/app/lib/katello/repo_discovery.rb +27 -19
  30. data/app/lib/katello/resources/cdn/katello_cdn.rb +41 -3
  31. data/app/lib/katello/resources/cdn.rb +4 -2
  32. data/app/lib/katello/util/deduplication_migrator.rb +105 -0
  33. data/app/models/katello/candlepin/repository_mapper.rb +1 -1
  34. data/app/models/katello/cdn_configuration.rb +38 -6
  35. data/app/models/katello/concerns/host_managed_extensions.rb +23 -1
  36. data/app/models/katello/concerns/organization_extensions.rb +5 -1
  37. data/app/models/katello/concerns/pulp_database_unit.rb +59 -173
  38. data/app/models/katello/concerns/remote_execution_proxy_selector_extensions.rb +11 -0
  39. data/app/models/katello/concerns/smart_proxy_extensions.rb +1 -1
  40. data/app/models/katello/content_view.rb +4 -4
  41. data/app/models/katello/content_view_filter.rb +1 -1
  42. data/app/models/katello/content_view_history.rb +1 -1
  43. data/app/models/katello/content_view_version.rb +6 -2
  44. data/app/models/katello/content_view_version_export_history.rb +1 -1
  45. data/app/models/katello/deb.rb +1 -3
  46. data/app/models/katello/docker_meta_tag.rb +1 -1
  47. data/app/models/katello/erratum.rb +0 -15
  48. data/app/models/katello/glue/pulp/repos.rb +1 -1
  49. data/app/models/katello/host/content_facet.rb +2 -27
  50. data/app/models/katello/host/info_provider.rb +9 -0
  51. data/app/models/katello/host/subscription_facet.rb +2 -2
  52. data/app/models/katello/hostgroup/content_facet.rb +2 -2
  53. data/app/models/katello/product.rb +1 -1
  54. data/app/models/katello/product_content.rb +2 -2
  55. data/app/models/katello/repository.rb +10 -9
  56. data/app/models/katello/root_repository.rb +24 -13
  57. data/app/models/katello/rpm.rb +8 -2
  58. data/app/models/setting/content.rb +6 -3
  59. data/app/services/cert/rhsm_client.rb +1 -5
  60. data/app/services/katello/content_unit_indexer.rb +166 -0
  61. data/app/services/katello/organization_creator.rb +12 -4
  62. data/app/services/katello/pulp/repository/docker.rb +1 -1
  63. data/app/services/katello/pulp/repository/yum.rb +0 -54
  64. data/app/services/katello/pulp/repository.rb +0 -6
  65. data/app/services/katello/pulp3/ansible_collection.rb +26 -10
  66. data/app/services/katello/pulp3/api/apt.rb +7 -0
  67. data/app/services/katello/pulp3/content_view_version/import.rb +11 -2
  68. data/app/services/katello/pulp3/deb.rb +10 -9
  69. data/app/services/katello/pulp3/docker_manifest.rb +6 -5
  70. data/app/services/katello/pulp3/docker_manifest_list.rb +23 -6
  71. data/app/services/katello/pulp3/docker_tag.rb +16 -7
  72. data/app/services/katello/pulp3/erratum.rb +51 -56
  73. data/app/services/katello/pulp3/file_unit.rb +9 -6
  74. data/app/services/katello/pulp3/generic_content_unit.rb +11 -12
  75. data/app/services/katello/pulp3/module_stream.rb +76 -30
  76. data/app/services/katello/pulp3/package_group.rb +5 -5
  77. data/app/services/katello/pulp3/pulp_content_unit.rb +19 -11
  78. data/app/services/katello/pulp3/repository/apt.rb +5 -3
  79. data/app/services/katello/pulp3/repository/docker.rb +14 -7
  80. data/app/services/katello/pulp3/repository/generic.rb +1 -1
  81. data/app/services/katello/pulp3/repository/yum.rb +10 -12
  82. data/app/services/katello/pulp3/repository.rb +26 -7
  83. data/app/services/katello/pulp3/repository_mirror.rb +18 -5
  84. data/app/services/katello/pulp3/rpm.rb +13 -13
  85. data/app/services/katello/pulp3/srpm.rb +10 -9
  86. data/app/services/katello/repository_type.rb +15 -4
  87. data/app/services/katello/repository_type_manager.rb +1 -1
  88. data/app/services/katello/ui_notifications/subscriptions/manifest_expired_warning.rb +1 -1
  89. data/app/services/katello/upstream_connection_checker.rb +2 -2
  90. data/app/views/foreman/job_templates/change_content_source.erb +1 -31
  91. data/app/views/foreman/job_templates/install_errata.erb +6 -9
  92. data/app/views/foreman/job_templates/install_errata_by_search_query.erb +26 -0
  93. data/app/views/foreman/job_templates/install_packages_by_search_query.erb +19 -0
  94. data/app/views/katello/api/v2/cdn_configurations/show.json.rabl +1 -1
  95. data/app/views/katello/api/v2/environments/show.json.rabl +9 -0
  96. data/app/views/katello/api/v2/repositories/show.json.rabl +2 -0
  97. data/app/views/katello/layouts/react.html.erb +0 -1
  98. data/app/views/katello/sync_management/_repo.html.erb +36 -25
  99. data/config/initializers/monkeys.rb +0 -1
  100. data/config/routes/api/v2.rb +1 -0
  101. data/config/routes/overrides.rb +3 -0
  102. data/config/routes.rb +2 -0
  103. data/db/migrate/20150930183738_migrate_content_hosts.rb +1 -1
  104. data/db/migrate/20180612164926_add_content_org_id.rb +2 -2
  105. data/db/migrate/20211201154845_add_unique_indexes.rb +20 -0
  106. data/db/migrate/20211208034230_add_content_view_and_lifecycle_environment.rb +6 -0
  107. data/db/migrate/20211220185935_clean_duplicate_content_units.rb +144 -0
  108. data/db/migrate/20220110223754_update_disconnected_settings.rb +20 -0
  109. data/db/migrate/20220120163252_fix_docker_download_policy.rb +11 -0
  110. data/db/migrate/20220124191056_add_type_to_cdn_configuration.rb +22 -0
  111. data/db/migrate/20220127120843_fix_debian_download_policy.rb +11 -0
  112. data/db/migrate/20220204171908_rename_docker_tags_whitelist_and_add_exclude_tags.rb +8 -0
  113. data/db/migrate/20220207140355_change_deb_attributes_size_limit.rb +7 -0
  114. data/db/seeds.d/111-upgrade_tasks.rb +2 -1
  115. data/engines/bastion/app/views/bastion/layouts/application.html.erb +0 -1
  116. data/engines/bastion/app/views/bastion/layouts/assets.html.erb +0 -1
  117. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/content-credentials/content-credentials.controller.js +1 -1
  118. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/content-credentials/details/content-credential-products.controller.js +1 -1
  119. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/content-credentials/details/content-credential-repositories.controller.js +1 -1
  120. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/content-hosts/bulk/views/content-hosts-bulk-traces-modal.html +1 -1
  121. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/environments/details/views/environment.html +26 -1
  122. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/i18n/bastion_katello.pot +212 -152
  123. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/packages/packages.controller.js +1 -0
  124. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/products/details/repositories/details/repository-details-info.controller.js +20 -8
  125. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/products/details/repositories/details/repository-details-manage-content.controller.js +2 -3
  126. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/products/details/repositories/details/repository-details.controller.js +8 -3
  127. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/products/details/repositories/details/views/repository-info.html +34 -30
  128. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/products/details/repositories/new/new-repository.controller.js +16 -2
  129. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/products/details/repositories/new/views/new-repository.html +35 -15
  130. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/products/discovery/discovery.controller.js +5 -4
  131. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/products/products.routes.js +4 -3
  132. data/lib/katello/engine.rb +4 -0
  133. data/lib/katello/permission_creator.rb +1 -1
  134. data/lib/katello/permissions/host_permissions.rb +3 -1
  135. data/lib/katello/plugin.rb +4 -1
  136. data/lib/katello/repository_types/deb.rb +0 -1
  137. data/lib/katello/repository_types/docker.rb +4 -4
  138. data/lib/katello/repository_types/file.rb +0 -1
  139. data/lib/katello/repository_types/ostree.rb +4 -0
  140. data/lib/katello/repository_types/python.rb +5 -1
  141. data/lib/katello/repository_types/yum.rb +2 -9
  142. data/lib/katello/tasks/content_view_import_only.rake +34 -0
  143. data/lib/katello/tasks/import_applicability.rake +1 -1
  144. data/lib/katello/tasks/jenkins.rake +0 -2
  145. data/lib/katello/tasks/repository.rake +4 -1
  146. data/lib/katello/tasks/upgrades/4.4/publish_import_cvvs.rake +17 -0
  147. data/lib/katello/version.rb +1 -1
  148. data/locale/action_names.rb +8 -7
  149. data/locale/bn/katello.po +1402 -650
  150. data/locale/cs/katello.po +1217 -96
  151. data/locale/de/katello.po +2359 -1347
  152. data/locale/en/katello.po +1216 -94
  153. data/locale/es/katello.po +2201 -1172
  154. data/locale/fr/katello.po +2601 -1615
  155. data/locale/gu/katello.po +1564 -814
  156. data/locale/hi/katello.po +1563 -810
  157. data/locale/it/katello.po +1311 -282
  158. data/locale/ja/katello.po +2534 -1518
  159. data/locale/katello.pot +3430 -1326
  160. data/locale/kn/katello.po +1564 -812
  161. data/locale/ko/katello.po +1441 -409
  162. data/locale/mr/katello.po +1564 -776
  163. data/locale/or/katello.po +1565 -813
  164. data/locale/pa/katello.po +1559 -792
  165. data/locale/pt/katello.po +1314 -277
  166. data/locale/pt_BR/katello.po +2226 -1181
  167. data/locale/ru/katello.po +1587 -563
  168. data/locale/ta/katello.po +1373 -619
  169. data/locale/te/katello.po +1564 -810
  170. data/locale/zh_CN/katello.po +2936 -1890
  171. data/locale/zh_TW/katello.po +1508 -606
  172. data/webpack/__mocks__/foremanReact/{redux/actions/toasts.js → components/ToastsList/index.js} +3 -2
  173. data/webpack/components/ActionableDetail.js +35 -21
  174. data/webpack/components/Content/Details/__tests__/ContentDetailInfo.test.js +0 -2
  175. data/webpack/components/Content/Details/__tests__/ContentDetailRepositories.test.js +0 -2
  176. data/webpack/components/Content/Details/__tests__/ContentDetails.test.js +0 -2
  177. data/webpack/components/Content/__tests__/ContentPage.test.js +0 -2
  178. data/webpack/components/Content/__tests__/ContentTable.test.js +0 -2
  179. data/webpack/components/EditableSwitch.js +8 -2
  180. data/webpack/components/EditableTextInput/EditableTextInput.js +44 -86
  181. data/webpack/components/EditableTextInput/__tests__/editableTextInput.test.js +3 -3
  182. data/webpack/components/Errata/index.js +19 -11
  183. data/webpack/components/Packages/index.js +1 -1
  184. data/webpack/components/Search/Search.js +5 -2
  185. data/webpack/components/Search/__tests__/search.test.js +2 -3
  186. data/webpack/components/SelectOrg/SetOrganization.js +1 -1
  187. data/webpack/components/Table/PageControls.js +3 -6
  188. data/webpack/components/Table/TableHooks.js +46 -7
  189. data/webpack/components/Table/TableWrapper.js +14 -3
  190. data/webpack/components/TypeAhead/TypeAhead.js +5 -1
  191. data/webpack/components/TypeAhead/pf4Search/TypeAheadInput.js +4 -1
  192. data/webpack/components/TypeAhead/pf4Search/TypeAheadSearch.js +2 -1
  193. data/webpack/components/extensions/HostDetails/Cards/ContentViewDetailsCard.js +1 -1
  194. data/webpack/components/extensions/HostDetails/Cards/ErrataOverviewCard.js +3 -4
  195. data/webpack/components/extensions/HostDetails/HostDetailsActions.js +2 -0
  196. data/webpack/components/extensions/HostDetails/HostPackages/HostPackagesActions.js +18 -2
  197. data/webpack/components/extensions/HostDetails/HostPackages/HostPackagesConstants.js +12 -0
  198. data/webpack/components/extensions/HostDetails/Tabs/ContentTab/SecondaryTabsRoutes.js +1 -1
  199. data/webpack/components/extensions/HostDetails/Tabs/{ErrataTab.js → ErrataTab/ErrataTab.js} +30 -21
  200. data/webpack/components/extensions/HostDetails/Tabs/{ErrataTab.scss → ErrataTab/ErrataTab.scss} +0 -0
  201. data/webpack/components/extensions/HostDetails/Tabs/{ErratumExpansionContents.js → ErrataTab/ErratumExpansionContents.js} +0 -0
  202. data/webpack/components/extensions/HostDetails/Tabs/{ErratumExpansionDetail.js → ErrataTab/ErratumExpansionDetail.js} +0 -0
  203. data/webpack/components/extensions/HostDetails/{HostErrata → Tabs/ErrataTab}/HostErrataActions.js +3 -3
  204. data/webpack/components/extensions/HostDetails/{HostErrata → Tabs/ErrataTab}/HostErrataConstants.js +11 -0
  205. data/webpack/components/extensions/HostDetails/{HostErrata → Tabs/ErrataTab}/HostErrataSelectors.js +0 -0
  206. data/webpack/components/extensions/HostDetails/Tabs/PackageInstallModal.js +279 -0
  207. data/webpack/components/extensions/HostDetails/Tabs/PackageInstallModal.scss +3 -0
  208. data/webpack/components/extensions/HostDetails/Tabs/PackagesTab.js +125 -8
  209. data/webpack/components/extensions/HostDetails/Tabs/RemoteExecutionActions.js +27 -4
  210. data/webpack/components/extensions/HostDetails/Tabs/RemoteExecutionConstants.js +2 -1
  211. data/webpack/components/extensions/HostDetails/Tabs/RepositorySetsTab/RepositorySetsTab.js +6 -3
  212. data/webpack/components/extensions/HostDetails/Tabs/{EnableTracerEmptyState.js → TracesTab/EnableTracerEmptyState.js} +0 -0
  213. data/webpack/components/extensions/HostDetails/Tabs/{EnableTracerModal.js → TracesTab/EnableTracerModal.js} +3 -2
  214. data/webpack/components/extensions/HostDetails/Tabs/{HostTracesActions.js → TracesTab/HostTracesActions.js} +2 -2
  215. data/webpack/components/extensions/HostDetails/Tabs/{HostTracesConstants.js → TracesTab/HostTracesConstants.js} +0 -0
  216. data/webpack/components/extensions/HostDetails/Tabs/{HostTracesSelectors.js → TracesTab/HostTracesSelectors.js} +0 -0
  217. data/webpack/components/extensions/HostDetails/Tabs/{TracesTab.js → TracesTab/TracesTab.js} +8 -5
  218. data/webpack/components/extensions/HostDetails/Tabs/{TracesTab.scss → TracesTab/TracesTab.scss} +0 -0
  219. data/webpack/components/extensions/HostDetails/Tabs/__tests__/errataTab.test.js +17 -18
  220. data/webpack/components/extensions/HostDetails/Tabs/__tests__/packageInstallModal.test.js +385 -0
  221. data/webpack/components/extensions/HostDetails/Tabs/__tests__/packages.fixtures.json +1 -1
  222. data/webpack/components/extensions/HostDetails/Tabs/__tests__/packagesTab.test.js +58 -7
  223. data/webpack/components/extensions/HostDetails/Tabs/__tests__/repositorySetsTab.test.js +2 -2
  224. data/webpack/components/extensions/HostDetails/Tabs/__tests__/tracesTab.test.js +4 -4
  225. data/webpack/components/extensions/HostDetails/Tabs/__tests__/yumInstallablePackages.fixtures.json +72 -0
  226. data/webpack/components/extensions/HostDetails/Tabs/customizedRexUrlHelpers.js +12 -5
  227. data/webpack/components/extensions/HostDetails/YumInstallablePackages/YumInstallablePackagesActions.js +18 -0
  228. data/webpack/components/extensions/HostDetails/YumInstallablePackages/YumInstallablePackagesConstants.js +3 -0
  229. data/webpack/components/extensions/HostDetails/YumInstallablePackages/YumInstallablePackagesSelectors.js +16 -0
  230. data/webpack/components/extensions/HostDetails/hostDetailsHelpers.js +19 -0
  231. data/webpack/components/pf3Table/components/Table.js +2 -3
  232. data/webpack/components/pf3Table/components/Table.test.js +0 -3
  233. data/webpack/components/pf3Table/components/__snapshots__/Table.test.js.snap +9 -8
  234. data/webpack/containers/Application/config.js +5 -0
  235. data/webpack/global_index.js +1 -1
  236. data/webpack/global_test_setup.js +1 -1
  237. data/webpack/index.js +7 -0
  238. data/webpack/scenes/AnsibleCollections/Details/__tests__/AnsibleCollectionDetails.test.js +0 -2
  239. data/webpack/scenes/AnsibleCollections/__tests__/AnsibleCollectionPage.test.js +0 -2
  240. data/webpack/scenes/AnsibleCollections/__tests__/AnsibleCollectionsTable.test.js +0 -2
  241. data/webpack/scenes/Content/ContentConfig.js +55 -5
  242. data/webpack/scenes/Content/ContentPage.js +1 -1
  243. data/webpack/scenes/Content/Details/ContentDetails.js +1 -1
  244. data/webpack/scenes/Content/Details/ContentInfo.js +1 -1
  245. data/webpack/scenes/Content/Details/ContentRepositories.js +1 -1
  246. data/webpack/scenes/Content/Details/__tests__/contentDetail.test.js +4 -4
  247. data/webpack/scenes/Content/Table/ContentTable.js +1 -1
  248. data/webpack/scenes/Content/__tests__/contentTable.test.js +3 -3
  249. data/webpack/scenes/ContentViews/ContentViewsConstants.js +2 -1
  250. data/webpack/scenes/ContentViews/Delete/__tests__/contentViewDelete.test.js +6 -6
  251. data/webpack/scenes/ContentViews/Details/ComponentContentViews/__tests__/contentViewComponents.test.js +2 -2
  252. data/webpack/scenes/ContentViews/Details/ContentViewDetailActions.js +21 -27
  253. data/webpack/scenes/ContentViews/Details/ContentViewDetailSelectors.js +5 -5
  254. data/webpack/scenes/ContentViews/Details/ContentViewInfo.js +8 -3
  255. data/webpack/scenes/ContentViews/Details/DetailsContainer.js +11 -16
  256. data/webpack/scenes/ContentViews/Details/Filters/ContentViewFilterDetails.js +2 -2
  257. data/webpack/scenes/ContentViews/Details/Filters/ContentViewFilterDetailsHeader.js +14 -8
  258. data/webpack/scenes/ContentViews/Details/Filters/MatchContentModal/__tests__/CVRpmMatchContentModal.test.js +2 -2
  259. data/webpack/scenes/ContentViews/Details/Filters/__tests__/CVContainerImageFilterContent.test.js +2 -3
  260. data/webpack/scenes/ContentViews/Details/Filters/__tests__/CVRpmFilterContent.test.js +2 -9
  261. data/webpack/scenes/ContentViews/Details/Filters/__tests__/contentViewFilterDetails.test.js +3 -5
  262. data/webpack/scenes/ContentViews/Details/Filters/__tests__/contentViewFilters.test.js +2 -10
  263. data/webpack/scenes/ContentViews/Details/Filters/__tests__/cvErrataIDFilter.test.js +2 -3
  264. data/webpack/scenes/ContentViews/Details/Filters/__tests__/cvModuleStreamFilter.test.js +2 -3
  265. data/webpack/scenes/ContentViews/Details/Histories/__tests__/contentViewHistory.test.js +2 -2
  266. data/webpack/scenes/ContentViews/Details/Repositories/ContentCounts.js +1 -1
  267. data/webpack/scenes/ContentViews/Details/Repositories/ContentViewRepositories.js +23 -2
  268. data/webpack/scenes/ContentViews/Details/Repositories/__tests__/contentViewAddRemove.test.js +11 -5
  269. data/webpack/scenes/ContentViews/Details/Versions/ContentViewVersionContent.js +16 -17
  270. data/webpack/scenes/ContentViews/Details/Versions/ContentViewVersions.js +1 -1
  271. data/webpack/scenes/ContentViews/Details/Versions/Delete/__tests__/cvVersionRemove.test.js +6 -6
  272. data/webpack/scenes/ContentViews/Details/Versions/VersionDetails/ContentViewVersionDetailConfig.js +30 -34
  273. data/webpack/scenes/ContentViews/Details/Versions/VersionDetails/ContentViewVersionDetails.js +9 -8
  274. data/webpack/scenes/ContentViews/Details/Versions/VersionDetails/ContentViewVersionDetailsHeader.js +13 -15
  275. data/webpack/scenes/ContentViews/Details/Versions/VersionDetails/ContentViewVersionRepositoryCell.js +1 -1
  276. data/webpack/scenes/ContentViews/Details/Versions/VersionDetails/__tests__/ContentViewVersionDetails.test.js +4 -4
  277. data/webpack/scenes/ContentViews/Details/Versions/__tests__/contentViewVersions.test.js +3 -3
  278. data/webpack/scenes/ContentViews/Details/__tests__/contentViewDetail.test.js +5 -3
  279. data/webpack/scenes/ContentViews/__tests__/contentViewPage.test.js +2 -2
  280. data/webpack/scenes/Hosts/ChangeContentSource/actions.js +43 -0
  281. data/webpack/scenes/Hosts/ChangeContentSource/components/ContentSourceForm.js +87 -0
  282. data/webpack/scenes/Hosts/ChangeContentSource/components/ContentSourceTemplate.js +90 -0
  283. data/webpack/scenes/Hosts/ChangeContentSource/components/FormField.js +43 -0
  284. data/webpack/scenes/Hosts/ChangeContentSource/constants.js +3 -0
  285. data/webpack/scenes/Hosts/ChangeContentSource/helpers.js +27 -0
  286. data/webpack/scenes/Hosts/ChangeContentSource/index.js +126 -0
  287. data/webpack/scenes/Hosts/ChangeContentSource/selectors.js +42 -0
  288. data/webpack/scenes/Hosts/ChangeContentSource/styles.scss +11 -0
  289. data/webpack/scenes/ModuleStreams/Details/Profiles/__tests__/ModuleStreamDetailProfiles.test.js +0 -1
  290. data/webpack/scenes/ModuleStreams/Details/__tests__/ModuleStreamDetails.test.js +0 -2
  291. data/webpack/scenes/ModuleStreams/__tests__/ModuleStreamPage.test.js +0 -2
  292. data/webpack/scenes/ModuleStreams/__tests__/ModuleStreamsTable.test.js +0 -2
  293. data/webpack/scenes/Organizations/OrganizationActions.js +5 -1
  294. data/webpack/scenes/RedHatRepositories/RedHatRepositoriesPage.js +31 -1
  295. data/webpack/scenes/RedHatRepositories/__tests__/RedHatRepositoriesPage.test.js +16 -0
  296. data/webpack/scenes/RedHatRepositories/__tests__/__snapshots__/RedHatRepositoriesPage.test.js.snap +11 -2
  297. data/webpack/scenes/RedHatRepositories/helpers.js +5 -5
  298. data/webpack/scenes/RedHatRepositories/index.js +11 -3
  299. data/webpack/scenes/Subscriptions/Manifest/CdnConfigurationTab/AirGappedTypeForm.js +81 -0
  300. data/webpack/scenes/Subscriptions/Manifest/CdnConfigurationTab/CdnConfigurationConstants.js +13 -0
  301. data/webpack/scenes/Subscriptions/Manifest/{CdnConfigurationForm.scss → CdnConfigurationTab/CdnConfigurationForm.scss} +0 -0
  302. data/webpack/scenes/Subscriptions/Manifest/CdnConfigurationTab/CdnTypeForm.js +106 -0
  303. data/webpack/scenes/Subscriptions/Manifest/CdnConfigurationTab/UpstreamServerTypeForm.js +259 -0
  304. data/webpack/scenes/Subscriptions/Manifest/CdnConfigurationTab/__tests__/AirGappedTypeForm.test.js +44 -0
  305. data/webpack/scenes/Subscriptions/Manifest/CdnConfigurationTab/__tests__/CdnTypeForm.test.js +67 -0
  306. data/webpack/scenes/Subscriptions/Manifest/{__tests__/CdnConfigurationForm.test.js → CdnConfigurationTab/__tests__/UpstreamServerTypeForm.test.js} +46 -17
  307. data/webpack/scenes/Subscriptions/Manifest/CdnConfigurationTab/index.js +97 -0
  308. data/webpack/scenes/Subscriptions/Manifest/ManageManifestModal.js +6 -1
  309. data/webpack/scenes/Subscriptions/UpstreamSubscriptions/__tests__/UpstreamSubscriptionsPage.test.js +0 -1
  310. data/webpack/scenes/Subscriptions/__tests__/SubscriptionsPage.test.js +0 -1
  311. data/webpack/scenes/Subscriptions/__tests__/__snapshots__/SubscriptionsActions.test.js.snap +3 -2
  312. data/webpack/scenes/Subscriptions/components/SubscriptionsTable/__tests__/SubscriptionsTable.test.js +4 -0
  313. data/webpack/scenes/Subscriptions/components/SubscriptionsTable/__tests__/__snapshots__/SubscriptionsTable.test.js.snap +9 -0
  314. data/webpack/scenes/Tasks/TaskActions.js +1 -1
  315. data/webpack/scenes/Tasks/__tests__/__snapshots__/TaskActions.test.js.snap +3 -2
  316. data/webpack/services/api/testHelpers.js +5 -3
  317. data/webpack/utils/helpers.js +6 -3
  318. metadata +81 -58
  319. data/app/services/katello/pulp/deb.rb +0 -55
  320. data/app/services/katello/pulp/distribution.rb +0 -7
  321. data/app/services/katello/pulp/docker_blob.rb +0 -7
  322. data/app/services/katello/pulp/docker_manifest.rb +0 -13
  323. data/app/services/katello/pulp/docker_manifest_list.rb +0 -14
  324. data/app/services/katello/pulp/docker_tag.rb +0 -14
  325. data/app/services/katello/pulp/erratum.rb +0 -129
  326. data/app/services/katello/pulp/file_unit.rb +0 -21
  327. data/app/services/katello/pulp/module_stream.rb +0 -39
  328. data/app/services/katello/pulp/package_category.rb +0 -7
  329. data/app/services/katello/pulp/package_group.rb +0 -20
  330. data/app/services/katello/pulp/pulp_content_unit.rb +0 -156
  331. data/app/services/katello/pulp/rpm.rb +0 -57
  332. data/app/services/katello/pulp/srpm.rb +0 -29
  333. data/app/services/katello/pulp/yum_metadata_file.rb +0 -30
  334. data/lib/monkeys/pulp3_13_checksumfix.rb +0 -17
  335. data/webpack/__mocks__/foremanReact/components/Pagination/PaginationHooks.js +0 -2
  336. data/webpack/__mocks__/foremanReact/components/Pagination/PaginationWrapper.js +0 -2
  337. data/webpack/scenes/Subscriptions/Manifest/CdnConfigurationForm.js +0 -185
@@ -0,0 +1,279 @@
1
+ import React, { useState } from 'react';
2
+ import { Modal, Button, Dropdown, DropdownItem, DropdownToggle, DropdownDirection, DropdownToggleAction } from '@patternfly/react-core';
3
+ import { CaretDownIcon, CaretUpIcon } from '@patternfly/react-icons';
4
+ import { useSelector, useDispatch } from 'react-redux';
5
+ import { FormattedMessage } from 'react-intl';
6
+ import { Thead, Th, Tbody, Tr, Td, TableVariant } from '@patternfly/react-table';
7
+ import { noop } from 'foremanReact/common/helpers';
8
+ import { translate as __ } from 'foremanReact/common/I18n';
9
+ import { urlBuilder } from 'foremanReact/common/urlHelpers';
10
+ import { selectAPIResponse } from 'foremanReact/redux/API/APISelectors';
11
+ import PropTypes from 'prop-types';
12
+ import TableWrapper from '../../../Table/TableWrapper';
13
+ import { useBulkSelect } from '../../../Table/TableHooks';
14
+ import { HOST_YUM_INSTALLABLE_PACKAGES_KEY } from '../YumInstallablePackages/YumInstallablePackagesConstants';
15
+ import { selectHostYumInstallablePackagesStatus } from '../YumInstallablePackages/YumInstallablePackagesSelectors';
16
+ import { getHostYumInstallablePackages } from '../YumInstallablePackages/YumInstallablePackagesActions';
17
+ import './PackageInstallModal.scss';
18
+ import { installPackageBySearch } from './RemoteExecutionActions';
19
+ import { katelloPackageInstallBySearchUrl, katelloPackageInstallUrl } from './customizedRexUrlHelpers';
20
+ import hostIdNotReady from '../HostDetailsActions';
21
+ import { installPackageViaKatelloAgent } from '../HostPackages/HostPackagesActions';
22
+
23
+ const InstallDropdown = ({
24
+ isDisabled, installViaRex, installViaKatelloAgent,
25
+ bulkCustomizedRexUrl, showKatelloAgent,
26
+ disableInstallViaKatelloAgent,
27
+ }) => {
28
+ const [isActionOpen, setIsActionOpen] = useState(false);
29
+ const onActionSelect = () => {
30
+ setIsActionOpen(false);
31
+ };
32
+ const onActionToggle = () => {
33
+ setIsActionOpen(prev => !prev);
34
+ };
35
+
36
+ const dropdownItems = [
37
+ <DropdownItem
38
+ key="install-k-agent"
39
+ component="button"
40
+ onClick={installViaKatelloAgent}
41
+ isDisabled={disableInstallViaKatelloAgent}
42
+ >
43
+ {__('Install via katello-agent')}
44
+ </DropdownItem>,
45
+ <DropdownItem key="install-rex" component="button" onClick={installViaRex}>
46
+ {__('Install via remote execution')}
47
+ </DropdownItem>,
48
+ <DropdownItem
49
+ key="install-customized-rex"
50
+ component="a"
51
+ href={bulkCustomizedRexUrl}
52
+ onClick={onActionSelect}
53
+ >
54
+ {__('Install via customized remote execution')}
55
+ </DropdownItem>,
56
+ ];
57
+
58
+ if (!showKatelloAgent) dropdownItems.shift();
59
+ const defaultRemoteAction = showKatelloAgent ? installViaKatelloAgent : installViaRex;
60
+
61
+ return (
62
+ <Dropdown
63
+ direction={DropdownDirection.up}
64
+ onSelect={onActionSelect}
65
+ toggle={
66
+ <DropdownToggle
67
+ isPrimary
68
+ isDisabled={isDisabled}
69
+ splitButtonItems={[
70
+ <DropdownToggleAction key="install" onClick={defaultRemoteAction}>
71
+ Install
72
+ </DropdownToggleAction>,
73
+ ]}
74
+ splitButtonVariant="action"
75
+ toggleIndicator={isActionOpen ? CaretUpIcon : CaretDownIcon}
76
+ onToggle={onActionToggle}
77
+ />
78
+ }
79
+ isOpen={isActionOpen}
80
+ dropdownItems={dropdownItems}
81
+ />
82
+ );
83
+ };
84
+
85
+ InstallDropdown.propTypes = {
86
+ isDisabled: PropTypes.bool,
87
+ installViaRex: PropTypes.func,
88
+ installViaKatelloAgent: PropTypes.func,
89
+ bulkCustomizedRexUrl: PropTypes.string,
90
+ showKatelloAgent: PropTypes.bool,
91
+ disableInstallViaKatelloAgent: PropTypes.bool,
92
+ };
93
+
94
+ InstallDropdown.defaultProps = {
95
+ isDisabled: false,
96
+ installViaRex: noop,
97
+ installViaKatelloAgent: noop,
98
+ bulkCustomizedRexUrl: '',
99
+ showKatelloAgent: false,
100
+ disableInstallViaKatelloAgent: false,
101
+ };
102
+
103
+ const PackageInstallModal = ({
104
+ isOpen, closeModal, hostId, hostName, showKatelloAgent,
105
+ }) => {
106
+ const emptyContentTitle = __('No packages available to install');
107
+ const emptyContentBody = __('No packages available to install on this host. Please check the host\'s content view and lifecycle environment.');
108
+ const emptySearchTitle = __('No matching packages found');
109
+ const emptySearchBody = __('Try changing your search settings.');
110
+ const columnHeaders = ['', __('Package'), __('Version')];
111
+ const response =
112
+ useSelector(state => selectAPIResponse(state, HOST_YUM_INSTALLABLE_PACKAGES_KEY));
113
+ const status = useSelector(state => selectHostYumInstallablePackagesStatus(state));
114
+ const dispatch = useDispatch();
115
+ const { results, ...metadata } = response;
116
+ const [suppressFirstFetch, setSuppressFirstFetch] = useState(false);
117
+
118
+ const {
119
+ searchQuery,
120
+ updateSearchQuery,
121
+ isSelected,
122
+ selectOne,
123
+ selectNone,
124
+ fetchBulkParams,
125
+ isSelectable,
126
+ selectedCount,
127
+ selectedResults,
128
+ ...selectAll
129
+ } = useBulkSelect({ results, metadata });
130
+ const fetchItems = (params) => {
131
+ if (!hostId) return hostIdNotReady;
132
+
133
+ if (results?.length > 0 && suppressFirstFetch) {
134
+ // If the modal has already been opened, no need to re-fetch the data that's already present
135
+ setSuppressFirstFetch(false);
136
+ return { type: 'HOST_APPLICABLE_PACKAGES_NOOP' };
137
+ }
138
+ return getHostYumInstallablePackages(hostId, params);
139
+ };
140
+
141
+ const selectedPackageNames = () => selectedResults.map(({ name }) => name);
142
+
143
+ const installViaRex = () => {
144
+ dispatch(installPackageBySearch({ hostname: hostName, search: fetchBulkParams() }));
145
+ selectNone();
146
+ closeModal();
147
+ };
148
+
149
+ const installViaKatelloAgent = () => {
150
+ dispatch(installPackageViaKatelloAgent(hostId, { packages: selectedPackageNames() }));
151
+ selectNone();
152
+ closeModal();
153
+ };
154
+
155
+ const handleModalClose = () => {
156
+ setSuppressFirstFetch(true);
157
+ closeModal();
158
+ };
159
+
160
+ const bulkCustomizedRexUrl = selectedCount ?
161
+ katelloPackageInstallBySearchUrl({ hostname: hostName, search: fetchBulkParams() }) :
162
+ '#';
163
+ const simpleBulkCustomizedRexUrl
164
+ = katelloPackageInstallUrl({ hostname: hostName, packages: selectedPackageNames() });
165
+ const enableSimpleRexUrl = !!selectedResults.length;
166
+
167
+ const modalActions = ([
168
+ <InstallDropdown
169
+ key="install"
170
+ isDisabled={!selectedCount}
171
+ installViaRex={installViaRex}
172
+ installViaKatelloAgent={installViaKatelloAgent}
173
+ bulkCustomizedRexUrl={enableSimpleRexUrl ? simpleBulkCustomizedRexUrl : bulkCustomizedRexUrl}
174
+ showKatelloAgent={showKatelloAgent}
175
+ disableInstallViaKatelloAgent={selectedResults.length === 0}
176
+ />,
177
+ <Button key="cancel" variant="link" onClick={handleModalClose}>
178
+ Cancel
179
+ </Button>,
180
+ ]);
181
+
182
+ return (
183
+ <Modal
184
+ isOpen={isOpen}
185
+ onClose={handleModalClose}
186
+ title={__('Install packages')}
187
+ width="50%"
188
+ actions={modalActions}
189
+ id="package-install-modal"
190
+ >
191
+ <FormattedMessage
192
+ className="pkg-install-modal-blurb"
193
+ id="pkg-install-modal-blurb"
194
+ defaultMessage={__('Select packages to install to the host {hostName}.')}
195
+ values={{
196
+ hostName: <strong>{hostName}</strong>,
197
+ }}
198
+ />
199
+ <TableWrapper
200
+ {...{
201
+ metadata,
202
+ emptyContentTitle,
203
+ emptyContentBody,
204
+ emptySearchTitle,
205
+ emptySearchBody,
206
+ status,
207
+ searchQuery,
208
+ updateSearchQuery,
209
+ selectedCount,
210
+ selectNone,
211
+ }
212
+ }
213
+ additionalListeners={[hostId]}
214
+ fetchItems={fetchItems}
215
+ searchPlaceholderText={__('Search available packages')}
216
+ autocompleteEndpoint={`/hosts/${hostId}/packages/auto_complete_search`}
217
+ foremanApiAutoComplete
218
+ variant={TableVariant.compact}
219
+ {...selectAll}
220
+ displaySelectAllCheckbox
221
+ >
222
+ <Thead>
223
+ <Tr>
224
+ {columnHeaders.map(col =>
225
+ <Th key={col}>{col}</Th>)}
226
+ <Th />
227
+ </Tr>
228
+ </Thead>
229
+ <Tbody>
230
+ {results?.map((pkg, rowIndex) => {
231
+ const {
232
+ id,
233
+ name: packageName,
234
+ rpm_id: rpmId,
235
+ version,
236
+ } = pkg;
237
+ return (
238
+ <Tr key={id}>
239
+ <Td
240
+ select={{
241
+ disable: false,
242
+ isSelected: isSelected(id),
243
+ onSelect: (_event, selected) => selectOne(selected, id, pkg),
244
+ rowIndex,
245
+ variant: 'checkbox',
246
+ }}
247
+ />
248
+ <Td>
249
+ {rpmId
250
+ ? <a href={urlBuilder(`packages/${rpmId}`, '')}>{packageName}</a>
251
+ : packageName
252
+ }
253
+ </Td>
254
+ <Td>
255
+ {version}
256
+ </Td>
257
+ </Tr>
258
+ );
259
+ })
260
+ }
261
+ </Tbody>
262
+ </TableWrapper>
263
+ </Modal>
264
+ );
265
+ };
266
+
267
+ PackageInstallModal.propTypes = {
268
+ isOpen: PropTypes.bool.isRequired,
269
+ closeModal: PropTypes.func.isRequired,
270
+ hostId: PropTypes.number.isRequired,
271
+ hostName: PropTypes.string.isRequired,
272
+ showKatelloAgent: PropTypes.bool,
273
+ };
274
+
275
+ PackageInstallModal.defaultProps = {
276
+ showKatelloAgent: false,
277
+ };
278
+
279
+ export default PackageInstallModal;
@@ -0,0 +1,3 @@
1
+ #package-install-modal .pf-c-input-group input {
2
+ min-width: 15em;
3
+ }
@@ -1,24 +1,51 @@
1
1
  import React, { useCallback, useState } from 'react';
2
2
  import { useSelector } from 'react-redux';
3
- import { Button, Hint, HintBody } from '@patternfly/react-core';
3
+ import {
4
+ Button,
5
+ Hint,
6
+ HintBody,
7
+ DropdownItem,
8
+ DropdownSeparator,
9
+ Dropdown,
10
+ Skeleton,
11
+ Split,
12
+ SplitItem,
13
+ ActionList,
14
+ ActionListItem,
15
+ KebabToggle,
16
+ } from '@patternfly/react-core';
4
17
  import { TableVariant, Thead, Tbody, Tr, Th, Td } from '@patternfly/react-table';
5
18
  import { translate as __ } from 'foremanReact/common/I18n';
6
19
  import { selectAPIResponse } from 'foremanReact/redux/API/APISelectors';
7
20
 
8
21
  import { urlBuilder } from 'foremanReact/common/urlHelpers';
22
+ import SelectableDropdown from '../../../SelectableDropdown';
9
23
  import TableWrapper from '../../../../components/Table/TableWrapper';
24
+ import { useUrlParams } from '../../../../components/Table/TableHooks';
10
25
  import { PackagesStatus, PackagesLatestVersion } from '../../../../components/Packages';
11
26
  import { getInstalledPackagesWithLatest } from '../HostPackages/HostPackagesActions';
12
27
  import { selectHostPackagesStatus } from '../HostPackages/HostPackagesSelectors';
13
- import { HOST_PACKAGES_KEY } from '../HostPackages/HostPackagesConstants';
28
+ import { HOST_PACKAGES_KEY, PACKAGES_VERSION_STATUSES, VERSION_STATUSES_TO_PARAM } from '../HostPackages/HostPackagesConstants';
14
29
  import './PackagesTab.scss';
30
+ import hostIdNotReady from '../HostDetailsActions';
31
+ import PackageInstallModal from './PackageInstallModal';
32
+ import defaultRemoteActionMethod, { KATELLO_AGENT } from '../hostDetailsHelpers';
15
33
 
16
34
  export const PackagesTab = () => {
17
35
  const hostDetails = useSelector(state => selectAPIResponse(state, 'HOST_DETAILS'));
18
- const { id: hostId } = hostDetails;
19
- const actionButtons = <Button isDisabled> {__('Upgrade')} </Button>;
36
+ const { id: hostId, name: hostName } = hostDetails;
20
37
 
21
- const [searchQuery, updateSearchQuery] = useState('');
38
+ const { searchParam } = useUrlParams();
39
+ const [searchQuery, updateSearchQuery] = useState(searchParam || '');
40
+ const PACKAGE_STATUS = __('Status');
41
+ const [packageStatusSelected, setPackageStatusSelected] = useState(PACKAGE_STATUS);
42
+ const activeFilters = [packageStatusSelected];
43
+ const defaultFilters = [PACKAGE_STATUS];
44
+ const [isBulkActionOpen, setIsBulkActionOpen] = useState(false);
45
+ const toggleBulkAction = () => setIsBulkActionOpen(prev => !prev);
46
+ const [isModalOpen, setIsModalOpen] = useState(false);
47
+ const closeModal = () => setIsModalOpen(false);
48
+ const showKatelloAgent = (defaultRemoteActionMethod({ hostDetails }) === KATELLO_AGENT);
22
49
 
23
50
  const emptyContentTitle = __('This host does not have any packages.');
24
51
  const emptyContentBody = __('Packages will appear here when available.');
@@ -32,13 +59,27 @@ export const PackagesTab = () => {
32
59
  ];
33
60
 
34
61
  const fetchItems = useCallback(
35
- params => (hostId ? getInstalledPackagesWithLatest(hostId, params) : null),
36
- [hostId],
62
+ (params) => {
63
+ if (!hostId) return hostIdNotReady;
64
+ const modifiedParams = { ...params };
65
+ if (packageStatusSelected !== PACKAGE_STATUS) {
66
+ modifiedParams.status = VERSION_STATUSES_TO_PARAM[packageStatusSelected];
67
+ }
68
+ return getInstalledPackagesWithLatest(hostId, modifiedParams);
69
+ },
70
+ [hostId, PACKAGE_STATUS, packageStatusSelected],
37
71
  );
38
72
 
39
73
  const response = useSelector(state => selectAPIResponse(state, HOST_PACKAGES_KEY));
40
74
  const { results, ...metadata } = response;
41
75
  const status = useSelector(state => selectHostPackagesStatus(state));
76
+
77
+ if (!hostId) return <Skeleton />;
78
+ const handleInstallPackagesClick = () => {
79
+ setIsBulkActionOpen(false);
80
+ setIsModalOpen(true);
81
+ };
82
+
42
83
  const rowActions = [
43
84
  {
44
85
  title: __('Upgrade via remote execution'), disabled: true,
@@ -48,6 +89,70 @@ export const PackagesTab = () => {
48
89
  },
49
90
  ];
50
91
 
92
+ const handlePackageStatusSelected = newStatus => setPackageStatusSelected((prevStatus) => {
93
+ if (prevStatus === newStatus) {
94
+ return PACKAGE_STATUS;
95
+ }
96
+ return newStatus;
97
+ });
98
+
99
+ const toggleGroup = (
100
+ <Split hasGutter>
101
+ <SplitItem>
102
+ <SelectableDropdown
103
+ id="package-status-dropdown"
104
+ title={PACKAGE_STATUS}
105
+ showTitle={false}
106
+ items={Object.values(PACKAGES_VERSION_STATUSES)}
107
+ selected={packageStatusSelected}
108
+ setSelected={handlePackageStatusSelected}
109
+ />
110
+ </SplitItem>
111
+ </Split>
112
+ );
113
+
114
+ const dropdownItems = [
115
+ <DropdownItem
116
+ aria-label="remove_pkg_from_host"
117
+ key="remove_pkg_from_host"
118
+ component="button"
119
+ isDisabled
120
+ >
121
+ {__('Remove')}
122
+ </DropdownItem>,
123
+ <DropdownSeparator key="separator" />,
124
+ <DropdownItem
125
+ aria-label="install_pkg_on_host"
126
+ key="install_pkg_on_host"
127
+ component="button"
128
+ onClick={handleInstallPackagesClick}
129
+ >
130
+ {__('Install packages')}
131
+ </DropdownItem>,
132
+ ];
133
+
134
+ const actionButtons = (
135
+ <>
136
+ <Split hasGutter>
137
+ <SplitItem>
138
+ <ActionList isIconList>
139
+ <ActionListItem>
140
+ <Button isDisabled> {__('Upgrade')} </Button>
141
+ </ActionListItem>
142
+ <ActionListItem>
143
+ <Dropdown
144
+ toggle={<KebabToggle aria-label="Packages bulk actions" onToggle={toggleBulkAction} />}
145
+ isOpen={isBulkActionOpen}
146
+ isPlain
147
+ dropdownItems={dropdownItems}
148
+ />
149
+ </ActionListItem>
150
+ </ActionList>
151
+ </SplitItem>
152
+ </Split>
153
+ </>
154
+ );
155
+
51
156
  return (
52
157
  <div>
53
158
  <div id="packages-hint" className="margin-0-24 margin-top-16">
@@ -70,12 +175,15 @@ export const PackagesTab = () => {
70
175
  emptySearchTitle,
71
176
  emptySearchBody,
72
177
  status,
178
+ activeFilters,
179
+ defaultFilters,
73
180
  actionButtons,
74
181
  searchQuery,
75
182
  updateSearchQuery,
183
+ toggleGroup,
76
184
  }
77
185
  }
78
- additionalListeners={[hostId]}
186
+ additionalListeners={[hostId, packageStatusSelected]}
79
187
  fetchItems={fetchItems}
80
188
  autocompleteEndpoint={`/hosts/${hostId}/packages/auto_complete_search`}
81
189
  foremanApiAutoComplete
@@ -120,6 +228,15 @@ export const PackagesTab = () => {
120
228
  </Tbody>
121
229
  </TableWrapper>
122
230
  </div>
231
+ {hostId &&
232
+ <PackageInstallModal
233
+ isOpen={isModalOpen}
234
+ closeModal={closeModal}
235
+ hostId={hostId}
236
+ hostName={hostName}
237
+ showKatelloAgent={showKatelloAgent}
238
+ />
239
+ }
123
240
  </div>
124
241
  );
125
242
  };
@@ -3,8 +3,9 @@ import { REX_JOB_INVOCATIONS_KEY, REX_FEATURES } from './RemoteExecutionConstant
3
3
  import { foremanApi } from '../../../../services/api';
4
4
  import { getResponseErrorMsgs } from '../../../../utils/helpers';
5
5
  import { renderTaskStartedToast } from '../../../../scenes/Tasks/helpers';
6
- import { ERRATA_SEARCH_QUERY } from '../HostErrata/HostErrataConstants';
7
- import { TRACES_SEARCH_QUERY } from './HostTracesConstants';
6
+ import { ERRATA_SEARCH_QUERY } from './ErrataTab/HostErrataConstants';
7
+ import { TRACES_SEARCH_QUERY } from './TracesTab/HostTracesConstants';
8
+ import { PACKAGE_SEARCH_QUERY } from '../YumInstallablePackages/YumInstallablePackagesConstants';
8
9
 
9
10
  const errorToast = (error) => {
10
11
  const message = getResponseErrorMsgs(error.response);
@@ -19,6 +20,7 @@ const baseParams = ({ feature, hostname, inputs = {} }) => ({
19
20
  },
20
21
  });
21
22
 
23
+ // used when we know the package name
22
24
  const katelloPackageInstallParams = ({ hostname, packageName }) =>
23
25
  baseParams({
24
26
  hostname,
@@ -26,6 +28,15 @@ const katelloPackageInstallParams = ({ hostname, packageName }) =>
26
28
  feature: REX_FEATURES.KATELLO_PACKAGE_INSTALL,
27
29
  });
28
30
 
31
+ // used when we know package Id(s)
32
+ const katelloPackageInstallBySearchParams = ({ hostname, search }) =>
33
+ baseParams({
34
+ hostname,
35
+ inputs: { [PACKAGE_SEARCH_QUERY]: search },
36
+ feature: REX_FEATURES.KATELLO_PACKAGE_INSTALL_BY_SEARCH,
37
+ });
38
+
39
+
29
40
  const katelloTracerResolveParams = ({ hostname, search }) =>
30
41
  baseParams({
31
42
  hostname,
@@ -38,7 +49,7 @@ const katelloHostErrataInstallParams = ({
38
49
  }) => baseParams({
39
50
  hostname,
40
51
  inputs: { [ERRATA_SEARCH_QUERY]: search },
41
- feature: REX_FEATURES.KATELLO_HOST_ERRATA_INSTALL,
52
+ feature: REX_FEATURES.KATELLO_HOST_ERRATA_INSTALL_BY_SEARCH,
42
53
  });
43
54
 
44
55
  export const installPackage = ({ hostname, packageName }) => post({
@@ -53,6 +64,18 @@ export const installPackage = ({ hostname, packageName }) => post({
53
64
  errorToast: error => errorToast(error),
54
65
  });
55
66
 
67
+ export const installPackageBySearch = ({ hostname, search }) => post({
68
+ type: API_OPERATIONS.POST,
69
+ key: REX_JOB_INVOCATIONS_KEY,
70
+ url: foremanApi.getApiUrl('/job_invocations'),
71
+ params: katelloPackageInstallBySearchParams({ hostname, search }),
72
+ handleSuccess: response => renderTaskStartedToast({
73
+ humanized: { action: `Install packages on ${hostname}` },
74
+ id: response?.data?.dynflow_task?.id,
75
+ }),
76
+ errorToast: error => errorToast(error),
77
+ });
78
+
56
79
  export const resolveTraces = ({ hostname, search }) => post({
57
80
  type: API_OPERATIONS.POST,
58
81
  key: REX_JOB_INVOCATIONS_KEY,
@@ -75,7 +98,7 @@ export const installErrata = ({
75
98
  hostname, search,
76
99
  }),
77
100
  handleSuccess: response => renderTaskStartedToast({
78
- humanized: { action: `Install Errata on ${hostname}` },
101
+ humanized: { action: `Install errata on ${hostname}` },
79
102
  id: response?.data?.dynflow_task?.id,
80
103
  }),
81
104
  errorToast: error => errorToast(error),
@@ -1,6 +1,7 @@
1
1
  export const REX_JOB_INVOCATIONS_KEY = 'REX_JOB_INVOCATIONS';
2
2
  export const REX_FEATURES = {
3
3
  KATELLO_PACKAGE_INSTALL: 'katello_package_install',
4
+ KATELLO_PACKAGE_INSTALL_BY_SEARCH: 'katello_package_install_by_search',
4
5
  KATELLO_HOST_TRACER_RESOLVE: 'katello_host_tracer_resolve',
5
- KATELLO_HOST_ERRATA_INSTALL: 'katello_errata_install',
6
+ KATELLO_HOST_ERRATA_INSTALL_BY_SEARCH: 'katello_errata_install_by_search',
6
7
  };
@@ -20,9 +20,10 @@ import TableWrapper from '../../../../../components/Table/TableWrapper';
20
20
  import { enableRepoSetRepo, disableRepoSetRepo, resetRepoSetRepo, getHostRepositorySets } from './RepositorySetsActions';
21
21
  import { selectRepositorySetsStatus } from './RepositorySetsSelectors';
22
22
  import { selectHostDetailsStatus } from '../../HostDetailsSelectors.js';
23
- import { useBulkSelect } from '../../../../../components/Table/TableHooks';
23
+ import { useBulkSelect, useUrlParams } from '../../../../../components/Table/TableHooks';
24
24
  import { REPOSITORY_SETS_KEY } from './RepositorySetsConstants.js';
25
25
  import './RepositorySetsTab.scss';
26
+ import hostIdNotReady from '../../HostDetailsActions';
26
27
 
27
28
  const getEnabledValue = ({ enabled, enabledContentOverride }) => {
28
29
  const isOverridden = (enabledContentOverride !== null);
@@ -78,11 +79,12 @@ const RepositorySetsTab = () => {
78
79
  lifecycleEnvironmentLibrary === false;
79
80
  const simpleContentAccess = (Number(subscriptionStatus) === 5);
80
81
  const dispatch = useDispatch();
82
+ const { searchParam, show } = useUrlParams();
81
83
  const toggleGroupStates = ['noLimit', 'limitToEnvironment'];
82
84
  const [noLimit, limitToEnvironment] = toggleGroupStates;
83
85
  const defaultToggleGroupState = nonLibraryHost ? limitToEnvironment : noLimit;
84
86
  const [toggleGroupState, setToggleGroupState] =
85
- useState(defaultToggleGroupState);
87
+ useState(show ?? defaultToggleGroupState);
86
88
  const [alertShowing, setAlertShowing] = useState(false);
87
89
  const emptyContentTitle = __('No repository sets to show.');
88
90
  const emptyContentBody = __('Repository sets will appear here when available.');
@@ -101,7 +103,7 @@ const RepositorySetsTab = () => {
101
103
  content_access_mode_all: simpleContentAccess,
102
104
  host_id: hostId,
103
105
  ...params,
104
- }) : null),
106
+ }) : hostIdNotReady),
105
107
  [hostId, toggleGroupState, limitToEnvironment, simpleContentAccess],
106
108
  );
107
109
 
@@ -115,6 +117,7 @@ const RepositorySetsTab = () => {
115
117
  results,
116
118
  metadata,
117
119
  isSelectable: () => false,
120
+ initialSearchQuery: searchParam || '',
118
121
  });
119
122
 
120
123
  const hostDetailsStatus = useSelector(state => selectHostDetailsStatus(state));
@@ -15,7 +15,8 @@ import { translate as __ } from 'foremanReact/common/I18n';
15
15
  import { useSelector, useDispatch } from 'react-redux';
16
16
  import { selectAPIResponse } from 'foremanReact/redux/API/APISelectors';
17
17
  import { installTracerPackage } from './HostTracesActions';
18
- import { katelloPackageInstallUrl } from './customizedRexUrlHelpers';
18
+ import { katelloPackageInstallUrl } from '../customizedRexUrlHelpers';
19
+ import { KATELLO_TRACER_PACKAGE } from './HostTracesConstants';
19
20
 
20
21
  const EnableTracerModal = ({ isOpen, setIsOpen }) => {
21
22
  const title = __('Enable Tracer');
@@ -43,7 +44,7 @@ const EnableTracerModal = ({ isOpen, setIsOpen }) => {
43
44
  <DropdownItem key={`option_${text}`} onClick={() => setSelectedOption(text)}>{text}</DropdownItem>
44
45
  ));
45
46
 
46
- const customizedRexUrl = katelloPackageInstallUrl({ hostname });
47
+ const customizedRexUrl = katelloPackageInstallUrl({ hostname, packages: KATELLO_TRACER_PACKAGE });
47
48
 
48
49
  const getEnableTracerButton = () => {
49
50
  const [viaRex] = dropdownOptions;
@@ -3,8 +3,8 @@ import {
3
3
  HOST_TRACES_KEY,
4
4
  KATELLO_TRACER_PACKAGE,
5
5
  } from './HostTracesConstants';
6
- import { installPackage } from './RemoteExecutionActions';
7
- import { foremanApi } from '../../../../services/api';
6
+ import { installPackage } from '../RemoteExecutionActions';
7
+ import { foremanApi } from '../../../../../services/api';
8
8
 
9
9
  export const getHostTraces = (hostId, params) => get({
10
10
  type: API_OPERATIONS.GET,
@@ -8,13 +8,14 @@ import { TableVariant, Thead, Tbody, Tr, Th, Td } from '@patternfly/react-table'
8
8
  import { useSelector, useDispatch } from 'react-redux';
9
9
  import { selectAPIResponse } from 'foremanReact/redux/API/APISelectors';
10
10
  import EnableTracerEmptyState from './EnableTracerEmptyState';
11
- import TableWrapper from '../../../Table/TableWrapper';
12
- import { useBulkSelect } from '../../../Table/TableHooks';
11
+ import TableWrapper from '../../../../Table/TableWrapper';
12
+ import { useBulkSelect, useUrlParams } from '../../../../Table/TableHooks';
13
13
  import { getHostTraces } from './HostTracesActions';
14
- import { resolveTraces } from './RemoteExecutionActions';
14
+ import { resolveTraces } from '../RemoteExecutionActions';
15
15
  import { selectHostTracesStatus } from './HostTracesSelectors';
16
- import { resolveTraceUrl } from './customizedRexUrlHelpers';
16
+ import { resolveTraceUrl } from '../customizedRexUrlHelpers';
17
17
  import './TracesTab.scss';
18
+ import hostIdNotReady from '../../HostDetailsActions';
18
19
 
19
20
  const TracesTab = () => {
20
21
  const hostDetails = useSelector(state => selectAPIResponse(state, 'HOST_DETAILS'));
@@ -31,9 +32,10 @@ const TracesTab = () => {
31
32
  const emptySearchBody = __('Try changing your search settings.');
32
33
  const fetchItems = useCallback(
33
34
  params =>
34
- (hostId ? getHostTraces(hostId, params) : null),
35
+ (hostId ? getHostTraces(hostId, params) : hostIdNotReady),
35
36
  [hostId],
36
37
  );
38
+ const { searchParam } = useUrlParams();
37
39
  const [isBulkActionOpen, setIsBulkActionOpen] = useState(false);
38
40
  const toggleBulkAction = () => setIsBulkActionOpen(prev => !prev);
39
41
  const response = useSelector(state => selectAPIResponse(state, 'HOST_TRACES'));
@@ -46,6 +48,7 @@ const TracesTab = () => {
46
48
  results,
47
49
  metadata: meta,
48
50
  isSelectable: result => !!result.restart_command,
51
+ initialSearchQuery: searchParam || '',
49
52
  });
50
53
 
51
54