diff --git a/app/controllers/api/v2/job_invocations_controller.rb b/app/controllers/api/v2/job_invocations_controller.rb
index a27b9464b..1d1467b6f 100644
--- a/app/controllers/api/v2/job_invocations_controller.rb
+++ b/app/controllers/api/v2/job_invocations_controller.rb
@@ -3,10 +3,11 @@ module V2
class JobInvocationsController < ::Api::V2::BaseController
include ::Api::Version2
include ::Foreman::Renderer
+ include RemoteExecutionHelper
before_action :find_optional_nested_object, :only => %w{output raw_output}
before_action :find_host, :only => %w{output raw_output}
- before_action :find_resource, :only => %w{show update destroy clone cancel rerun outputs}
+ before_action :find_resource, :only => %w{show update destroy clone cancel rerun outputs hosts}
wrap_parameters JobInvocation, :include => (JobInvocation.attribute_names + [:ssh])
@@ -27,7 +28,14 @@ def show
if params[:host_status] == 'true'
template_invocations = @template_invocations.includes(:run_host_job_task).to_a
- @host_statuses = Hash[template_invocations.map { |ti| [ti.host_id, template_invocation_status(ti)] }]
+ hosts = @hosts.to_a
+ @host_statuses = Hash[hosts.map do |host|
+ template_invocation = template_invocations.find { |ti| ti.host_id == host.id }
+ task = template_invocation.try(:run_host_job_task)
+ [host.id, template_invocation_status(task, @job_invocation.task)]
+ end]
+ @smart_proxy_id = Hash[template_invocations.map { |ti| [ti.host_id, ti.smart_proxy_id] }]
+ @smart_proxy_name = Hash[template_invocations.map { |ti| [ti.host_id, ti.smart_proxy_name] }]
end
end
@@ -111,6 +119,29 @@ def output
render :json => host_output(@nested_obj, @host, :default => [], :since => params[:since])
end
+ api :GET, '/job_invocations/:id/hosts', N_('List hosts belonging to job invocation')
+ param_group :search_and_pagination, ::Api::V2::BaseController
+ add_scoped_search_description_for(JobInvocation)
+ param :id, :identifier, :required => true
+ def hosts
+ @hosts = @job_invocation.targeting.hosts.authorized(:view_hosts, Host)
+ @total = @job_invocation.targeting.hosts.size
+ @template_invocations = @job_invocation.template_invocations
+ .where(host: @hosts)
+ .includes(:input_values)
+ template_invocations = @template_invocations.includes(:run_host_job_task).to_a
+ hosts = @hosts.to_a
+ @host_statuses = Hash[hosts.map do |host|
+ template_invocation = template_invocations.find { |ti| ti.host_id == host.id }
+ task = template_invocation.try(:run_host_job_task)
+ [host.id, template_invocation_status(task, @job_invocation.task)]
+ end]
+ @smart_proxy_id = Hash[template_invocations.map { |ti| [ti.host_id, ti.smart_proxy_id] }]
+ @smart_proxy_name = Hash[template_invocations.map { |ti| [ti.host_id, ti.smart_proxy_name] }]
+ @hosts = @hosts.search_for(params[:search], :order => params[:order]).paginate(:page => params[:page], :per_page => params[:per_page])
+ render :hosts, :layout => 'api/v2/layouts/index_layout'
+ end
+
api :GET, '/job_invocations/:id/hosts/:host_id/raw', N_('Get raw output for a host')
param :id, :identifier, :required => true
param :host_id, :identifier, :required => true
@@ -187,7 +218,7 @@ def allowed_nested_id
def action_permission
case params[:action]
- when 'output', 'raw_output', 'outputs'
+ when 'output', 'raw_output', 'outputs', 'hosts'
:view
when 'cancel'
:cancel
@@ -255,17 +286,6 @@ def delayed_task_output(task, default: nil)
def parent_scope
resource_class.where(nil)
end
-
- def template_invocation_status(template_invocation)
- task = template_invocation.try(:run_host_job_task)
- parent_task = @job_invocation.task
-
- return(parent_task.result == 'cancelled' ? 'cancelled' : 'N/A') if task.nil?
- return task.state if task.state == 'running' || task.state == 'planned'
- return 'error' if task.result == 'warning'
-
- task.result
- end
end
end
end
diff --git a/app/views/api/v2/job_invocations/hosts.json.rabl b/app/views/api/v2/job_invocations/hosts.json.rabl
new file mode 100644
index 000000000..0467f9ce5
--- /dev/null
+++ b/app/views/api/v2/job_invocations/hosts.json.rabl
@@ -0,0 +1,15 @@
+collection @hosts
+
+attribute :name, :operatingsystem_id, :operatingsystem_name, :hostgroup_id, :hostgroup_name
+
+node :job_status do |host|
+ @host_statuses[host.id]
+end
+
+node :smart_proxy_id do |host|
+ @smart_proxy_id[host.id]
+end
+
+node :smart_proxy_name do |host|
+ @smart_proxy_name[host.id]
+end
diff --git a/config/routes.rb b/config/routes.rb
index 51d97f21b..3cf6759ac 100644
--- a/config/routes.rb
+++ b/config/routes.rb
@@ -63,6 +63,7 @@
get '/raw', :to => 'job_invocations#raw_output'
end
member do
+ get 'hosts'
post 'cancel'
post 'rerun'
get 'template_invocations', :to => 'template_invocations#template_invocations'
diff --git a/webpack/JobInvocationDetail/JobInvocationActions.js b/webpack/JobInvocationDetail/JobInvocationActions.js
index a70b80e6b..f80425065 100644
--- a/webpack/JobInvocationDetail/JobInvocationActions.js
+++ b/webpack/JobInvocationDetail/JobInvocationActions.js
@@ -15,7 +15,7 @@ import {
UPDATE_JOB,
} from './JobInvocationConstants';
-export const getData = url => dispatch => {
+export const getJobInvocation = url => dispatch => {
const fetchData = withInterval(
get({
key: JOB_INVOCATION_KEY,
diff --git a/webpack/JobInvocationDetail/JobInvocationConstants.js b/webpack/JobInvocationDetail/JobInvocationConstants.js
index 2673d3aaf..a67f916a9 100644
--- a/webpack/JobInvocationDetail/JobInvocationConstants.js
+++ b/webpack/JobInvocationDetail/JobInvocationConstants.js
@@ -1,14 +1,21 @@
+/* eslint-disable camelcase */
+import React from 'react';
import { foremanUrl } from 'foremanReact/common/helpers';
+import { translate as __ } from 'foremanReact/common/I18n';
+import { useForemanHostDetailsPageUrl } from 'foremanReact/Root/Context/ForemanContext';
+import JobStatusIcon from '../react_app/components/RecentJobsCard/JobStatusIcon';
export const JOB_INVOCATION_KEY = 'JOB_INVOCATION_KEY';
export const CURRENT_PERMISSIONS = 'CURRENT_PERMISSIONS';
export const UPDATE_JOB = 'UPDATE_JOB';
export const CANCEL_JOB = 'CANCEL_JOB';
export const GET_TASK = 'GET_TASK';
+export const GET_TEMPLATE_INVOCATIONS = 'GET_TEMPLATE_INVOCATIONS';
export const CHANGE_ENABLED_RECURRING_LOGIC = 'CHANGE_ENABLED_RECURRING_LOGIC';
export const CANCEL_RECURRING_LOGIC = 'CANCEL_RECURRING_LOGIC';
export const GET_REPORT_TEMPLATES = 'GET_REPORT_TEMPLATES';
export const GET_REPORT_TEMPLATE_INPUTS = 'GET_REPORT_TEMPLATE_INPUTS';
+export const JOB_INVOCATION_HOSTS = 'JOB_INVOCATION_HOSTS';
export const currentPermissionsUrl = foremanUrl(
'/api/v2/permissions/current_permissions'
);
@@ -20,6 +27,12 @@ export const STATUS = {
CANCELLED: 'cancelled',
};
+export const STATUS_UPPERCASE = {
+ RESOLVED: 'RESOLVED',
+ ERROR: 'ERROR',
+ PENDING: 'PENDING',
+};
+
export const DATE_OPTIONS = {
day: 'numeric',
month: 'short',
@@ -29,3 +42,74 @@ export const DATE_OPTIONS = {
hour12: false,
timeZoneName: 'short',
};
+
+const Columns = () => {
+ const getColumnsStatus = ({ hostJobStatus }) => {
+ switch (hostJobStatus) {
+ case 'success':
+ return { title: __('Succeeded'), status: 0 };
+ case 'error':
+ return { title: __('Failed'), status: 1 };
+ case 'planned':
+ return { title: __('Scheduled'), status: 2 };
+ case 'running':
+ return { title: __('Pending'), status: 3 };
+ case 'cancelled':
+ return { title: __('Cancelled'), status: 4 };
+ case 'Awaiting start':
+ return { title: __('Awaiting start'), status: 5 };
+ default:
+ return { title: hostJobStatus, status: 6 };
+ }
+ };
+ const hostDetailsPageUrl = useForemanHostDetailsPageUrl();
+
+ return {
+ name: {
+ title: __('Name'),
+ wrapper: ({ name }) => (
+ {name}
+ ),
+ weight: 1,
+ },
+ groups: {
+ title: __('Host group'),
+ wrapper: ({ hostgroup_id, hostgroup_name }) => (
+ {hostgroup_name}
+ ),
+ weight: 2,
+ },
+ os: {
+ title: __('OS'),
+ wrapper: ({ operatingsystem_id, operatingsystem_name }) => (
+
+ {operatingsystem_name}
+
+ ),
+ weight: 3,
+ },
+ smart_proxy: {
+ title: __('Smart proxy'),
+ wrapper: ({ smart_proxy_name, smart_proxy_id }) => (
+ {smart_proxy_name}
+ ),
+ weight: 4,
+ },
+ status: {
+ title: __('Status'),
+ wrapper: ({ job_status }) => {
+ const { title, status } = getColumnsStatus({
+ hostJobStatus: job_status,
+ });
+ return (
+
+ {title || __('Unknown')}
+
+ );
+ },
+ weight: 5,
+ },
+ };
+};
+
+export default Columns;
diff --git a/webpack/JobInvocationDetail/JobInvocationDetail.scss b/webpack/JobInvocationDetail/JobInvocationDetail.scss
index 9482277a0..6cc58986e 100644
--- a/webpack/JobInvocationDetail/JobInvocationDetail.scss
+++ b/webpack/JobInvocationDetail/JobInvocationDetail.scss
@@ -38,4 +38,3 @@
height: $chart_size;
}
}
-
\ No newline at end of file
diff --git a/webpack/JobInvocationDetail/JobInvocationHostTable.js b/webpack/JobInvocationDetail/JobInvocationHostTable.js
new file mode 100644
index 000000000..e3b6f1a1f
--- /dev/null
+++ b/webpack/JobInvocationDetail/JobInvocationHostTable.js
@@ -0,0 +1,163 @@
+/* eslint-disable camelcase */
+import PropTypes from 'prop-types';
+import React, { useMemo } from 'react';
+import { Icon } from 'patternfly-react';
+import { translate as __ } from 'foremanReact/common/I18n';
+import { Tr, Td } from '@patternfly/react-table';
+import {
+ Title,
+ EmptyState,
+ EmptyStateVariant,
+ EmptyStateBody,
+} from '@patternfly/react-core';
+import { foremanUrl } from 'foremanReact/common/helpers';
+import { useAPI } from 'foremanReact/common/hooks/API/APIHooks';
+import { Table } from 'foremanReact/components/PF4/TableIndexPage/Table/Table';
+import TableIndexPage from 'foremanReact/components/PF4/TableIndexPage/TableIndexPage';
+import { useSetParamsAndApiAndSearch } from 'foremanReact/components/PF4/TableIndexPage/Table/TableIndexHooks';
+import {
+ useBulkSelect,
+ useUrlParams,
+} from 'foremanReact/components/PF4/TableIndexPage/Table/TableHooks';
+import { getControllerSearchProps } from 'foremanReact/constants';
+import Columns, {
+ JOB_INVOCATION_HOSTS,
+ STATUS_UPPERCASE,
+} from './JobInvocationConstants';
+
+const JobInvocationHostTable = ({ id, targeting }) => {
+ const columns = Columns();
+ const columnNamesKeys = Object.keys(columns);
+ const apiOptions = { key: JOB_INVOCATION_HOSTS, search: urlSearchQuery };
+ const {
+ search: urlSearchQuery = '',
+ page: urlPage,
+ per_page: urlPerPage,
+ } = useUrlParams();
+ const defaultParams = { search: urlSearchQuery };
+ if (urlPage) defaultParams.page = Number(urlPage);
+ if (urlPerPage) defaultParams.per_page = Number(urlPerPage);
+ const { response, status, setAPIOptions } = useAPI(
+ 'get',
+ `/api/job_invocations/${id}/hosts`,
+ {
+ params: { ...defaultParams, key: JOB_INVOCATION_HOSTS },
+ }
+ );
+
+ const { setParamsAndAPI, params } = useSetParamsAndApiAndSearch({
+ defaultParams,
+ apiOptions,
+ setAPIOptions,
+ });
+
+ const combinedResponse = {
+ response: {
+ search: '',
+ can_create: false,
+ results: response?.results || [],
+ total: response?.total || 0,
+ per_page: response?.perPage,
+ page: response?.page,
+ subtotal: response?.subtotal || 0,
+ message: response?.message || 'error',
+ },
+ status,
+ setAPIOptions,
+ };
+
+ const { updateSearchQuery } = useBulkSelect({
+ initialSearchQuery: urlSearchQuery,
+ });
+
+ const controller = 'hosts';
+ const memoDefaultSearchProps = useMemo(
+ () => getControllerSearchProps(controller),
+ [controller]
+ );
+ memoDefaultSearchProps.autocomplete.url = foremanUrl(
+ `/${controller}/auto_complete_search`
+ );
+
+ const customEmptyState = (
+
+
+
+
+
+
+
+ {__('No Results')}
+
+
+
+ {targeting?.targeting_type === 'dynamic_query' ? (
+ <>
+ {__('The dynamic query is still being processed. You can ')}
+
+ {__('view the hosts')}
+
+ {__(' targeted by the query.')}
+ >
+ ) : (
+ __('No hosts found')
+ )}
+
+
+
+ |
+
+ );
+
+ return (
+
+ {}}
+ errorMessage={
+ status === STATUS_UPPERCASE.ERROR && response?.message
+ ? response.message
+ : null
+ }
+ isPending={status === STATUS_UPPERCASE.PENDING}
+ isDeleteable={false}
+ >
+ {response?.results?.map((result, rowIndex) => (
+
+ {columnNamesKeys.map(k => (
+ {columns[k].wrapper(result)} |
+ ))}
+
+ ))}
+
+
+ );
+};
+
+JobInvocationHostTable.propTypes = {
+ id: PropTypes.string.isRequired,
+ targeting: PropTypes.object.isRequired,
+};
+
+JobInvocationHostTable.defaultProps = {};
+
+export default JobInvocationHostTable;
diff --git a/webpack/JobInvocationDetail/JobInvocationSelectors.js b/webpack/JobInvocationDetail/JobInvocationSelectors.js
index cd2eb57ca..f5ba99d66 100644
--- a/webpack/JobInvocationDetail/JobInvocationSelectors.js
+++ b/webpack/JobInvocationDetail/JobInvocationSelectors.js
@@ -1,10 +1,10 @@
import { selectAPIResponse } from 'foremanReact/redux/API/APISelectors';
-import { JOB_INVOCATION_KEY } from './JobInvocationConstants';
+import { JOB_INVOCATION_KEY, GET_TASK } from './JobInvocationConstants';
export const selectItems = state =>
selectAPIResponse(state, JOB_INVOCATION_KEY);
-export const selectTask = state => selectAPIResponse(state, 'GET_TASK');
+export const selectTask = state => selectAPIResponse(state, GET_TASK);
export const selectTaskCancelable = state =>
selectTask(state).available_actions?.cancellable || false;
diff --git a/webpack/JobInvocationDetail/__tests__/MainInformation.test.js b/webpack/JobInvocationDetail/__tests__/MainInformation.test.js
index 83d6e045c..bfcc4d158 100644
--- a/webpack/JobInvocationDetail/__tests__/MainInformation.test.js
+++ b/webpack/JobInvocationDetail/__tests__/MainInformation.test.js
@@ -76,6 +76,10 @@ api.get.mockImplementation(({ handleSuccess, ...action }) => {
return { type: 'get', ...action };
});
+jest.mock('../JobInvocationHostTable.js', () => () => (
+ Mock Table
+));
+
const reportTemplateJobId = mockReportTemplatesResponse.results[0].id;
const mockStore = configureMockStore([thunk]);
@@ -207,7 +211,7 @@ describe('JobInvocationDetailPage', () => {
{ key: GET_REPORT_TEMPLATES, url: '/api/report_templates' },
{
key: JOB_INVOCATION_KEY,
- url: `/api/job_invocations/${jobId}`,
+ url: `/api/job_invocations/${jobId}?host_status=true`,
},
{
key: GET_REPORT_TEMPLATE_INPUTS,
diff --git a/webpack/JobInvocationDetail/__tests__/fixtures.js b/webpack/JobInvocationDetail/__tests__/fixtures.js
index c4f9e7846..d984a84f0 100644
--- a/webpack/JobInvocationDetail/__tests__/fixtures.js
+++ b/webpack/JobInvocationDetail/__tests__/fixtures.js
@@ -1,4 +1,7 @@
export const jobInvocationData = {
+ search: '',
+ per_page: 20,
+ page: 1,
id: 123,
description: 'Description',
job_category: 'Commands',
@@ -40,6 +43,9 @@ export const jobInvocationData = {
};
export const jobInvocationDataScheduled = {
+ search: '',
+ per_page: 20,
+ page: 1,
id: 456,
description: 'Description',
job_category: 'Commands',
@@ -62,6 +68,9 @@ export const jobInvocationDataScheduled = {
};
export const jobInvocationDataRecurring = {
+ search: '',
+ per_page: 20,
+ page: 1,
id: 789,
description: 'Description',
job_category: 'Commands',
diff --git a/webpack/JobInvocationDetail/index.js b/webpack/JobInvocationDetail/index.js
index 1605a46ae..43816ccde 100644
--- a/webpack/JobInvocationDetail/index.js
+++ b/webpack/JobInvocationDetail/index.js
@@ -1,12 +1,17 @@
import PropTypes from 'prop-types';
import React, { useEffect } from 'react';
import { useDispatch, useSelector } from 'react-redux';
-import { Divider, Flex } from '@patternfly/react-core';
+import {
+ Divider,
+ Flex,
+ PageSection,
+ PageSectionVariants,
+} from '@patternfly/react-core';
import { translate as __, documentLocale } from 'foremanReact/common/I18n';
import PageLayout from 'foremanReact/routes/common/PageLayout/PageLayout';
import { useAPI } from 'foremanReact/common/hooks/API/APIHooks';
import { stopInterval } from 'foremanReact/redux/middlewares/IntervalMiddleware';
-import { getData, getTask } from './JobInvocationActions';
+import { getJobInvocation, getTask } from './JobInvocationActions';
import {
CURRENT_PERMISSIONS,
DATE_OPTIONS,
@@ -19,6 +24,7 @@ import JobInvocationOverview from './JobInvocationOverview';
import { selectItems } from './JobInvocationSelectors';
import JobInvocationSystemStatusChart from './JobInvocationSystemStatusChart';
import JobInvocationToolbarButtons from './JobInvocationToolbarButtons';
+import JobInvocationHostTable from './JobInvocationHostTable';
const JobInvocationDetailPage = ({
match: {
@@ -32,6 +38,7 @@ const JobInvocationDetailPage = ({
status_label: statusLabel,
task,
start_at: startAt,
+ targeting,
} = items;
const finished =
statusLabel === STATUS.FAILED ||
@@ -57,7 +64,7 @@ const JobInvocationDetailPage = ({
}
useEffect(() => {
- dispatch(getData(`/api/job_invocations/${id}`));
+ dispatch(getJobInvocation(`/api/job_invocations/${id}?host_status=true`));
if (finished && !autoRefresh) {
dispatch(stopInterval(JOB_INVOCATION_KEY));
}
@@ -82,45 +89,55 @@ const JobInvocationDetailPage = ({
};
return (
-
- }
- searchable={false}
- >
-
+
+ }
+ searchable={false}
>
-
-
-
+
+
+
+
-
-
+
+
+ {items.id !== undefined && (
+
+ )}
+
+ >
);
};
diff --git a/webpack/__mocks__/foremanReact/components/HostDetails/DetailsCard/DefaultLoaderEmptyState.js b/webpack/__mocks__/foremanReact/components/HostDetails/DetailsCard/DefaultLoaderEmptyState.js
index a8f9748d1..533d49bd7 100644
--- a/webpack/__mocks__/foremanReact/components/HostDetails/DetailsCard/DefaultLoaderEmptyState.js
+++ b/webpack/__mocks__/foremanReact/components/HostDetails/DetailsCard/DefaultLoaderEmptyState.js
@@ -1,8 +1,7 @@
import React from 'react';
-import { translate as __ } from '../../../common/I18n';
const DefaultLoaderEmptyState = () => (
- {__('Not available')}
+ Not available
);
export default DefaultLoaderEmptyState;
diff --git a/webpack/react_app/components/RecentJobsCard/JobStatusIcon.js b/webpack/react_app/components/RecentJobsCard/JobStatusIcon.js
index 3fb1d3493..a5b819d67 100644
--- a/webpack/react_app/components/RecentJobsCard/JobStatusIcon.js
+++ b/webpack/react_app/components/RecentJobsCard/JobStatusIcon.js
@@ -3,29 +3,60 @@ import PropTypes from 'prop-types';
import {
CheckCircleIcon,
ExclamationCircleIcon,
+ BuildIcon,
+ RunningIcon,
+ ExclamationTriangleIcon,
QuestionCircleIcon,
} from '@patternfly/react-icons';
-import { JOB_SUCCESS_STATUS, JOB_ERROR_STATUS } from './constants';
+import {
+ JOB_SUCCESS_STATUS,
+ JOB_ERROR_STATUS,
+ JOB_PLANNED_STATUS,
+ JOB_RUNNING_STATUS,
+ JOB_CANCELLED_STATUS,
+ JOB_AWAITING_STATUS,
+} from './constants';
import './styles.scss';
const JobStatusIcon = ({ status, children, ...props }) => {
switch (status) {
case JOB_SUCCESS_STATUS:
return (
-
- {children}
+
+ {children}
);
case JOB_ERROR_STATUS:
return (
-
- {children}
+
+ {children}
+
+ );
+ case JOB_PLANNED_STATUS:
+ return (
+
+ {children}
+
+ );
+ case JOB_RUNNING_STATUS:
+ return (
+
+ {children}
+
+ );
+ case JOB_CANCELLED_STATUS:
+ return (
+
+ {' '}
+ {children}
);
+ case JOB_AWAITING_STATUS:
+ return {children};
default:
return (
-
- {children}
+
+ {children}
);
}
diff --git a/webpack/react_app/components/RecentJobsCard/constants.js b/webpack/react_app/components/RecentJobsCard/constants.js
index bd298c706..a5b9cde50 100644
--- a/webpack/react_app/components/RecentJobsCard/constants.js
+++ b/webpack/react_app/components/RecentJobsCard/constants.js
@@ -5,6 +5,10 @@ export const SCHEDULED_TAB = 2;
export const JOB_SUCCESS_STATUS = 0;
export const JOB_ERROR_STATUS = 1;
+export const JOB_PLANNED_STATUS = 2;
+export const JOB_RUNNING_STATUS = 3;
+export const JOB_CANCELLED_STATUS = 4;
+export const JOB_AWAITING_STATUS = 5;
export const JOB_BASE_URL = '/job_invocations?search=targeted_host_id+%3D+';
export const JOB_API_URL =