diff --git a/client/src/api/schema/schema.ts b/client/src/api/schema/schema.ts
index 87e99248cd64..1a8a270d0032 100644
--- a/client/src/api/schema/schema.ts
+++ b/client/src/api/schema/schema.ts
@@ -2594,6 +2594,23 @@ export interface paths {
patch?: never;
trace?: never;
};
+ "/api/invocations/{invocation_id}/metrics": {
+ parameters: {
+ query?: never;
+ header?: never;
+ path?: never;
+ cookie?: never;
+ };
+ /** Get Invocation Metrics */
+ get: operations["get_invocation_metrics_api_invocations__invocation_id__metrics_get"];
+ put?: never;
+ post?: never;
+ delete?: never;
+ options?: never;
+ head?: never;
+ patch?: never;
+ trace?: never;
+ };
"/api/invocations/{invocation_id}/prepare_store_download": {
parameters: {
query?: never;
@@ -18224,6 +18241,45 @@ export interface components {
[key: string]: number;
};
};
+ /**
+ * WorkflowJobMetric
+ * @example {
+ * "name": "start_epoch",
+ * "plugin": "core",
+ * "raw_value": "1614261340.0000000",
+ * "title": "Job Start Time",
+ * "value": "2021-02-25 14:55:40"
+ * }
+ */
+ WorkflowJobMetric: {
+ /**
+ * Name
+ * @description The name of the metric variable.
+ */
+ name: string;
+ /**
+ * Plugin
+ * @description The instrumenter plugin that generated this metric.
+ */
+ plugin: string;
+ /**
+ * Raw Value
+ * @description The raw value of the metric as a string.
+ */
+ raw_value: string;
+ /**
+ * Title
+ * @description A descriptive title for this metric.
+ */
+ title: string;
+ /** Tool Id */
+ tool_id: string;
+ /**
+ * Value
+ * @description The textual representation of the metric value.
+ */
+ value: string;
+ };
/** WorkflowLandingRequest */
WorkflowLandingRequest: {
/** Request State */
@@ -26928,6 +26984,50 @@ export interface operations {
};
};
};
+ get_invocation_metrics_api_invocations__invocation_id__metrics_get: {
+ parameters: {
+ query?: never;
+ header?: {
+ /** @description The user ID that will be used to effectively make this API call. Only admins and designated users can make API calls on behalf of other users. */
+ "run-as"?: string | null;
+ };
+ path: {
+ /** @description The encoded database identifier of the Invocation. */
+ invocation_id: string;
+ };
+ cookie?: never;
+ };
+ requestBody?: never;
+ responses: {
+ /** @description Successful Response */
+ 200: {
+ headers: {
+ [name: string]: unknown;
+ };
+ content: {
+ "application/json": components["schemas"]["WorkflowJobMetric"][];
+ };
+ };
+ /** @description Request Error */
+ "4XX": {
+ headers: {
+ [name: string]: unknown;
+ };
+ content: {
+ "application/json": components["schemas"]["MessageExceptionModel"];
+ };
+ };
+ /** @description Server Error */
+ "5XX": {
+ headers: {
+ [name: string]: unknown;
+ };
+ content: {
+ "application/json": components["schemas"]["MessageExceptionModel"];
+ };
+ };
+ };
+ };
prepare_store_download_api_invocations__invocation_id__prepare_store_download_post: {
parameters: {
query?: never;
diff --git a/client/src/components/WorkflowInvocationState/MetricsBoxPlots.vue b/client/src/components/WorkflowInvocationState/MetricsBoxPlots.vue
new file mode 100644
index 000000000000..1876190cc903
--- /dev/null
+++ b/client/src/components/WorkflowInvocationState/MetricsBoxPlots.vue
@@ -0,0 +1,169 @@
+
+
+
+
+
+
+
diff --git a/client/src/components/WorkflowInvocationState/WorkflowInvocationMetrics.vue b/client/src/components/WorkflowInvocationState/WorkflowInvocationMetrics.vue
new file mode 100644
index 000000000000..cb9aed99233f
--- /dev/null
+++ b/client/src/components/WorkflowInvocationState/WorkflowInvocationMetrics.vue
@@ -0,0 +1,52 @@
+
+
+
+
+
+
diff --git a/client/src/components/WorkflowInvocationState/WorkflowInvocationState.vue b/client/src/components/WorkflowInvocationState/WorkflowInvocationState.vue
index e035bc9b7dbe..bcf03f61b816 100644
--- a/client/src/components/WorkflowInvocationState/WorkflowInvocationState.vue
+++ b/client/src/components/WorkflowInvocationState/WorkflowInvocationState.vue
@@ -13,6 +13,7 @@ import { isTerminal, jobCount, runningCount } from "./util";
import InvocationReport from "../Workflow/InvocationReport.vue";
import WorkflowInvocationExportOptions from "./WorkflowInvocationExportOptions.vue";
+import WorkflowInvocationMetrics from "./WorkflowInvocationMetrics.vue";
import WorkflowInvocationHeader from "./WorkflowInvocationHeader.vue";
import WorkflowInvocationInputOutputTabs from "./WorkflowInvocationInputOutputTabs.vue";
import WorkflowInvocationOverview from "./WorkflowInvocationOverview.vue";
@@ -185,6 +186,9 @@ function cancelWorkflowSchedulingLocal() {
+
+
+
diff --git a/lib/galaxy/managers/jobs.py b/lib/galaxy/managers/jobs.py
index 1d4a61dac6f0..c93291f9863e 100644
--- a/lib/galaxy/managers/jobs.py
+++ b/lib/galaxy/managers/jobs.py
@@ -1,5 +1,6 @@
import json
import logging
+from collections.abc import Sequence
from datetime import (
date,
datetime,
@@ -26,6 +27,7 @@
null,
or_,
true,
+ union,
)
from sqlalchemy.orm import aliased
from sqlalchemy.sql import select
@@ -54,6 +56,7 @@
ImplicitCollectionJobs,
ImplicitCollectionJobsJobAssociation,
Job,
+ JobMetricNumeric,
JobParameter,
User,
Workflow,
@@ -729,6 +732,38 @@ def invocation_job_source_iter(sa_session, invocation_id):
yield ("ImplicitCollectionJobs", row[1], row[2])
+def get_job_metrics_for_invocation(sa_session: galaxy_scoped_session, invocation_id: int):
+ single_job_stmnt = (
+ select(JobMetricNumeric, Job.tool_id)
+ .join(Job, JobMetricNumeric.job_id == Job.id)
+ .join(
+ WorkflowInvocationStep,
+ and_(
+ WorkflowInvocationStep.workflow_invocation_id == invocation_id, WorkflowInvocationStep.job_id == Job.id
+ ),
+ )
+ )
+ collection_job_stmnt = (
+ select(JobMetricNumeric, Job.tool_id)
+ .join(Job, JobMetricNumeric.job_id == Job.id)
+ .join(ImplicitCollectionJobsJobAssociation, Job.id == ImplicitCollectionJobsJobAssociation.job_id)
+ .join(
+ ImplicitCollectionJobs,
+ ImplicitCollectionJobs.id == ImplicitCollectionJobsJobAssociation.implicit_collection_jobs_id,
+ )
+ .join(
+ WorkflowInvocationStep,
+ and_(
+ WorkflowInvocationStep.workflow_invocation_id == invocation_id,
+ WorkflowInvocationStep.implicit_collection_jobs_id == ImplicitCollectionJobs.id,
+ ),
+ )
+ )
+ # should be sa_session.execute(single_job_stmnt.union(collection_job_stmnt)).all() but that returns
+ # columns instead of the job metrics ORM instance.
+ return list(sa_session.execute(single_job_stmnt).all()) + list(sa_session.execute(collection_job_stmnt).all())
+
+
def fetch_job_states(sa_session, job_source_ids, job_source_types):
assert len(job_source_ids) == len(job_source_types)
job_ids = set()
@@ -911,6 +946,10 @@ def summarize_job_metrics(trans, job):
Precondition: the caller has verified the job is accessible to the user
represented by the trans parameter.
"""
+ return summarize_metrics(trans, job.metrics)
+
+
+def summarize_metrics(trans: ProvidesUserContext, job_metrics):
safety_level = Safety.SAFE
if trans.user_is_admin:
safety_level = Safety.UNSAFE
@@ -922,7 +961,7 @@ def summarize_job_metrics(trans, job):
m.metric_value,
m.plugin,
)
- for m in job.metrics
+ for m in job_metrics
]
dictifiable_metrics = trans.app.job_metrics.dictifiable_metrics(raw_metrics, safety_level)
return [d.dict() for d in dictifiable_metrics]
diff --git a/lib/galaxy/schema/schema.py b/lib/galaxy/schema/schema.py
index 0d9646316e59..88460a3f425f 100644
--- a/lib/galaxy/schema/schema.py
+++ b/lib/galaxy/schema/schema.py
@@ -2171,6 +2171,10 @@ class JobMetric(Model):
)
+class WorkflowJobMetric(JobMetric):
+ tool_id: str
+
+
class JobMetricCollection(RootModel):
"""Represents a collection of metrics associated with a Job."""
diff --git a/lib/galaxy/webapps/galaxy/api/workflows.py b/lib/galaxy/webapps/galaxy/api/workflows.py
index 6f94550e2134..4a568033d388 100644
--- a/lib/galaxy/webapps/galaxy/api/workflows.py
+++ b/lib/galaxy/webapps/galaxy/api/workflows.py
@@ -77,6 +77,7 @@
ShareWithPayload,
ShareWithStatus,
SharingStatus,
+ WorkflowJobMetric,
WorkflowLandingRequest,
WorkflowSortByEnum,
)
@@ -1760,3 +1761,11 @@ def workflow_invocation_jobs_summary(
) -> InvocationJobsResponse:
"""An alias for `GET /api/invocations/{invocation_id}/jobs_summary`. `workflow_id` is ignored."""
return self.invocation_jobs_summary(trans=trans, invocation_id=invocation_id)
+
+ @router.get("/api/invocations/{invocation_id}/metrics")
+ def get_invocation_metrics(
+ self,
+ invocation_id: InvocationIDPathParam,
+ trans: ProvidesUserContext = DependsOnTrans,
+ ) -> List[WorkflowJobMetric]:
+ return self.invocations_service.show_invocation_metrics(trans=trans, invocation_id=invocation_id)
diff --git a/lib/galaxy/webapps/galaxy/services/invocations.py b/lib/galaxy/webapps/galaxy/services/invocations.py
index 2195c7202dd7..701775c41f31 100644
--- a/lib/galaxy/webapps/galaxy/services/invocations.py
+++ b/lib/galaxy/webapps/galaxy/services/invocations.py
@@ -20,7 +20,9 @@
from galaxy.managers.histories import HistoryManager
from galaxy.managers.jobs import (
fetch_job_states,
+ get_job_metrics_for_invocation,
invocation_job_source_iter,
+ summarize_metrics,
)
from galaxy.managers.workflows import WorkflowsManager
from galaxy.model import (
@@ -147,6 +149,18 @@ def show_invocation_step(self, trans, step_id) -> InvocationStep:
)
return self.serialize_workflow_invocation_step(wfi_step)
+ def show_invocation_metrics(self, trans: ProvidesHistoryContext, invocation_id: int):
+ extended_job_metrics = get_job_metrics_for_invocation(trans.sa_session, invocation_id)
+ job_metrics = []
+ tool_ids = []
+ for row in extended_job_metrics:
+ job_metrics.append(row[0])
+ tool_ids.append(row[1])
+ metrics_dict_list = summarize_metrics(trans, job_metrics)
+ for tool_id, metrics_dict in zip(tool_ids, metrics_dict_list):
+ metrics_dict["tool_id"] = tool_id
+ return metrics_dict_list
+
def update_invocation_step(self, trans, step_id, action):
wfi_step = self._workflows_manager.update_invocation_step(trans, step_id, action)
return self.serialize_workflow_invocation_step(wfi_step)
diff --git a/lib/galaxy_test/api/test_workflows.py b/lib/galaxy_test/api/test_workflows.py
index 8408837df347..54aadc9ab949 100644
--- a/lib/galaxy_test/api/test_workflows.py
+++ b/lib/galaxy_test/api/test_workflows.py
@@ -3474,6 +3474,39 @@ def test_workflow_new_autocreated_history(self):
invocation_id = run_workflow_dict["id"]
self.workflow_populator.wait_for_invocation_and_jobs(new_history_id, workflow_id, invocation_id)
+ def test_invocation_job_metrics_simple(self):
+ with self.dataset_populator.test_history() as history_id:
+ summary = self._run_workflow(WORKFLOW_SIMPLE, test_data={"input1": "hello world"}, history_id=history_id)
+ self.workflow_populator.wait_for_invocation_and_jobs(
+ history_id=history_id, workflow_id=summary.workflow_id, invocation_id=summary.invocation_id
+ )
+ job_metrics = self._get(f"invocations/{summary.invocation_id}/metrics").json()
+ galaxy_slots = [m for m in job_metrics if m["name"] == "galaxy_slots"]
+ assert len(galaxy_slots) == 1
+
+ def test_invocation_job_metrics_map_over(self):
+ with self.dataset_populator.test_history() as history_id:
+ summary = self._run_workflow(
+ WORKFLOW_SIMPLE,
+ test_data={
+ "input1": {
+ "collection_type": "list",
+ "name": "the_dataset_list",
+ "elements": [
+ {"identifier": "el1", "value": "1.fastq", "type": "File"},
+ {"identifier": "el2", "value": "1.fastq", "type": "File"},
+ ],
+ }
+ },
+ history_id=history_id,
+ )
+ self.workflow_populator.wait_for_invocation_and_jobs(
+ history_id=history_id, workflow_id=summary.workflow_id, invocation_id=summary.invocation_id
+ )
+ job_metrics = self._get(f"invocations/{summary.invocation_id}/metrics").json()
+ galaxy_slots = [m for m in job_metrics if m["name"] == "galaxy_slots"]
+ assert len(galaxy_slots) == 2
+
def test_workflow_output_dataset(self):
with self.dataset_populator.test_history() as history_id:
summary = self._run_workflow(WORKFLOW_SIMPLE, test_data={"input1": "hello world"}, history_id=history_id)