// 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 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.IP}:5252/ok`, { mode: "no-cors" }) .then(() => { setIsRunningCheck(false) setCanConnectOverTS(true) }) .catch(() => setIsRunningCheck(false)) }, [auth.viewerIdentity, isRunningCheck, node.IP]) /** * 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. // If not already connected, reroute to the Tailscale IP // before sending user through check mode. window.location.href = `http://${node.IP}:5252/?check=now` } }, [node.IP, auth.viewerIdentity, newSession]) return (
{!auth.canManageNode ? "Viewing" : "Managing"} {auth.viewerIdentity && ` as ${auth.viewerIdentity.loginName}`}
{!auth.canManageNode && (!auth.viewerIdentity || auth.authNeeded === AuthType.tailscale ? ( <>

{auth.viewerIdentity ? ( <> To make changes, sign in to confirm your identity. This extra step helps us keep your device secure. ) : ( <> You can see most of this device's details. To make changes, you need to sign in. )}

) : (

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}

)}
) }