diff --git a/extensions/tailscale/CHANGELOG.md b/extensions/tailscale/CHANGELOG.md index 28707715a25a3..0667584f43643 100644 --- a/extensions/tailscale/CHANGELOG.md +++ b/extensions/tailscale/CHANGELOG.md @@ -1,5 +1,9 @@ # Tailscale Changelog +## [Add new features] - 2025-01-02 + +- Add `netcheck` command + ## [Improvement] - 2024-11-08 - Provide UI indicator, and HUD message on `connect (tailscale up)` and `disconnect (tailscale down)` commands diff --git a/extensions/tailscale/package.json b/extensions/tailscale/package.json index e29cc2d48208f..7a74ba853fad6 100644 --- a/extensions/tailscale/package.json +++ b/extensions/tailscale/package.json @@ -13,7 +13,8 @@ "itsmingjie", "erics118", "brandenw", - "j3lte" + "j3lte", + "itsmatteomanf" ], "categories": [ "Developer Tools", @@ -90,6 +91,13 @@ "subtitle": "Tailscale", "description": "Tailscale down", "mode": "no-view" + }, + { + "name": "netcheck", + "title": "Netcheck", + "subtitle": "Tailscale", + "description": "Tailscale netcheck", + "mode": "view" } ], "preferences": [ diff --git a/extensions/tailscale/src/netcheck.tsx b/extensions/tailscale/src/netcheck.tsx new file mode 100644 index 0000000000000..75433f0ef759b --- /dev/null +++ b/extensions/tailscale/src/netcheck.tsx @@ -0,0 +1,151 @@ +import { List, Icon, ActionPanel, Action, Color } from "@raycast/api"; +import { useEffect, useState } from "react"; +import { getErrorDetails, ErrorDetails, getNetcheck, getDerpMap, NetcheckResponse, Derp } from "./shared"; + +const getDerps = (netcheck: NetcheckResponse, setPreferredDERP: (derp: Derp) => void) => { + // the netcheck json response contains a map of region IDs to latencies, so we need to map them to the actual region data + // the pretty printed command does that behind the scenes + const _derpMap = getDerpMap(); + return Object.keys(netcheck.RegionLatency) + .map((key) => { + const _key = parseInt(key); + const derp = { + id: key, + code: _derpMap[_key].RegionCode, + name: _derpMap[_key].RegionName, + latency: netcheck.RegionLatency[key] ? (netcheck.RegionLatency[key] / 1000000).toFixed(1) : undefined, + latencies: { + v4: netcheck.RegionV4Latency[key] ? (netcheck.RegionV4Latency[key] / 1000000).toFixed(1) : undefined, + v6: netcheck.RegionV6Latency[key] ? (netcheck.RegionV6Latency[key] / 1000000).toFixed(1) : undefined, + }, + nodes: _derpMap[_key].Nodes, + }; + netcheck.PreferredDERP === _key && setPreferredDERP(derp); + return derp; + }) + .sort((a, b) => { + if (a.latency && b.latency) return parseFloat(a.latency) - parseFloat(b.latency); + if (a.latency) return 1; + if (b.latency) return -1; + return 0; + }); +}; + +export default function MyDeviceList() { + const [netcheck, setNetcheck] = useState(); + const [portMappings, setPortMappings] = useState([]); + const [derps, setDerps] = useState([]); + const [preferredDERP, setPreferredDERP] = useState(); + const [error, setError] = useState(); + + useEffect(() => { + async function fetch() { + try { + const _netcheck = getNetcheck(); + setDerps(getDerps(_netcheck, setPreferredDERP)); + setNetcheck(_netcheck); + setPortMappings( + ["UPnP", "PMP", "PCP"].reduce((acc, mapping) => { + _netcheck[mapping] && acc.push(mapping); + return acc; + }, [] as string[]), + ); + } catch (error) { + setError(getErrorDetails(error, "Couldn’t load netcheck.")); + } + } + fetch(); + }, []); + + return ( + + {error ? ( + + ) : ( + <> + + + {netcheck?.IPv4 && } + {netcheck?.IPv4 && } + + } + /> + + {netcheck?.IPv6 && } + {netcheck?.IPv6 && } + + } + /> + + + + + {derps.map((derp) => ( + + + + + {derp.latency && } + + } + /> + ))} + + + )} + + ); +} + +export function derpDetails(derp: Derp) { + return ( + + + + + + {derp.latencies.v4 && } + {derp.latencies.v6 && } + + + {derp.nodes?.map((node) => ( + + + {node.IPv4 && } + {node.IPv6 && } + + } + /> + ))} + + + ); +} diff --git a/extensions/tailscale/src/shared.tsx b/extensions/tailscale/src/shared.tsx index 19b973cc6310d..0f0303142ea96 100644 --- a/extensions/tailscale/src/shared.tsx +++ b/extensions/tailscale/src/shared.tsx @@ -60,6 +60,56 @@ export type StatusResponse = { >; }; +/** + * NetcheckResponse are the fields returned by `tailscale netcheck --format json`. + * These are mentioned to not be stable and may change in the future. Doubtful, but possible. + */ +export type NetcheckResponse = { + UDP: boolean; + IPv4: boolean; + GlobalV4: string; + IPv6: boolean; + GlobalV6: string; + MappingVariesByDestIP: boolean; + UPnP: boolean; + PMP: boolean; + PCP: boolean; + PreferredDERP: number; + RegionLatency: Record; + RegionV4Latency: Record; + RegionV6Latency: Record; +}; + +export type DerpRegion = { + RegionId: number; + RegionCode: string; + RegionName: string; + Latitude: number; + Longitude: number; + Nodes: DerpNode[]; +}; + +type DerpNode = { + Name: string; + RegionID: number; + HostName: string; + IPv4: string; + IPv6: string; + CanPort80: boolean; +}; + +export type Derp = { + id: string; + code: string; + name: string; + latency: string | undefined; + latencies: { + v4: string | undefined; + v6: string | undefined; + }; + nodes: DerpNode[]; +}; + export function getStatus(peers = true) { const resp = tailscale(`status --json --peers=${peers}`); const data = JSON.parse(resp) as StatusResponse; @@ -69,6 +119,19 @@ export function getStatus(peers = true) { return data; } +export function getNetcheck() { + const resp = tailscale("netcheck --format json"); + return JSON.parse(resp); +} + +/** + * This funtion relies on a debug command, so it may not be stable on the returned value. + */ +export function getDerpMap() { + const resp = tailscale("debug netmap"); + return JSON.parse(resp).DERPMap.Regions as DerpRegion[]; +} + export function getDevices(status: StatusResponse) { const devices: Device[] = []; const self = status.Self;