Skip to content

Commit

Permalink
feat: Add new ports view to the EM UI (#1919)
Browse files Browse the repository at this point in the history
## Description:
Adding local or cloud based links to the UI

## Is this change user facing?
YES

## References (if applicable):
<!-- Add relevant Github Issues, Discord threads, or other helpful
information. -->

---------

Co-authored-by: Laurent Luce <[email protected]>
  • Loading branch information
adschwartz and laurentluce authored Dec 11, 2023
1 parent 97b9b80 commit 027d74b
Show file tree
Hide file tree
Showing 10 changed files with 101 additions and 62 deletions.
132 changes: 81 additions & 51 deletions enclave-manager/web/src/components/enclaves/tables/PortsTable.tsx
Original file line number Diff line number Diff line change
@@ -1,103 +1,133 @@
import { ExternalLinkIcon } from "@chakra-ui/icons";
import { Flex, Icon, Link, Text, Tooltip } from "@chakra-ui/react";
import { Flex, Link, Text } from "@chakra-ui/react";
import { ColumnDef, createColumnHelper } from "@tanstack/react-table";
import { Port } from "enclave-manager-sdk/build/api_container_service_pb";
import { useMemo } from "react";
import { FiAlertTriangle } from "react-icons/fi";
import { KURTOSIS_CLOUD_HOST, KURTOSIS_CLOUD_PROTOCOL } from "../../../client/constants";
import { useKurtosisClient } from "../../../client/enclaveManager/KurtosisClientContext";
import { CopyButton } from "../../CopyButton";
import { isDefined } from "../../../utils";
import { DataTable } from "../../DataTable";
import { transportProtocolToString } from "../utils";

type PortsTableRow = {
port: { transportProtocol: string; privatePort: number; name: string };
port: {
transportProtocol: string;
privatePort: number;
publicPort: number;
name: string;
applicationProtocol: string;
};
link: string;
};
const shortUUID = (fullUUID: string) => fullUUID.substring(0, 12);

const getPortTableRows = (
instanceUUID: string,
enclaveUUID: string,
serviceUUID: string,
privatePorts: Record<string, Port>,
publicPorts: Record<string, Port>,
publicIp: string,
): PortsTableRow[] => {
return Object.entries(privatePorts).map(([name, port]) => ({
port: { transportProtocol: transportProtocolToString(port.transportProtocol), privatePort: port.number, name },
link: `${port.maybeApplicationProtocol ? port.maybeApplicationProtocol + "://" : ""}${publicIp}:${
publicPorts[name].number
}`,
}));
return Object.entries(privatePorts).map(([name, port]) => {
let link;
if (isDefined(instanceUUID) && instanceUUID.length > 0) {
link =
`${KURTOSIS_CLOUD_PROTOCOL}://` +
`${port.number}-${shortUUID(serviceUUID)}-${shortUUID(enclaveUUID)}-${shortUUID(instanceUUID)}` +
`.${KURTOSIS_CLOUD_HOST}`;
} else {
link = `${port.maybeApplicationProtocol ? port.maybeApplicationProtocol + "://" : ""}${publicIp}:${
publicPorts[name].number
}`;
}
return {
port: {
applicationProtocol: port.maybeApplicationProtocol,
transportProtocol: transportProtocolToString(port.transportProtocol),
privatePort: port.number,
publicPort: publicPorts[name].number,
name,
},
link: link,
};
});
};

const columnHelper = createColumnHelper<PortsTableRow>();

type PortsTableProps = {
instanceUUID: string;
enclaveUUID: string;
serviceUUID: string;
privatePorts: Record<string, Port>;
publicPorts: Record<string, Port>;
publicIp: string;
};

export const PortsTable = ({ privatePorts, publicPorts, publicIp }: PortsTableProps) => {
export const PortsTable = ({
instanceUUID,
enclaveUUID,
serviceUUID,
privatePorts,
publicPorts,
publicIp,
}: PortsTableProps) => {
const kurtosisClient = useKurtosisClient();

const columns = useMemo<ColumnDef<PortsTableRow, any>[]>(
() => [
columnHelper.accessor("port", {
header: "Port",
header: "Name",
cell: ({ row, getValue }) => (
<Flex flexDirection={"column"} gap={"10px"}>
<Text>{row.original.port.name || "Unknown protocol"}</Text>
<Text fontSize={"xs"} color={"gray.400"} fontWeight={"semibold"}>
{row.original.port.privatePort}/{row.original.port.transportProtocol}
<Text>
{row.original.port.applicationProtocol?.startsWith("http") ? (
<Link href={row.original.link} isExternal>
{row.original.port.name}&nbsp;
<ExternalLinkIcon mx="2px" />
</Link>
) : (
row.original.port.name
)}
</Text>
</Flex>
),
}),
columnHelper.accessor("link", {
header: "Link",
minSize: 800,
cell: ({ row }) => (
<Text width={"100%"}>
{row.original.link.startsWith("http") ? (
<Link href={row.original.link} isExternal>
{row.original.link}
<ExternalLinkIcon mx="2px" />
</Link>
) : (
row.original.link
)}
{kurtosisClient.isRunningInCloud() && (
<Tooltip
label={
"Only enclaves started using the CLI will have their ports available. This port may not work if it was started using the app."
}
shouldWrapChildren
>
<Icon m="0 10px" as={FiAlertTriangle} color={"orange.400"} />
</Tooltip>
)}
</Text>
columnHelper.accessor("port", {
header: "Private / Public Ports",
cell: ({ row, getValue }) => (
<Flex flexDirection={"column"} gap={"10px"}>
<Text>
{row.original.port.privatePort} / {row.original.port.publicPort}
</Text>
</Flex>
),
}),
columnHelper.accessor("port", {
header: "Application Protocol",
cell: ({ row, getValue }) => (
<Flex flexDirection={"column"} gap={"10px"}>
<Text>{row.original.port.applicationProtocol}</Text>
</Flex>
),
}),
columnHelper.display({
id: "copyButton",
cell: ({ row }) => (
<Flex justifyContent={"flex-end"}>
<CopyButton
contentName={"link"}
isIconButton
aria-label={"Copy this port"}
valueToCopy={`${row.original.link}`}
/>
columnHelper.accessor("port", {
header: "Transport Protocol",
cell: ({ row, getValue }) => (
<Flex flexDirection={"column"} gap={"10px"}>
<Text>{row.original.port.transportProtocol}</Text>
</Flex>
),
}),
],
[kurtosisClient],
[],
);

return (
<DataTable
columns={columns}
data={getPortTableRows(privatePorts, publicPorts, publicIp)}
data={getPortTableRows(instanceUUID, enclaveUUID, serviceUUID, privatePorts, publicPorts, publicIp)}
defaultSorting={[{ id: "port", desc: true }]}
/>
);
Expand Down
2 changes: 2 additions & 0 deletions enclave-manager/web/src/components/enclaves/utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,8 @@ export function transportProtocolToString(protocol: Port_TransportProtocol) {
return "SCTP";
case Port_TransportProtocol.UDP:
return "UDP";
default:
return "";
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,8 +11,9 @@ import { EnclaveFullInfo } from "../../types";
import { useEnclaveFromParams } from "../EnclaveRouteContext";
import { ServiceLogs } from "./logs/ServiceLogs";
import { ServiceOverview } from "./overview/ServiceOverview";
import Cookies from "js-cookie";

const tabs: { path: string; element: FunctionComponent<{ enclave: EnclaveFullInfo; service: ServiceInfo }> }[] = [
const tabs: { path: string; element: FunctionComponent<{ enclave: EnclaveFullInfo; service: ServiceInfo; instanceUUID: string }> }[] = [
{ path: "overview", element: ServiceOverview },
{ path: "logs", element: ServiceLogs },
];
Expand Down Expand Up @@ -67,6 +68,8 @@ const ServiceImpl = ({ enclave, service }: ServiceImplProps) => {
navigator(`/enclave/${enclave.shortenedUuid}/service/${service.shortenedUuid}/${tab.path}`);
};

const instanceUUID = Cookies.get("_kurtosis_instance_id") || "";

return (
<Tabs isManual isLazy index={activeIndex} onChange={handleTabChange}>
<AppPageLayout>
Expand All @@ -77,7 +80,7 @@ const ServiceImpl = ({ enclave, service }: ServiceImplProps) => {
<TabPanels>
{tabs.map((tab) => (
<TabPanel key={tab.path}>
<tab.element enclave={enclave} service={service} />
<tab.element enclave={enclave} service={service} instanceUUID={instanceUUID} />
</TabPanel>
))}
</TabPanels>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -15,9 +15,10 @@ import { EnclaveFullInfo } from "../../../types";
type ServiceOverviewProps = {
enclave: EnclaveFullInfo;
service: ServiceInfo;
instanceUUID: string;
};

export const ServiceOverview = ({ service, enclave }: ServiceOverviewProps) => {
export const ServiceOverview = ({ service, enclave, instanceUUID }: ServiceOverviewProps) => {
return (
<Flex flexDirection={"column"} gap={FLEX_STANDARD_GAP}>
<Grid templateColumns={"repeat(4, 1fr)"} gap={FLEX_STANDARD_GAP}>
Expand All @@ -44,6 +45,9 @@ export const ServiceOverview = ({ service, enclave }: ServiceOverviewProps) => {
</Grid>
<TitledBox title={"Ports"}>
<PortsTable
instanceUUID={instanceUUID}
enclaveUUID={enclave.enclaveUuid}
serviceUUID={service.serviceUuid}
privatePorts={service.privatePorts}
publicPorts={service.maybePublicPorts}
publicIp={service.maybePublicIpAddr}
Expand Down
6 changes: 3 additions & 3 deletions engine/server/webapp/asset-manifest.json
Original file line number Diff line number Diff line change
@@ -1,10 +1,10 @@
{
"files": {
"main.js": "./static/js/main.2ecb326d.js",
"main.js": "./static/js/main.de76da51.js",
"index.html": "./index.html",
"main.2ecb326d.js.map": "./static/js/main.2ecb326d.js.map"
"main.de76da51.js.map": "./static/js/main.de76da51.js.map"
},
"entrypoints": [
"static/js/main.2ecb326d.js"
"static/js/main.de76da51.js"
]
}
2 changes: 1 addition & 1 deletion engine/server/webapp/index.html
Original file line number Diff line number Diff line change
@@ -1 +1 @@
<!doctype html><html lang="en"><head><meta charset="utf-8"/><link rel="icon" href="./favicon.ico"/><meta name="viewport" content="width=device-width,initial-scale=1"/><meta name="theme-color" content="#000000"/><meta name="description" content="Kurtosis Enclave Manager"/><title>Kurtosis Enclave Manager</title><script defer="defer" src="./static/js/main.2ecb326d.js"></script></head><body><noscript>You need to enable JavaScript to run this app.</noscript><div id="root"></div></body></html>
<!doctype html><html lang="en"><head><meta charset="utf-8"/><link rel="icon" href="./favicon.ico"/><meta name="viewport" content="width=device-width,initial-scale=1"/><meta name="theme-color" content="#000000"/><meta name="description" content="Kurtosis Enclave Manager"/><title>Kurtosis Enclave Manager</title><script defer="defer" src="./static/js/main.de76da51.js"></script></head><body><noscript>You need to enable JavaScript to run this app.</noscript><div id="root"></div></body></html>
3 changes: 0 additions & 3 deletions engine/server/webapp/static/js/main.2ecb326d.js

This file was deleted.

3 changes: 3 additions & 0 deletions engine/server/webapp/static/js/main.de76da51.js

Large diffs are not rendered by default.

Large diffs are not rendered by default.

0 comments on commit 027d74b

Please sign in to comment.