mirror of https://github.com/portainer/portainer
297 lines
7.5 KiB
TypeScript
297 lines
7.5 KiB
TypeScript
import { CellContext, createColumnHelper } from '@tanstack/react-table';
|
|
import { ChevronDown, ChevronRight } from 'lucide-react';
|
|
import clsx from 'clsx';
|
|
import { useState } from 'react';
|
|
import _ from 'lodash';
|
|
|
|
import UpdatesAvailable from '@/assets/ico/icon_updates-available.svg?c';
|
|
import UpToDate from '@/assets/ico/icon_up-to-date.svg?c';
|
|
import { isoDateFromTimestamp } from '@/portainer/filters/filters';
|
|
import { isBE } from '@/react/portainer/feature-flags/feature-flags.service';
|
|
import { getDashboardRoute } from '@/react/portainer/environments/utils';
|
|
import { GitCommitLink } from '@/react/portainer/gitops/GitCommitLink';
|
|
|
|
import { Button } from '@@/buttons';
|
|
import { Icon } from '@@/Icon';
|
|
import { Link } from '@@/Link';
|
|
|
|
import { DeploymentStatus, EdgeStackStatus, StatusType } from '../../types';
|
|
|
|
import { EnvironmentActions } from './EnvironmentActions';
|
|
import { ActionStatus } from './ActionStatus';
|
|
import { EdgeStackEnvironment } from './types';
|
|
|
|
const columnHelper = createColumnHelper<EdgeStackEnvironment>();
|
|
|
|
export const columns = _.compact([
|
|
columnHelper.accessor('Name', {
|
|
id: 'name',
|
|
header: 'Name',
|
|
cell({ row: { original: env } }) {
|
|
const { to, params } = getDashboardRoute(env);
|
|
return (
|
|
<Link to={to} params={params}>
|
|
{env.Name}
|
|
</Link>
|
|
);
|
|
},
|
|
}),
|
|
columnHelper.accessor((env) => endpointStatusLabel(env.StackStatus.Status), {
|
|
id: 'status',
|
|
header: 'Status',
|
|
cell({ row: { original: env } }) {
|
|
return (
|
|
<ul className="list-none space-y-2">
|
|
{env.StackStatus.Status.map((s) => (
|
|
<li key={`status-${s.Type}-${s.Time}`}>
|
|
<Status value={s.Type} />
|
|
</li>
|
|
))}
|
|
</ul>
|
|
);
|
|
},
|
|
}),
|
|
columnHelper.accessor((env) => _.last(env.StackStatus.Status)?.Time, {
|
|
id: 'statusDate',
|
|
header: 'Time',
|
|
cell({ row: { original: env } }) {
|
|
return (
|
|
<ul className="list-none space-y-2">
|
|
{env.StackStatus.Status.map((s) => (
|
|
<li key={`time-${s.Type}-${s.Time}`}>
|
|
{isoDateFromTimestamp(s.Time)}
|
|
</li>
|
|
))}
|
|
</ul>
|
|
);
|
|
},
|
|
}),
|
|
...(isBE
|
|
? [
|
|
columnHelper.accessor((env) => endpointTargetVersionLabel(env), {
|
|
id: 'targetVersion',
|
|
header: 'Target version',
|
|
cell: TargetVersionCell,
|
|
}),
|
|
columnHelper.accessor(
|
|
(env) => endpointDeployedVersionLabel(env.StackStatus),
|
|
{
|
|
id: 'deployedVersion',
|
|
header: 'Deployed version',
|
|
cell: DeployedVersionCell,
|
|
}
|
|
),
|
|
]
|
|
: []),
|
|
columnHelper.accessor(
|
|
(env) =>
|
|
env.StackStatus.Status.find((s) => s.Type === StatusType.Error)?.Error,
|
|
{
|
|
id: 'error',
|
|
header: 'Error',
|
|
cell: ErrorCell,
|
|
}
|
|
),
|
|
...(isBE
|
|
? [
|
|
columnHelper.display({
|
|
id: 'actions',
|
|
header: 'Actions',
|
|
cell({ row: { original: env } }) {
|
|
return <EnvironmentActions environment={env} />;
|
|
},
|
|
}),
|
|
columnHelper.display({
|
|
id: 'actionStatus',
|
|
header: 'Action Status',
|
|
cell({ row: { original: env } }) {
|
|
return <ActionStatus environmentId={env.Id} />;
|
|
},
|
|
}),
|
|
]
|
|
: []),
|
|
]);
|
|
|
|
function ErrorCell({ getValue }: CellContext<EdgeStackEnvironment, string>) {
|
|
const [isExpanded, setIsExpanded] = useState(false);
|
|
|
|
const value = getValue();
|
|
if (!value) {
|
|
return '-';
|
|
}
|
|
|
|
return (
|
|
<Button
|
|
color="none"
|
|
className="flex cursor-pointer"
|
|
onClick={() => setIsExpanded(!isExpanded)}
|
|
>
|
|
<div className="pr-1 pt-0.5">
|
|
<Icon icon={isExpanded ? ChevronDown : ChevronRight} />
|
|
</div>
|
|
<div
|
|
className={clsx('overflow-hidden whitespace-normal', {
|
|
'h-[1.5em]': isExpanded,
|
|
})}
|
|
>
|
|
{value}
|
|
</div>
|
|
</Button>
|
|
);
|
|
}
|
|
|
|
function endpointStatusLabel(statusArray: Array<DeploymentStatus>) {
|
|
const labels = [];
|
|
|
|
statusArray.forEach((status) => {
|
|
if (status.Type === StatusType.Acknowledged) {
|
|
labels.push('Acknowledged');
|
|
}
|
|
if (status.Type === StatusType.ImagesPulled) {
|
|
labels.push('Images pre-pulled');
|
|
}
|
|
if (status.Type === StatusType.Running) {
|
|
labels.push('Deployed');
|
|
}
|
|
if (status.Type === StatusType.Error) {
|
|
labels.push('Failed');
|
|
}
|
|
if (status.Type === StatusType.PausedDeploying) {
|
|
labels.push('Paused');
|
|
}
|
|
if (status.Type === StatusType.RollingBack) {
|
|
labels.push('Rolling Back');
|
|
}
|
|
if (status.Type === StatusType.RolledBack) {
|
|
labels.push('Rolled Back');
|
|
}
|
|
});
|
|
|
|
if (!labels.length) {
|
|
labels.push('Pending');
|
|
}
|
|
|
|
return _.uniq(labels).join(', ');
|
|
}
|
|
|
|
function TargetVersionCell({
|
|
row,
|
|
getValue,
|
|
}: CellContext<EdgeStackEnvironment, string>) {
|
|
const value = getValue();
|
|
if (!value) {
|
|
return '';
|
|
}
|
|
|
|
return (
|
|
<>
|
|
{row.original.TargetCommitHash ? (
|
|
<div>
|
|
<GitCommitLink
|
|
baseURL={row.original.GitConfigURL}
|
|
commitHash={row.original.TargetCommitHash}
|
|
/>
|
|
</div>
|
|
) : (
|
|
<div>{value}</div>
|
|
)}
|
|
</>
|
|
);
|
|
}
|
|
|
|
function endpointTargetVersionLabel(env: EdgeStackEnvironment) {
|
|
if (env.TargetCommitHash) {
|
|
return env.TargetCommitHash.slice(0, 7).toString();
|
|
}
|
|
return env.TargetFileVersion.toString() || '';
|
|
}
|
|
|
|
function DeployedVersionCell({
|
|
row,
|
|
getValue,
|
|
}: CellContext<EdgeStackEnvironment, string>) {
|
|
const value = getValue();
|
|
if (!value || value === '0') {
|
|
return (
|
|
<div>
|
|
<Icon icon={UpdatesAvailable} className="!mr-2" />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
let statusIcon = <Icon icon={UpToDate} className="!mr-2" />;
|
|
if (
|
|
(row.original.TargetCommitHash &&
|
|
row.original.TargetCommitHash.slice(0, 7) !== value) ||
|
|
(!row.original.TargetCommitHash && row.original.TargetFileVersion !== value)
|
|
) {
|
|
statusIcon = <Icon icon={UpdatesAvailable} className="!mr-2" />;
|
|
}
|
|
|
|
return (
|
|
<>
|
|
{row.original.TargetCommitHash ? (
|
|
<div>
|
|
{statusIcon}
|
|
<GitCommitLink
|
|
baseURL={row.original.GitConfigURL}
|
|
commitHash={row.original.TargetCommitHash}
|
|
/>
|
|
</div>
|
|
) : (
|
|
<div>
|
|
{statusIcon}
|
|
{value}
|
|
</div>
|
|
)}
|
|
</>
|
|
);
|
|
}
|
|
|
|
function endpointDeployedVersionLabel(status: EdgeStackStatus) {
|
|
if (status.DeploymentInfo?.ConfigHash) {
|
|
return status.DeploymentInfo?.ConfigHash.slice(0, 7).toString();
|
|
}
|
|
return status.DeploymentInfo?.FileVersion.toString() || '';
|
|
}
|
|
|
|
function Status({ value }: { value: StatusType }) {
|
|
const color = getStateColor(value);
|
|
|
|
return (
|
|
<div className="flex items-center gap-2">
|
|
<span
|
|
className={clsx('h-2 w-2 rounded-full', {
|
|
'bg-orange-5': color === 'orange',
|
|
'bg-green-5': color === 'green',
|
|
'bg-error-5': color === 'red',
|
|
})}
|
|
/>
|
|
|
|
<span>{_.startCase(StatusType[value])}</span>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function getStateColor(type: StatusType): 'orange' | 'green' | 'red' {
|
|
switch (type) {
|
|
case StatusType.Acknowledged:
|
|
case StatusType.ImagesPulled:
|
|
case StatusType.DeploymentReceived:
|
|
case StatusType.Running:
|
|
case StatusType.RemoteUpdateSuccess:
|
|
case StatusType.Removed:
|
|
return 'green';
|
|
case StatusType.Error:
|
|
return 'red';
|
|
case StatusType.Pending:
|
|
case StatusType.Deploying:
|
|
case StatusType.Removing:
|
|
case StatusType.PausedDeploying:
|
|
case StatusType.RollingBack:
|
|
case StatusType.RolledBack:
|
|
default:
|
|
return 'orange';
|
|
}
|
|
}
|