// Copyright (c) Tailscale Inc & AUTHORS // SPDX-License-Identifier: BSD-3-Clause import cx from "classnames" import React, { useCallback, useEffect, useState } from "react" import { ReactComponent as ChevronDown } from "src/assets/icons/chevron-down.svg" import { ReactComponent as Eye } from "src/assets/icons/eye.svg" import { ReactComponent as User } from "src/assets/icons/user.svg" import { AuthResponse, AuthType } from "src/hooks/auth" import { NodeData } from "src/hooks/node-data" import Button from "src/ui/button" import Popover from "src/ui/popover" import ProfilePic from "src/ui/profile-pic" export default function LoginToggle({ node, auth, newSession, }: { node: NodeData auth: AuthResponse newSession: () => Promise }) { const [open, setOpen] = useState(false) return ( } side="bottom" align="end" open={open} onOpenChange={setOpen} asChild > {!auth.canManageNode ? ( ) : (
)}
) } function LoginPopoverContent({ node, auth, newSession, }: { node: NodeData auth: AuthResponse newSession: () => Promise }) { /** * canConnectOverTS indicates whether the current viewer * is able to hit the node's web client that's being served * at http://${node.IP}:5252. If false, this means that the * viewer must connect to the correct tailnet before being * able to sign in. */ const [canConnectOverTS, setCanConnectOverTS] = useState(false) const [isRunningCheck, setIsRunningCheck] = useState(false) const checkTSConnection = useCallback(() => { if (auth.viewerIdentity) { setCanConnectOverTS(true) // already connected over ts return } // Otherwise, test connection to the ts IP. if (isRunningCheck) { return // already checking } setIsRunningCheck(true) fetch(`http://${node.IPv4}:5252/ok`, { mode: "no-cors" }) .then(() => { setIsRunningCheck(false) setCanConnectOverTS(true) }) .catch(() => setIsRunningCheck(false)) }, [auth.viewerIdentity, isRunningCheck, node.IPv4]) /** * Checking connection for first time on page load. * * While not connected, we check again whenever the mouse * enters the popover component, to pick up on the user * leaving to turn on Tailscale then returning to the view. * See `onMouseEnter` on the div below. */ // eslint-disable-next-line react-hooks/exhaustive-deps useEffect(() => checkTSConnection(), []) const handleSignInClick = useCallback(() => { if (auth.viewerIdentity) { newSession() } else { // Must be connected over Tailscale to log in. // Send user to Tailscale IP and start check mode const manageURL = `http://${node.IPv4}:5252/?check=now` if (window.self !== window.top) { // if we're inside an iframe, open management client in new window window.open(manageURL, "_blank") } else { window.location.href = manageURL } } }, [node.IPv4, auth.viewerIdentity, newSession]) return (
{!auth.canManageNode ? "Viewing" : "Managing"} {auth.viewerIdentity && ` as ${auth.viewerIdentity.loginName}`}
{!auth.canManageNode && ( <> {!auth.viewerIdentity ? ( // User is not connected over Tailscale. // These states are only possible on the login client. <> {!canConnectOverTS ? ( <>

{!node.ACLAllowsAnyIncomingTraffic ? ( // Tailnet ACLs don't allow access. <> The current tailnet policy file does not allow connecting to this device. ) : ( // ACLs allow access, but user can't connect. <> Cannot access this device's Tailscale IP. Make sure you are connected to your tailnet, and that your policy file allows access. )}{" "} Learn more →

) : ( // User can connect to Tailcale IP; sign in when ready. <>

You can see most of this device's details. To make changes, you need to sign in.

)} ) : auth.authNeeded === AuthType.tailscale ? ( // User is connected over Tailscale, but needs to complete check mode. <>

To make changes, sign in to confirm your identity. This extra step helps us keep your device secure.

) : ( // User is connected over tailscale, but doesn't have permission to manage.

You don’t have permission to make changes to this device, but you can view most of its details.

)} )} {auth.viewerIdentity && ( <>

We recognize you because you are accessing this page from{" "} {auth.viewerIdentity.nodeName || auth.viewerIdentity.nodeIP}

)}
) } function SignInButton({ auth, onClick, }: { auth: AuthResponse onClick: () => void }) { return ( ) }