katello 4.12.0.rc3 → 4.13.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 (244) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +0 -1
  3. data/app/assets/javascripts/katello/locale/bn/katello.js +3365 -3350
  4. data/app/assets/javascripts/katello/locale/bn_IN/katello.js +3136 -3121
  5. data/app/assets/javascripts/katello/locale/ca/katello.js +3588 -3576
  6. data/app/assets/javascripts/katello/locale/cs/katello.js +3499 -3487
  7. data/app/assets/javascripts/katello/locale/cs_CZ/katello.js +4186 -4186
  8. data/app/assets/javascripts/katello/locale/de/katello.js +5553 -5562
  9. data/app/assets/javascripts/katello/locale/de_AT/katello.js +3008 -2993
  10. data/app/assets/javascripts/katello/locale/de_DE/katello.js +3066 -3051
  11. data/app/assets/javascripts/katello/locale/el/katello.js +3376 -3370
  12. data/app/assets/javascripts/katello/locale/en/katello.js +3008 -2993
  13. data/app/assets/javascripts/katello/locale/en_GB/katello.js +3076 -3073
  14. data/app/assets/javascripts/katello/locale/en_US/katello.js +3008 -2993
  15. data/app/assets/javascripts/katello/locale/es/katello.js +5366 -5372
  16. data/app/assets/javascripts/katello/locale/et_EE/katello.js +3008 -2993
  17. data/app/assets/javascripts/katello/locale/fr/katello.js +5975 -5984
  18. data/app/assets/javascripts/katello/locale/gl/katello.js +3125 -3113
  19. data/app/assets/javascripts/katello/locale/gu/katello.js +3119 -3104
  20. data/app/assets/javascripts/katello/locale/he_IL/katello.js +3020 -3005
  21. data/app/assets/javascripts/katello/locale/hi/katello.js +3137 -3122
  22. data/app/assets/javascripts/katello/locale/id/katello.js +3008 -2993
  23. data/app/assets/javascripts/katello/locale/it/katello.js +4469 -4466
  24. data/app/assets/javascripts/katello/locale/ja/katello.js +5969 -5978
  25. data/app/assets/javascripts/katello/locale/ka/katello.js +5649 -5652
  26. data/app/assets/javascripts/katello/locale/kn/katello.js +3136 -3121
  27. data/app/assets/javascripts/katello/locale/ko/katello.js +4717 -4720
  28. data/app/assets/javascripts/katello/locale/locale/katello.js +1050 -1084
  29. data/app/assets/javascripts/katello/locale/ml_IN/katello.js +3008 -2993
  30. data/app/assets/javascripts/katello/locale/mr/katello.js +3136 -3121
  31. data/app/assets/javascripts/katello/locale/nl_NL/katello.js +3116 -3101
  32. data/app/assets/javascripts/katello/locale/or/katello.js +3137 -3122
  33. data/app/assets/javascripts/katello/locale/pa/katello.js +3136 -3121
  34. data/app/assets/javascripts/katello/locale/pl/katello.js +3210 -3195
  35. data/app/assets/javascripts/katello/locale/pl_PL/katello.js +3008 -2993
  36. data/app/assets/javascripts/katello/locale/pt/katello.js +3009 -2994
  37. data/app/assets/javascripts/katello/locale/pt_BR/katello.js +5362 -5368
  38. data/app/assets/javascripts/katello/locale/ro/katello.js +3008 -2993
  39. data/app/assets/javascripts/katello/locale/ro_RO/katello.js +3008 -2993
  40. data/app/assets/javascripts/katello/locale/ru/katello.js +4638 -4641
  41. data/app/assets/javascripts/katello/locale/sl/katello.js +3051 -3036
  42. data/app/assets/javascripts/katello/locale/sv_SE/katello.js +3156 -3144
  43. data/app/assets/javascripts/katello/locale/ta/katello.js +3365 -3350
  44. data/app/assets/javascripts/katello/locale/ta_IN/katello.js +3121 -3106
  45. data/app/assets/javascripts/katello/locale/te/katello.js +3136 -3121
  46. data/app/assets/javascripts/katello/locale/tr/katello.js +3025 -3010
  47. data/app/assets/javascripts/katello/locale/vi/katello.js +3008 -2993
  48. data/app/assets/javascripts/katello/locale/vi_VN/katello.js +3008 -2993
  49. data/app/assets/javascripts/katello/locale/zh/katello.js +3008 -2993
  50. data/app/assets/javascripts/katello/locale/zh_CN/katello.js +5968 -5977
  51. data/app/assets/javascripts/katello/locale/zh_TW/katello.js +4694 -4697
  52. data/app/assets/javascripts/katello/sync_management/sync_management.js +1 -0
  53. data/app/controllers/katello/api/registry/registry_proxies_controller.rb +51 -124
  54. data/app/controllers/katello/api/rhsm/candlepin_dynflow_proxy_controller.rb +12 -20
  55. data/app/controllers/katello/api/v2/activation_keys_controller.rb +10 -4
  56. data/app/controllers/katello/api/v2/capsule_content_controller.rb +24 -0
  57. data/app/controllers/katello/api/v2/content_view_versions_controller.rb +9 -2
  58. data/app/controllers/katello/api/v2/debs_controller.rb +1 -1
  59. data/app/controllers/katello/api/v2/errata_controller.rb +1 -1
  60. data/app/controllers/katello/api/v2/host_subscriptions_controller.rb +12 -4
  61. data/app/controllers/katello/api/v2/hosts_bulk_actions_controller.rb +3 -3
  62. data/app/controllers/katello/api/v2/organizations_controller.rb +0 -11
  63. data/app/controllers/katello/api/v2/packages_controller.rb +1 -1
  64. data/app/controllers/katello/api/v2/products_bulk_actions_controller.rb +1 -1
  65. data/app/controllers/katello/api/v2/repositories_controller.rb +18 -12
  66. data/app/controllers/katello/api/v2/repository_sets_controller.rb +2 -1
  67. data/app/controllers/katello/api/v2/simple_content_access_controller.rb +9 -22
  68. data/app/controllers/katello/concerns/api/v2/authorization.rb +1 -1
  69. data/app/helpers/katello/concerns/dashboard_helper_extensions.rb +0 -10
  70. data/app/helpers/katello/hosts_and_hostgroups_helper.rb +14 -2
  71. data/app/helpers/katello/katello_urls_helper.rb +26 -1
  72. data/app/helpers/katello/subscription_mailer_helper.rb +1 -1
  73. data/app/jobs/create_manifest_expire_soon_warning_notifications.rb +11 -0
  74. data/app/lib/actions/candlepin/owner/regenerate_upstream_identity_cert.rb +21 -0
  75. data/app/lib/actions/katello/capsule_content/sync.rb +1 -1
  76. data/app/lib/actions/katello/capsule_content/verify_checksum.rb +75 -0
  77. data/app/lib/actions/katello/content_view/promote.rb +1 -1
  78. data/app/lib/actions/katello/content_view/publish.rb +1 -1
  79. data/app/lib/actions/katello/content_view_version/verify_checksum.rb +29 -0
  80. data/app/lib/actions/katello/host/hypervisors_update.rb +1 -0
  81. data/app/lib/actions/katello/host/update_content_view.rb +2 -2
  82. data/app/lib/actions/katello/organization/manifest_import.rb +5 -0
  83. data/app/lib/actions/katello/organization/manifest_refresh.rb +3 -0
  84. data/app/lib/actions/katello/repository/metadata_generate.rb +7 -1
  85. data/app/lib/actions/katello/repository/remove_content.rb +1 -0
  86. data/app/lib/actions/katello/repository/sync.rb +2 -1
  87. data/app/lib/actions/katello/repository/upload_files.rb +1 -0
  88. data/app/lib/actions/pulp3/capsule_content/verify_checksum.rb +27 -0
  89. data/app/lib/actions/pulp3/orchestration/content_view_version/export_repository.rb +7 -9
  90. data/app/lib/actions/pulp3/orchestration/content_view_version/syncable_export.rb +5 -4
  91. data/app/lib/katello/concerns/base_template_scope_extensions.rb +7 -2
  92. data/app/lib/katello/errors.rb +4 -0
  93. data/app/lib/katello/http_resource.rb +6 -1
  94. data/app/lib/katello/resources/candlepin/consumer.rb +1 -1
  95. data/app/lib/katello/resources/candlepin/upstream_consumer.rb +18 -6
  96. data/app/lib/katello/resources/candlepin/upstream_job.rb +1 -1
  97. data/app/lib/katello/resources/cdn.rb +4 -13
  98. data/app/lib/katello/resources/registry.rb +25 -0
  99. data/app/mailers/katello/subscription_mailer.rb +3 -6
  100. data/app/models/katello/candlepin/repository_mapper.rb +1 -1
  101. data/app/models/katello/concerns/organization_extensions.rb +42 -3
  102. data/app/models/katello/content_view.rb +28 -0
  103. data/app/models/katello/content_view_environment_content_facet.rb +4 -2
  104. data/app/models/katello/glue/provider.rb +19 -12
  105. data/app/models/katello/glue/pulp/repos.rb +3 -8
  106. data/app/models/katello/host/content_facet.rb +1 -1
  107. data/app/models/katello/host/subscription_facet.rb +1 -1
  108. data/app/models/katello/host_collection.rb +12 -3
  109. data/app/models/katello/ping.rb +1 -1
  110. data/app/models/katello/repository.rb +33 -0
  111. data/app/models/katello/root_repository.rb +0 -4
  112. data/app/services/katello/content_unit_indexer.rb +9 -0
  113. data/app/services/katello/pulp3/alternate_content_source.rb +6 -8
  114. data/app/services/katello/pulp3/api/core.rb +13 -0
  115. data/app/services/katello/pulp3/api/yum.rb +11 -0
  116. data/app/services/katello/pulp3/docker_manifest.rb +5 -1
  117. data/app/services/katello/pulp3/repository/generic.rb +1 -1
  118. data/app/services/katello/pulp3/repository.rb +26 -6
  119. data/app/services/katello/pulp3/repository_mirror.rb +13 -12
  120. data/app/services/katello/pulp3/service_common.rb +2 -10
  121. data/app/services/katello/pulp3/smart_proxy_repository.rb +0 -2
  122. data/app/services/katello/ui_notifications/subscriptions/manifest_expire_soon_warning.rb +75 -0
  123. data/app/views/foreman/job_templates/update_package_-_katello_ansible_default.erb +5 -1
  124. data/app/views/foreman/job_templates/update_packages_by_search_query_-_katello_ansible_default.erb +2 -2
  125. data/app/views/foreman/job_templates/upload_profile.erb +16 -0
  126. data/app/views/katello/api/v2/content_view_filter_rules/show.json.rabl +9 -0
  127. data/app/views/katello/api/v2/docker_manifests/show.json.rabl +1 -0
  128. data/app/views/katello/api/v2/hosts/host_collections.json.rabl +5 -1
  129. data/app/views/katello/api/v2/organizations/show.json.rabl +9 -1
  130. data/app/views/katello/hosts/_errata_counts.html.erb +1 -1
  131. data/app/views/overrides/activation_keys/_host_environment_select.html.erb +1 -1
  132. data/app/views/overrides/activation_keys/_host_media_type_select.html.erb +15 -5
  133. data/app/views/overrides/activation_keys/_host_tab_pane.html.erb +1 -29
  134. data/config/routes/api/registry.rb +4 -8
  135. data/config/routes/api/v2.rb +2 -0
  136. data/db/migrate/20240423112842_add_fields_to_katello_docker_manifest.rb +8 -0
  137. data/db/migrate/20240502192021_change_katello_repository_rpms_id_seq_to_big_int.rb +9 -0
  138. data/db/seeds.d/109-katello-notification-blueprints.rb +6 -0
  139. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/activation-keys/details/activation-key-repository-sets.controller.js +3 -3
  140. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/content-credentials/new/views/new-content-credential.html +2 -1
  141. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/content-hosts/details/content-host-repository-sets.controller.js +3 -3
  142. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/i18n/bastion_katello.pot +0 -15
  143. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/products/details/repositories/details/views/repository-info.html +8 -6
  144. data/engines/bastion_katello/app/assets/javascripts/bastion_katello/products/details/repositories/new/views/new-repository.html +12 -10
  145. data/lib/katello/permission_creator.rb +3 -3
  146. data/lib/katello/permissions/registry_permissions.rb +4 -7
  147. data/lib/katello/plugin.rb +10 -16
  148. data/lib/katello/repository_types/ostree.rb +7 -0
  149. data/lib/katello/scheduled_jobs.rb +7 -1
  150. data/lib/katello/tasks/clean_backend_objects.rake +1 -1
  151. data/lib/katello/tasks/repository.rake +22 -0
  152. data/lib/katello/version.rb +1 -1
  153. data/locale/action_names.rb +4 -3
  154. data/locale/bn/katello.po +166 -151
  155. data/locale/bn_IN/katello.po +166 -151
  156. data/locale/ca/katello.po +166 -151
  157. data/locale/cs/katello.po +166 -151
  158. data/locale/cs_CZ/LC_MESSAGES/katello.mo +0 -0
  159. data/locale/cs_CZ/katello.po +172 -157
  160. data/locale/de/LC_MESSAGES/katello.mo +0 -0
  161. data/locale/de/katello.po +178 -163
  162. data/locale/de_AT/katello.po +166 -151
  163. data/locale/de_DE/katello.po +166 -151
  164. data/locale/el/katello.po +166 -151
  165. data/locale/en/katello.po +166 -151
  166. data/locale/en_GB/katello.po +166 -151
  167. data/locale/en_US/katello.po +166 -151
  168. data/locale/es/LC_MESSAGES/katello.mo +0 -0
  169. data/locale/es/katello.po +178 -163
  170. data/locale/et_EE/katello.po +166 -151
  171. data/locale/fr/LC_MESSAGES/katello.mo +0 -0
  172. data/locale/fr/katello.po +179 -164
  173. data/locale/gl/katello.po +166 -151
  174. data/locale/gu/katello.po +166 -151
  175. data/locale/he_IL/katello.po +166 -151
  176. data/locale/hi/katello.po +166 -151
  177. data/locale/id/katello.po +166 -151
  178. data/locale/it/LC_MESSAGES/katello.mo +0 -0
  179. data/locale/it/katello.po +169 -154
  180. data/locale/ja/LC_MESSAGES/katello.mo +0 -0
  181. data/locale/ja/katello.po +179 -164
  182. data/locale/ka/LC_MESSAGES/katello.mo +0 -0
  183. data/locale/ka/katello.po +177 -162
  184. data/locale/katello.pot +1119 -1062
  185. data/locale/kn/katello.po +166 -151
  186. data/locale/ko/LC_MESSAGES/katello.mo +0 -0
  187. data/locale/ko/katello.po +174 -159
  188. data/locale/ml_IN/katello.po +166 -151
  189. data/locale/mr/katello.po +166 -151
  190. data/locale/nl_NL/katello.po +166 -151
  191. data/locale/or/katello.po +166 -151
  192. data/locale/pa/katello.po +166 -151
  193. data/locale/pl/katello.po +166 -151
  194. data/locale/pl_PL/katello.po +166 -151
  195. data/locale/pt/katello.po +166 -151
  196. data/locale/pt_BR/LC_MESSAGES/katello.mo +0 -0
  197. data/locale/pt_BR/katello.po +178 -163
  198. data/locale/ro/katello.po +166 -151
  199. data/locale/ro_RO/katello.po +166 -151
  200. data/locale/ru/LC_MESSAGES/katello.mo +0 -0
  201. data/locale/ru/katello.po +171 -156
  202. data/locale/sl/katello.po +166 -151
  203. data/locale/sv_SE/katello.po +166 -151
  204. data/locale/ta/katello.po +166 -151
  205. data/locale/ta_IN/katello.po +166 -151
  206. data/locale/te/katello.po +166 -151
  207. data/locale/tr/katello.po +166 -151
  208. data/locale/vi/katello.po +166 -151
  209. data/locale/vi_VN/katello.po +166 -151
  210. data/locale/zh/katello.po +166 -151
  211. data/locale/zh_CN/LC_MESSAGES/katello.mo +0 -0
  212. data/locale/zh_CN/katello.po +179 -164
  213. data/locale/zh_TW/LC_MESSAGES/katello.mo +0 -0
  214. data/locale/zh_TW/katello.po +171 -156
  215. data/webpack/ForemanColumnExtensions/index.js +129 -0
  216. data/webpack/components/ActivationKeysSearch/ActivationKeysSearch.test.js +28 -0
  217. data/webpack/components/ActivationKeysSearch/index.js +222 -0
  218. data/webpack/components/Table/TableWrapper.js +14 -0
  219. data/webpack/components/extensions/HostDetails/ActionsBar/index.js +1 -1
  220. data/webpack/components/extensions/HostDetails/Tabs/PackagesTab/PackagesTab.js +1 -1
  221. data/webpack/components/extensions/HostDetails/Tabs/__tests__/packageInstallModal.test.js +1 -0
  222. data/webpack/components/extensions/HostDetails/Tabs/__tests__/packagesTab.test.js +1 -0
  223. data/webpack/components/extensions/Hosts/ActionsBar/index.js +20 -1
  224. data/webpack/components/extensions/Hosts/BulkActions/BulkChangeHostCVModal/BulkChangeHostCVModal.js +220 -0
  225. data/webpack/components/extensions/Hosts/BulkActions/BulkChangeHostCVModal/actions.js +23 -0
  226. data/webpack/components/extensions/Hosts/BulkActions/BulkChangeHostCVModal/index.js +25 -0
  227. data/webpack/components/extensions/Hosts/BulkActions/__tests__/bulkChangeHostCVModal.test.js +133 -0
  228. data/webpack/global_index.js +19 -0
  229. data/webpack/scenes/ContentViews/Details/ComponentContentViews/ContentViewComponents.js +6 -3
  230. data/webpack/scenes/ContentViews/Publish/CVPublishForm.js +1 -1
  231. data/webpack/scenes/ContentViews/Publish/__tests__/publishContentView.test.js +30 -0
  232. data/webpack/scenes/Hosts/ChangeContentSource/actions.js +3 -1
  233. data/webpack/scenes/Hosts/ChangeContentSource/components/ContentSourceForm.js +63 -25
  234. data/webpack/scenes/Hosts/ChangeContentSource/index.js +24 -16
  235. data/webpack/scenes/RedHatRepositories/__tests__/__snapshots__/RedHatRepositoriesPage.test.js.snap +1 -0
  236. data/webpack/scenes/Subscriptions/Manifest/ManageManifestModal.js +64 -5
  237. data/webpack/scenes/Subscriptions/UpstreamSubscriptions/UpstreamSubscriptionsPage.js +16 -13
  238. data/webpack/scenes/Subscriptions/UpstreamSubscriptions/__tests__/__snapshots__/UpstreamSubscriptionsPage.test.js.snap +14 -8
  239. data/webpack/scenes/Subscriptions/__tests__/__snapshots__/SubscriptionsPage.test.js.snap +1 -0
  240. data/webpack/scenes/Subscriptions/components/SubscriptionsToolbar/SubscriptionsToolbar.js +1 -1
  241. metadata +59 -40
  242. data/app/assets/javascripts/katello/hosts/activation_key_edit.js +0 -167
  243. data/app/lib/actions/katello/host/upload_package_profile.rb +0 -45
  244. data/app/lib/actions/katello/host/upload_profiles.rb +0 -47
@@ -23,7 +23,7 @@ import { ENVIRONMENT_PATHS_KEY } from '../../../../scenes/ContentViews/component
23
23
  import EnvironmentPaths from '../../../../scenes/ContentViews/components/EnvironmentPaths/EnvironmentPaths';
24
24
  import ContentViewSelect from '../../../../scenes/ContentViews/components/ContentViewSelect/ContentViewSelect';
25
25
  import ContentViewSelectOption from '../../../../scenes/ContentViews/components/ContentViewSelect/ContentViewSelectOption';
26
- import { selectContentViewsStatus } from '../selectors';
26
+ import { selectContentViewsStatus, selectJobInvocationPath } from '../selectors';
27
27
  import { getCVPlaceholderText, shouldDisableCVSelect } from '../../../ContentViews/components/ContentViewSelect/helpers';
28
28
  import { selectEnvironmentPaths } from '../../../ContentViews/components/EnvironmentPaths/EnvironmentPathSelectors';
29
29
 
@@ -95,6 +95,9 @@ const ContentSourceForm = ({
95
95
  contentSources,
96
96
  handleContentSource,
97
97
  contentSourceId,
98
+ showCVOnlyAlert,
99
+ hostDetailsPath,
100
+ hostEditPath,
98
101
  contentHosts,
99
102
  isLoading,
100
103
  hostsUpdated,
@@ -108,13 +111,14 @@ const ContentSourceForm = ({
108
111
  );
109
112
  const contentViewsStatus = useSelector(selectContentViewsStatus);
110
113
  const environmentPathResponse = useSelector(selectEnvironmentPaths);
114
+ const jobInvocationPath = useSelector(selectJobInvocationPath);
111
115
  const envList = environmentPathResponse?.results?.map(path => path.environments).flat();
112
116
  const [csSelectOpen, setCSSelectOpen] = useState(false);
113
117
  const [cvSelectOpen, setCVSelectOpen] = useState(false);
114
118
  const hostCount = contentHosts.length;
115
119
 
116
120
  const handleCSSelect = (_event, selection) => {
117
- handleContentSource(selection);
121
+ handleContentSource(typeof selection === 'number' ? selection.toString() : selection);
118
122
  setCSSelectOpen(false);
119
123
  };
120
124
 
@@ -127,13 +131,12 @@ const ContentSourceForm = ({
127
131
  !!contentViewName &&
128
132
  !!contentSourceId &&
129
133
  hostCount !== 0);
130
-
131
134
  const contentSourcesIsDisabled = (isLoading || contentSources.length === 0 ||
132
135
  hostCount === 0);
133
- const environmentIsDisabled = (isLoading || environments === [] ||
136
+ const environmentIsDisabled = (isLoading ||
134
137
  contentSourceId === '');
135
138
  const viewIsDisabled = (isLoading || contentViews.length === 0 ||
136
- contentSourceId === '' || environments === []);
139
+ contentSourceId === '');
137
140
 
138
141
  const cvPlaceholderText = getCVPlaceholderText({
139
142
  contentSourceId,
@@ -229,6 +232,15 @@ const ContentSourceForm = ({
229
232
  env={environments[0]}
230
233
  />))}
231
234
  </ContentViewSelect>
235
+ {showCVOnlyAlert &&
236
+ <Alert
237
+ ouiaId="cv-only-alert"
238
+ variant="info"
239
+ className="margin-top-20"
240
+ title={__('Host content source will remain the same. Click Save below to update the host\'s content view environment.')}
241
+ />
242
+ }
243
+ {!showCVOnlyAlert &&
232
244
  <TextContent>
233
245
  <Text
234
246
  ouiaId="ccs-options-description"
@@ -252,27 +264,49 @@ const ContentSourceForm = ({
252
264
  />
253
265
  </Text>
254
266
  </TextContent>
267
+ }
255
268
  <ActionGroup style={{ display: 'block' }}>
256
- <Button
257
- variant="primary"
258
- id="generate_btn"
259
- ouiaId="update-source-button"
260
- onClick={e => handleSubmit(e, { shouldRedirect: true })}
261
- isDisabled={isLoading || !formIsValid() || hostsUpdated}
262
- isLoading={isLoading}
263
- >
264
- {__('Run job invocation')}
265
- </Button>
266
- <Button
267
- variant="secondary"
268
- id="generate_btn"
269
- ouiaId="update-source-button"
270
- onClick={showTemplate}
271
- isDisabled={isLoading || !formIsValid() || hostsUpdated}
272
- isLoading={isLoading}
273
- >
274
- {__('Update hosts manually')}
275
- </Button>
269
+ {!showCVOnlyAlert &&
270
+ <>
271
+ <Button
272
+ variant="primary"
273
+ id="generate_btn"
274
+ ouiaId="run-job-invocation-button"
275
+ onClick={e => handleSubmit(e, { redirectTo: jobInvocationPath })}
276
+ isDisabled={isLoading || !formIsValid() || hostsUpdated}
277
+ isLoading={isLoading}
278
+ >
279
+ {__('Run job invocation')}
280
+ </Button>
281
+ <Button
282
+ variant="secondary"
283
+ id="generate_btn"
284
+ ouiaId="update-source-button"
285
+ onClick={showTemplate}
286
+ isDisabled={isLoading || !formIsValid() || hostsUpdated}
287
+ isLoading={isLoading}
288
+ >
289
+ {__('Update hosts manually')}
290
+ </Button>
291
+ </>
292
+ }
293
+ {showCVOnlyAlert &&
294
+ <Button
295
+ variant="primary"
296
+ id="generate_btn"
297
+ ouiaId="change-cv-button"
298
+ onClick={e =>
299
+ handleSubmit(e, {
300
+ redirectTo: hostEditPath || hostDetailsPath,
301
+ showSuccessToast: true,
302
+ })
303
+ }
304
+ isDisabled={isLoading || !formIsValid() || hostsUpdated}
305
+ isLoading={isLoading}
306
+ >
307
+ {__('Save')}
308
+ </Button>
309
+ }
276
310
 
277
311
  </ActionGroup>
278
312
  </Form>);
@@ -288,6 +322,9 @@ ContentSourceForm.propTypes = {
288
322
  contentSources: PropTypes.arrayOf(PropTypes.shape({})),
289
323
  handleContentSource: PropTypes.func.isRequired,
290
324
  contentSourceId: PropTypes.string,
325
+ showCVOnlyAlert: PropTypes.bool,
326
+ hostDetailsPath: PropTypes.string.isRequired,
327
+ hostEditPath: PropTypes.string.isRequired,
291
328
  contentHosts: PropTypes.arrayOf(PropTypes.shape({})),
292
329
  isLoading: PropTypes.bool,
293
330
  hostsUpdated: PropTypes.bool,
@@ -300,6 +337,7 @@ ContentSourceForm.defaultProps = {
300
337
  contentViewName: '',
301
338
  contentSources: [],
302
339
  contentSourceId: '',
340
+ showCVOnlyAlert: false,
303
341
  contentHosts: [],
304
342
  isLoading: false,
305
343
  hostsUpdated: false,
@@ -17,7 +17,6 @@ import { selectApiDataStatus,
17
17
  selectContentHosts,
18
18
  selectContentHostsWithoutContent,
19
19
  selectContentSources,
20
- selectJobInvocationPath,
21
20
  selectContentViews,
22
21
  selectTemplate } from './selectors';
23
22
 
@@ -46,31 +45,36 @@ const ChangeContentSourcePage = () => {
46
45
  const contentHosts = useSelector(selectContentHosts);
47
46
  const hostsWithoutContent = useSelector(selectContentHostsWithoutContent);
48
47
  const contentSources = useSelector(selectContentSources);
49
- const jobInvocationPath = useSelector(selectJobInvocationPath);
50
48
 
51
49
  const template = useSelector(selectTemplate);
52
50
  const contentViews = useSelector(selectContentViews);
53
-
54
- const [contentSourceId, setCapsuleId] = useState('');
51
+ const { initialContentSourceId } = urlParams;
52
+ const [contentSourceId, setCapsuleId] = useState(initialContentSourceId ?? '');
53
+ // if this matches, we'll trust you that initialContentSourceId is the host's content source
54
+ const showCVOnlyAlert = (contentHosts.length === 1 &&
55
+ hostsWithoutContent.length === 0 &&
56
+ !!initialContentSourceId &&
57
+ initialContentSourceId === contentSourceId
58
+ );
59
+ const hostDetailsPath = showCVOnlyAlert ? `new/hosts/${contentHosts[0].name}` : '';
60
+ const hostEditPath = urlParams.fromPage === 'hostEdit' ? foremanUrl(`/hosts/${contentHosts[0]?.name}/edit`) : '';
55
61
  const [selectedEnvironment, setSelectedEnvironment] = useState([]);
56
62
  const [contentViewName, setContentViewName] = useState('');
57
63
  const [shouldShowTemplate, setShouldShowTemplate] = useState(false);
58
- const [redirect, setRedirect] = useState(false);
64
+ const [redirect, setRedirect] = useState('');
59
65
 
60
66
  const contentViewId = contentViews?.find(cv => cv.name === contentViewName)?.id;
61
67
  const hostIds = useMemo(() => getHostIds(urlParams.host_id), [urlParams.host_id]);
62
68
  const noHostSpecified = (hostIds.length === 0 && urlParams.searchParam === '');
63
69
  const environmentId = selectedEnvironment[0]?.id;
64
70
 
65
- const redirectToJobInvocationForm = () => setRedirect(true);
66
-
67
- const handleSuccess = ({ shouldRedirect }) => {
68
- if (shouldRedirect) {
69
- redirectToJobInvocationForm();
71
+ const handleSuccess = ({ redirectTo = '' }) => {
72
+ if (redirectTo) {
73
+ setRedirect(redirectTo);
70
74
  }
71
75
  };
72
76
 
73
- const handleSubmit = (e, { shouldRedirect = false }) => {
77
+ const handleSubmit = (e, { redirectTo = '', showSuccessToast = false } = {}) => {
74
78
  e.preventDefault();
75
79
 
76
80
  dispatch(changeContentSource(
@@ -78,7 +82,8 @@ const ChangeContentSourcePage = () => {
78
82
  contentViewId,
79
83
  contentSourceId,
80
84
  contentHosts.map(h => h.id),
81
- () => handleSuccess({ shouldRedirect }),
85
+ () => handleSuccess({ redirectTo }),
86
+ showSuccessToast ? () => __('Host content view environment(s) updated') : undefined,
82
87
  ));
83
88
  };
84
89
 
@@ -93,7 +98,7 @@ const ChangeContentSourcePage = () => {
93
98
  };
94
99
 
95
100
  const showTemplate = (e) => {
96
- handleSubmit(e, { shouldRedirect: false });
101
+ handleSubmit(e);
97
102
  setShouldShowTemplate(true);
98
103
  };
99
104
 
@@ -122,8 +127,8 @@ const ChangeContentSourcePage = () => {
122
127
  dispatch(getFormData(hostIds, urlParams.searchParam));
123
128
  }, [dispatch, hostIds, urlParams.searchParam]);
124
129
 
125
- if (redirect && jobInvocationPath) {
126
- window.location.assign(jobInvocationPath); // redirect to job invocation wizard
130
+ if (redirect && redirect !== '') {
131
+ window.location.assign(redirect);
127
132
  }
128
133
 
129
134
  return (
@@ -165,7 +170,7 @@ const ChangeContentSourcePage = () => {
165
170
  title={__('No hosts were specified')}
166
171
  />
167
172
  </GridItem>
168
- }
173
+ }
169
174
  { !noHostSpecified &&
170
175
  <>
171
176
  <Hosts
@@ -182,6 +187,9 @@ const ChangeContentSourcePage = () => {
182
187
  contentViewName={contentViewName}
183
188
  contentSources={contentSources}
184
189
  contentSourceId={contentSourceId}
190
+ showCVOnlyAlert={showCVOnlyAlert}
191
+ hostDetailsPath={hostDetailsPath}
192
+ hostEditPath={hostEditPath}
185
193
  handleContentSource={handleContentSource}
186
194
  contentHosts={contentHosts}
187
195
  isLoading={isLoading}
@@ -7,6 +7,7 @@ exports[`RedHatRepositories page should render <PermissionDenied /> when permiss
7
7
  "view_organizations",
8
8
  ]
9
9
  }
10
+ primaryButton={null}
10
11
  />
11
12
  `;
12
13
 
@@ -1,7 +1,10 @@
1
1
  import React, { Component } from 'react';
2
2
  import PropTypes from 'prop-types';
3
3
  import { Grid, Col, Row, Tabs, Tab, FormControl, ControlLabel } from 'react-bootstrap';
4
+ import { FormattedMessage } from 'react-intl';
4
5
  import { Button, Spinner } from 'patternfly-react';
6
+ import { Alert } from '@patternfly/react-core';
7
+ import { propsToCamelCase } from 'foremanReact/common/helpers';
5
8
  import ForemanModal from 'foremanReact/components/ForemanModal';
6
9
  import Slot from 'foremanReact/components/common/Slot';
7
10
  import { translate as __ } from 'foremanReact/common/I18n';
@@ -92,22 +95,29 @@ class ManageManifestModal extends Component {
92
95
  contentCredentials,
93
96
  } = this.props;
94
97
 
98
+ const {
99
+ manifestExpiringSoon,
100
+ manifestExpired,
101
+ manifestExpirationDate,
102
+ manifestExpireDaysRemaining,
103
+ } = propsToCamelCase(organization);
104
+
95
105
  const actionInProgress = (taskInProgress || manifestActionStarted);
96
106
  const showCdnConfigurationTab = canEditOrganizations;
97
107
  const showSubscriptionManifest = (canImportManifest || canDeleteManifest);
98
108
  const showManifestTab = (canEditOrganizations || showSubscriptionManifest);
99
109
 
100
110
  const emptyStateData = () => ({
101
- header: __('There is no Manifest History to display.'),
102
- description: __('Import a Manifest using the manifest tab above.'),
111
+ header: __('There is no manifest history to display.'),
112
+ description: __('Import a manifest using the Manifest tab above.'),
103
113
  documentation: {
104
- label: __('Learn more about adding Subscription Manifests '),
114
+ label: __('Learn more about adding subscription manifests '),
105
115
  url: 'https://access.redhat.com/solutions/3410771',
106
116
  },
107
117
  });
108
118
 
109
119
  const getManifestName = () => {
110
- let name = __('No Manifest Uploaded');
120
+ let name = __('No manifest imported');
111
121
 
112
122
  if (
113
123
  organization.owner_details &&
@@ -139,6 +149,46 @@ class ManageManifestModal extends Component {
139
149
  <React.Fragment>
140
150
  <Grid>
141
151
  <h3>{__('Subscription Manifest')}</h3>
152
+ {manifestExpiringSoon &&
153
+ <Alert
154
+ ouiaId="manifest-expiring-soon-alert"
155
+ variant="warning"
156
+ title={__('Manifest expiring soon')}
157
+ >
158
+ <FormattedMessage
159
+ defaultMessage={__('Your manifest will expire in {daysMessage}. To extend the expiration date, refresh your manifest. Or, if your Foreman is disconnected, import a new manifest.')}
160
+ values={{
161
+ daysMessage: (
162
+ <FormattedMessage
163
+ defaultMessage="{daysRemaining, plural, one {{singular}} other {# {plural}}}"
164
+ values={{
165
+ daysRemaining: manifestExpireDaysRemaining,
166
+ singular: __('day'),
167
+ plural: __('days'),
168
+ }}
169
+ id="manage-manifest-expire-days-i18n"
170
+ />
171
+ ),
172
+ }}
173
+ id="manage-manifest-expire-i18n"
174
+ />
175
+ </Alert>
176
+ }
177
+ {manifestExpired && isManifestImported &&
178
+ <Alert
179
+ ouiaId="manifest-expired-alert"
180
+ variant="danger"
181
+ title={__('Manifest expired')}
182
+ >
183
+ <FormattedMessage
184
+ defaultMessage={__('Your manifest expired on {expirationDate}. To continue using Red Hat content, import a new manifest.')}
185
+ values={{
186
+ expirationDate: new Date(manifestExpirationDate).toDateString(),
187
+ }}
188
+ id="manage-manifest-expired-i18n"
189
+ />
190
+ </Alert>
191
+ }
142
192
  <hr />
143
193
  <Row>
144
194
  <Col sm={5}>
@@ -148,13 +198,22 @@ class ManageManifestModal extends Component {
148
198
  {getManifestName()}
149
199
  </Col>
150
200
  </Row>
201
+ {isManifestImported && manifestExpirationDate &&
202
+ <Row>
203
+ <Col sm={5} />
204
+ <Col sm={7}>
205
+ {manifestExpired ? __('Expired ') : __('Expires ')}
206
+ {new Date(manifestExpirationDate).toDateString()}
207
+ </Col>
208
+ </Row>
209
+ }
151
210
  <Row>
152
211
  <Col sm={5}>
153
212
  {canImportManifest &&
154
213
  <ControlLabel
155
214
  style={{ paddingTop: '10px' }}
156
215
  >
157
- <div>{__('Import New Manifest')}</div>
216
+ <div>{__('Import new manifest')}</div>
158
217
  </ControlLabel>
159
218
  }
160
219
  </Col>
@@ -222,19 +222,22 @@ class UpstreamSubscriptionsPage extends Component {
222
222
 
223
223
  return (
224
224
  <Grid bsClass="container-fluid">
225
- <BreadcrumbsBar data={{
226
- isSwitchable: false,
227
- breadcrumbItems: [
228
- {
229
- caption: __('Subscriptions'),
230
- onClick: () => this.props.history.push('/subscriptions'),
231
- },
232
- {
233
- caption: __('Add Subscriptions'),
234
- },
235
- ],
236
- }}
237
- />
225
+ {!upstreamSubscriptions.loading &&
226
+ <div style={{ marginBottom: '10px' }}>
227
+ <BreadcrumbsBar
228
+ isLoadingResources={upstreamSubscriptions.loading}
229
+ breadcrumbItems={[
230
+ {
231
+ caption: __('Subscriptions'),
232
+ url: '/subscriptions/',
233
+ },
234
+ {
235
+ caption: String(__('Add Subscriptions')),
236
+ },
237
+ ]}
238
+ />
239
+ </div>
240
+ }
238
241
 
239
242
  <LoadingState loading={upstreamSubscriptions.loading} loadingText={__('Loading')}>
240
243
  <Row>
@@ -6,22 +6,28 @@ exports[`upstream subscriptions page should render 1`] = `
6
6
  componentClass="div"
7
7
  fluid={false}
8
8
  >
9
- <BreadcrumbsBar
10
- data={
9
+ <div
10
+ style={
11
11
  Object {
12
- "breadcrumbItems": Array [
12
+ "marginBottom": "10px",
13
+ }
14
+ }
15
+ >
16
+ <BreadcrumbsBar
17
+ breadcrumbItems={
18
+ Array [
13
19
  Object {
14
20
  "caption": "Subscriptions",
15
- "onClick": [Function],
21
+ "url": "/subscriptions/",
16
22
  },
17
23
  Object {
18
24
  "caption": "Add Subscriptions",
19
25
  },
20
- ],
21
- "isSwitchable": false,
26
+ ]
22
27
  }
23
- }
24
- />
28
+ isLoadingResources={false}
29
+ />
30
+ </div>
25
31
  <LoadingState
26
32
  loading={false}
27
33
  loadingText="Loading"
@@ -7,6 +7,7 @@ exports[`subscriptions page should render <PermissionDenied /> when permissions
7
7
  "view_subscriptions",
8
8
  ]
9
9
  }
10
+ primaryButton={null}
10
11
  />
11
12
  `;
12
13
 
@@ -49,7 +49,7 @@ const SubscriptionsToolbar = ({
49
49
  <FormGroup>
50
50
  {canManageSubscriptionAllocations &&
51
51
  <LinkContainer
52
- to="subscriptions/add"
52
+ to="/subscriptions/add"
53
53
  disabled={disableManifestActions || disableAddButton}
54
54
  >
55
55
  <TooltipButton