2023-11-08 17:33:27 -05:00
|
|
|
|
import cx from "classnames"
|
2023-11-06 19:01:16 -05:00
|
|
|
|
import React from "react"
|
2023-11-08 17:33:27 -05:00
|
|
|
|
import { apiFetch } from "src/api"
|
2023-11-15 16:04:44 -05:00
|
|
|
|
import { UpdateAvailableNotification } from "src/components/update-available"
|
2023-11-06 19:01:16 -05:00
|
|
|
|
import { NodeData } from "src/hooks/node-data"
|
2023-11-08 17:33:27 -05:00
|
|
|
|
import { useLocation } from "wouter"
|
|
|
|
|
import ACLTag from "../acl-tag"
|
2023-11-06 19:01:16 -05:00
|
|
|
|
|
2023-11-09 16:19:22 -05:00
|
|
|
|
export default function DeviceDetailsView({
|
|
|
|
|
readonly,
|
|
|
|
|
node,
|
|
|
|
|
}: {
|
|
|
|
|
readonly: boolean
|
|
|
|
|
node: NodeData
|
|
|
|
|
}) {
|
2023-11-08 17:33:27 -05:00
|
|
|
|
const [, setLocation] = useLocation()
|
|
|
|
|
|
2023-11-06 19:01:16 -05:00
|
|
|
|
return (
|
2023-11-13 14:54:24 -05:00
|
|
|
|
<>
|
2023-11-06 19:01:16 -05:00
|
|
|
|
<h1 className="mb-10">Device details</h1>
|
|
|
|
|
<div className="flex flex-col gap-4">
|
|
|
|
|
<div className="card">
|
|
|
|
|
<div className="flex items-center justify-between">
|
|
|
|
|
<div className="flex items-center gap-2">
|
|
|
|
|
<h1>{node.DeviceName}</h1>
|
2023-11-08 17:33:27 -05:00
|
|
|
|
<div
|
|
|
|
|
className={cx("w-2.5 h-2.5 rounded-full", {
|
|
|
|
|
"bg-emerald-500": node.Status === "Running",
|
|
|
|
|
"bg-gray-300": node.Status !== "Running",
|
|
|
|
|
})}
|
|
|
|
|
/>
|
2023-11-06 19:01:16 -05:00
|
|
|
|
</div>
|
2023-11-08 17:33:27 -05:00
|
|
|
|
<button
|
2023-11-09 16:19:22 -05:00
|
|
|
|
className={cx(
|
|
|
|
|
"px-3 py-2 bg-stone-50 rounded shadow border border-stone-200 text-neutral-800 text-sm font-medium",
|
|
|
|
|
{ "cursor-not-allowed": readonly }
|
|
|
|
|
)}
|
2023-11-08 17:33:27 -05:00
|
|
|
|
onClick={() =>
|
|
|
|
|
apiFetch("/local/v0/logout", "POST")
|
|
|
|
|
.then(() => setLocation("/"))
|
|
|
|
|
.catch((err) => alert("Logout failed: " + err.message))
|
|
|
|
|
}
|
2023-11-09 16:19:22 -05:00
|
|
|
|
disabled={readonly}
|
2023-11-08 17:33:27 -05:00
|
|
|
|
>
|
|
|
|
|
Disconnect…
|
2023-11-06 19:01:16 -05:00
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
2023-11-15 16:04:44 -05:00
|
|
|
|
{node.ClientVersion &&
|
|
|
|
|
!node.ClientVersion.RunningLatest &&
|
|
|
|
|
!readonly && (
|
|
|
|
|
<UpdateAvailableNotification details={node.ClientVersion} />
|
|
|
|
|
)}
|
2023-11-06 19:01:16 -05:00
|
|
|
|
<div className="card">
|
|
|
|
|
<h2 className="mb-2">General</h2>
|
|
|
|
|
<table>
|
|
|
|
|
<tbody>
|
2023-11-08 17:33:27 -05:00
|
|
|
|
<tr className="flex">
|
2023-11-06 19:01:16 -05:00
|
|
|
|
<td>Managed by</td>
|
2023-11-08 17:33:27 -05:00
|
|
|
|
<td className="flex gap-1 flex-wrap">
|
|
|
|
|
{node.IsTagged
|
|
|
|
|
? node.Tags.map((t) => <ACLTag key={t} tag={t} />)
|
|
|
|
|
: node.Profile.DisplayName}
|
|
|
|
|
</td>
|
2023-11-06 19:01:16 -05:00
|
|
|
|
</tr>
|
|
|
|
|
<tr>
|
|
|
|
|
<td>Machine name</td>
|
|
|
|
|
<td>{node.DeviceName}</td>
|
|
|
|
|
</tr>
|
|
|
|
|
<tr>
|
|
|
|
|
<td>OS</td>
|
2023-11-08 17:33:27 -05:00
|
|
|
|
<td>{node.OS}</td>
|
2023-11-06 19:01:16 -05:00
|
|
|
|
</tr>
|
|
|
|
|
<tr>
|
|
|
|
|
<td>ID</td>
|
2023-11-08 17:33:27 -05:00
|
|
|
|
<td>{node.ID}</td>
|
2023-11-06 19:01:16 -05:00
|
|
|
|
</tr>
|
|
|
|
|
<tr>
|
|
|
|
|
<td>Tailscale version</td>
|
|
|
|
|
<td>{node.IPNVersion}</td>
|
|
|
|
|
</tr>
|
|
|
|
|
<tr>
|
|
|
|
|
<td>Key expiry</td>
|
2023-11-08 17:33:27 -05:00
|
|
|
|
<td>
|
|
|
|
|
{node.KeyExpired
|
|
|
|
|
? "Expired"
|
|
|
|
|
: // TODO: present as relative expiry (e.g. "5 months from now")
|
|
|
|
|
new Date(node.KeyExpiry).toLocaleString()}
|
|
|
|
|
</td>
|
2023-11-06 19:01:16 -05:00
|
|
|
|
</tr>
|
|
|
|
|
</tbody>
|
|
|
|
|
</table>
|
|
|
|
|
</div>
|
|
|
|
|
<div className="card">
|
|
|
|
|
<h2 className="mb-2">Addresses</h2>
|
|
|
|
|
<table>
|
|
|
|
|
<tbody>
|
|
|
|
|
<tr>
|
|
|
|
|
<td>Tailscale IPv4</td>
|
|
|
|
|
<td>{node.IP}</td>
|
|
|
|
|
</tr>
|
|
|
|
|
<tr>
|
|
|
|
|
<td>Tailscale IPv6</td>
|
2023-11-08 17:33:27 -05:00
|
|
|
|
<td>{node.IPv6}</td>
|
2023-11-06 19:01:16 -05:00
|
|
|
|
</tr>
|
|
|
|
|
<tr>
|
|
|
|
|
<td>Short domain</td>
|
|
|
|
|
<td>{node.DeviceName}</td>
|
|
|
|
|
</tr>
|
|
|
|
|
<tr>
|
|
|
|
|
<td>Full domain</td>
|
2023-11-08 17:33:27 -05:00
|
|
|
|
<td>
|
|
|
|
|
{node.DeviceName}.{node.TailnetName}
|
|
|
|
|
</td>
|
2023-11-06 19:01:16 -05:00
|
|
|
|
</tr>
|
|
|
|
|
</tbody>
|
|
|
|
|
</table>
|
|
|
|
|
</div>
|
|
|
|
|
<p className="text-neutral-500 text-sm leading-tight text-center">
|
|
|
|
|
Want even more details? Visit{" "}
|
|
|
|
|
<a
|
|
|
|
|
// TODO: pipe control serve url from backend
|
|
|
|
|
href="https://login.tailscale.com/admin"
|
|
|
|
|
target="_blank"
|
|
|
|
|
className="text-indigo-700 text-sm"
|
|
|
|
|
>
|
|
|
|
|
this device’s page
|
|
|
|
|
</a>{" "}
|
|
|
|
|
in the admin console.
|
|
|
|
|
</p>
|
|
|
|
|
</div>
|
2023-11-13 14:54:24 -05:00
|
|
|
|
</>
|
2023-11-06 19:01:16 -05:00
|
|
|
|
)
|
|
|
|
|
}
|