Skip to content

Commit

Permalink
Add task instance details for each run (apache#45273)
Browse files Browse the repository at this point in the history
* init Details.tsx. add try number selector

* render try selector with latest try

* refactor to TrySelector.tsx

* add tooltip to try selector

* update with TaskTrySelect.tsx

* add table and populate fields

* rebasing. remove wrap button. change query for fetching task instance try details.
  • Loading branch information
dauinh authored Jan 3, 2025
1 parent aae1a57 commit 05fefb8
Show file tree
Hide file tree
Showing 3 changed files with 201 additions and 2 deletions.
3 changes: 2 additions & 1 deletion airflow/ui/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -78,5 +78,6 @@
"vite-plugin-css-injected-by-js": "^3.5.2",
"vitest": "^2.1.1",
"web-worker": "^1.3.0"
}
},
"packageManager": "[email protected]+sha512.76e2379760a4328ec4415815bcd6628dee727af3779aaa4c914e3944156c4299921a89f976381ee107d41f12cfa4b66681ca9c718f0668fa0831ed4c6d8ba56c"
}
197 changes: 197 additions & 0 deletions airflow/ui/src/pages/Run/Details.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,197 @@
/*!
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
import { Box, Flex, HStack, Table } from "@chakra-ui/react";
import { useParams, useSearchParams } from "react-router-dom";

import {
useTaskInstanceServiceGetMappedTaskInstance,
useTaskInstanceServiceGetTaskInstanceTryDetails,
} from "openapi/queries";
import { TaskTrySelect } from "src/components/TaskTrySelect";
import Time from "src/components/Time";
import { ClipboardRoot, ClipboardIconButton, Status } from "src/components/ui";
import { getDuration } from "src/utils";

export const Details = () => {
const { dagId = "", runId = "", taskId = "" } = useParams();
const [searchParams, setSearchParams] = useSearchParams();

const mapIndexParam = searchParams.get("map_index");
const tryNumberParam = searchParams.get("try_number");
const mapIndex = parseInt(mapIndexParam ?? "-1", 10);

const {
data: taskInstance,
error,
isLoading,
} = useTaskInstanceServiceGetMappedTaskInstance({
dagId,
dagRunId: runId,
mapIndex,
taskId,
});

const onSelectTryNumber = (newTryNumber: number) => {
if (newTryNumber === taskInstance?.try_number) {
searchParams.delete("try_number");
} else {
searchParams.set("try_number", newTryNumber.toString());
}
setSearchParams(searchParams);
};

const tryNumber = tryNumberParam === null ? taskInstance?.try_number : parseInt(tryNumberParam, 10);

const { data: tryInstance } = useTaskInstanceServiceGetTaskInstanceTryDetails({
dagId,
dagRunId: runId,
mapIndex,
taskId,
taskTryNumber: tryNumber,
});

return (
<Box p={2}>
{taskInstance === undefined || tryNumber === undefined || taskInstance.try_number <= 1 ? (
<div />
) : (
<TaskTrySelect
onSelectTryNumber={onSelectTryNumber}
selectedTryNumber={tryNumber}
taskInstance={taskInstance}
/>
)}
<Table.Root striped>
<Table.Body>
<Table.Row>
<Table.Cell>Status</Table.Cell>
<Table.Cell>
<Flex gap={1}>
<Status state={tryInstance?.state ?? undefined} />
{tryInstance?.state ?? "no status"}
</Flex>
</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Task ID</Table.Cell>
<Table.Cell>
<HStack>
{tryInstance?.task_id}
<ClipboardRoot value={tryInstance?.task_id}>
<ClipboardIconButton />
</ClipboardRoot>
</HStack>
</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Run ID</Table.Cell>
<Table.Cell>
<HStack>
{tryInstance?.dag_run_id}
<ClipboardRoot value={tryInstance?.dag_run_id}>
<ClipboardIconButton />
</ClipboardRoot>
</HStack>
</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Map Index</Table.Cell>
<Table.Cell>{tryInstance?.map_index}</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Operator</Table.Cell>
<Table.Cell>{tryInstance?.operator}</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Duration</Table.Cell>
<Table.Cell>
{getDuration(tryInstance?.start_date ?? null, tryInstance?.end_date ?? null)}s
</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Started</Table.Cell>
<Table.Cell>
<Time datetime={tryInstance?.start_date} />
</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Ended</Table.Cell>
<Table.Cell>
<Time datetime={tryInstance?.end_date} />
</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Process ID (PID)</Table.Cell>
<Table.Cell>
<HStack>
{tryInstance?.pid}
<ClipboardRoot value={tryInstance?.pid}>
<ClipboardIconButton />
</ClipboardRoot>
</HStack>
</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Hostname</Table.Cell>
<Table.Cell>
<HStack>
{tryInstance?.hostname}
<ClipboardRoot value={tryInstance?.hostname}>
<ClipboardIconButton />
</ClipboardRoot>
</HStack>
</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Pool</Table.Cell>
<Table.Cell>{tryInstance?.pool}</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Pool Slots</Table.Cell>
<Table.Cell>{tryInstance?.pool_slots}</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Executor</Table.Cell>
<Table.Cell>{tryInstance?.executor}</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Executor Config</Table.Cell>
<Table.Cell>{tryInstance?.executor_config}</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Unix Name</Table.Cell>
<Table.Cell>{tryInstance?.unixname}</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Max Tries</Table.Cell>
<Table.Cell>{tryInstance?.max_tries}</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Queue</Table.Cell>
<Table.Cell>{tryInstance?.queue}</Table.Cell>
</Table.Row>
<Table.Row>
<Table.Cell>Priority Weight</Table.Cell>
<Table.Cell>{tryInstance?.priority_weight}</Table.Cell>
</Table.Row>
</Table.Body>
</Table.Root>
</Box>
);
};
3 changes: 2 additions & 1 deletion airflow/ui/src/router.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@ import { Dashboard } from "src/pages/Dashboard";
import { ErrorPage } from "src/pages/Error";
import { Events } from "src/pages/Events";
import { Run } from "src/pages/Run";
import { Details } from "src/pages/Run/Details";
import { TaskInstances } from "src/pages/Run/TaskInstances";
import { Task, Instances } from "src/pages/Task";
import { TaskInstance, Logs } from "src/pages/TaskInstance";
Expand Down Expand Up @@ -90,7 +91,7 @@ export const router = createBrowserRouter(
{ element: <Events />, path: "events" },
{ element: <XCom />, path: "xcom" },
{ element: <Code />, path: "code" },
{ element: <div>Details</div>, path: "details" },
{ element: <Details />, path: "details" },
],
element: <TaskInstance />,
path: "dags/:dagId/runs/:runId/tasks/:taskId",
Expand Down

0 comments on commit 05fefb8

Please sign in to comment.