feat(frontend): add WinBox tunnel button component
This commit is contained in:
129
frontend/src/components/fleet/WinBoxButton.tsx
Normal file
129
frontend/src/components/fleet/WinBoxButton.tsx
Normal file
@@ -0,0 +1,129 @@
|
|||||||
|
import { useState } from 'react'
|
||||||
|
import { useMutation } from '@tanstack/react-query'
|
||||||
|
import { Monitor, Copy, X, Loader2 } from 'lucide-react'
|
||||||
|
import { remoteAccessApi } from '@/lib/api'
|
||||||
|
|
||||||
|
interface WinBoxButtonProps {
|
||||||
|
tenantId: string
|
||||||
|
deviceId: string
|
||||||
|
}
|
||||||
|
|
||||||
|
type State = 'idle' | 'requesting' | 'ready' | 'closing' | 'error'
|
||||||
|
|
||||||
|
export function WinBoxButton({ tenantId, deviceId }: WinBoxButtonProps) {
|
||||||
|
const [state, setState] = useState<State>('idle')
|
||||||
|
const [tunnelInfo, setTunnelInfo] = useState<{
|
||||||
|
tunnel_id: string
|
||||||
|
host: string
|
||||||
|
port: number
|
||||||
|
winbox_uri: string
|
||||||
|
} | null>(null)
|
||||||
|
const [error, setError] = useState<string | null>(null)
|
||||||
|
const [copied, setCopied] = useState(false)
|
||||||
|
|
||||||
|
const openMutation = useMutation({
|
||||||
|
mutationFn: () => remoteAccessApi.openWinbox(tenantId, deviceId),
|
||||||
|
onSuccess: (data) => {
|
||||||
|
setTunnelInfo(data)
|
||||||
|
setState('ready')
|
||||||
|
|
||||||
|
// Attempt deep link on Windows only
|
||||||
|
if (navigator.userAgent.includes('Windows')) {
|
||||||
|
window.open(data.winbox_uri, '_blank')
|
||||||
|
}
|
||||||
|
},
|
||||||
|
onError: (err: any) => {
|
||||||
|
setState('error')
|
||||||
|
setError(err.response?.data?.detail || 'Failed to open tunnel')
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const closeMutation = useMutation({
|
||||||
|
mutationFn: () => {
|
||||||
|
if (!tunnelInfo) throw new Error('No tunnel')
|
||||||
|
return remoteAccessApi.closeWinbox(tenantId, deviceId, tunnelInfo.tunnel_id)
|
||||||
|
},
|
||||||
|
onSuccess: () => {
|
||||||
|
setState('idle')
|
||||||
|
setTunnelInfo(null)
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
const copyAddress = async () => {
|
||||||
|
if (!tunnelInfo) return
|
||||||
|
const addr = `${tunnelInfo.host}:${tunnelInfo.port}`
|
||||||
|
try {
|
||||||
|
await navigator.clipboard.writeText(addr)
|
||||||
|
} catch {
|
||||||
|
// Fallback for HTTP
|
||||||
|
const ta = document.createElement('textarea')
|
||||||
|
ta.value = addr
|
||||||
|
document.body.appendChild(ta)
|
||||||
|
ta.select()
|
||||||
|
document.execCommand('copy')
|
||||||
|
document.body.removeChild(ta)
|
||||||
|
}
|
||||||
|
setCopied(true)
|
||||||
|
setTimeout(() => setCopied(false), 2000)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (state === 'idle' || state === 'error') {
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<button
|
||||||
|
onClick={() => {
|
||||||
|
setState('requesting')
|
||||||
|
setError(null)
|
||||||
|
openMutation.mutate()
|
||||||
|
}}
|
||||||
|
disabled={openMutation.isPending}
|
||||||
|
className="inline-flex items-center gap-2 px-4 py-2 rounded-md bg-primary text-primary-foreground hover:bg-primary/90 disabled:opacity-50"
|
||||||
|
>
|
||||||
|
{openMutation.isPending ? (
|
||||||
|
<Loader2 className="h-4 w-4 animate-spin" />
|
||||||
|
) : (
|
||||||
|
<Monitor className="h-4 w-4" />
|
||||||
|
)}
|
||||||
|
{openMutation.isPending ? 'Connecting...' : 'Open WinBox'}
|
||||||
|
</button>
|
||||||
|
{error && <p className="mt-2 text-sm text-destructive">{error}</p>}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (state === 'ready' && tunnelInfo) {
|
||||||
|
return (
|
||||||
|
<div className="rounded-md border p-4 space-y-3">
|
||||||
|
<p className="font-medium text-sm">WinBox tunnel ready</p>
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
Connect to: <code className="font-mono">{tunnelInfo.host}:{tunnelInfo.port}</code>
|
||||||
|
</p>
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<button
|
||||||
|
onClick={copyAddress}
|
||||||
|
className="inline-flex items-center gap-2 px-3 py-1.5 text-sm rounded-md border hover:bg-accent"
|
||||||
|
>
|
||||||
|
<Copy className="h-3 w-3" />
|
||||||
|
{copied ? 'Copied!' : 'Copy Address'}
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
onClick={() => {
|
||||||
|
setState('closing')
|
||||||
|
closeMutation.mutate()
|
||||||
|
}}
|
||||||
|
disabled={closeMutation.isPending}
|
||||||
|
className="inline-flex items-center gap-2 px-3 py-1.5 text-sm rounded-md border hover:bg-accent disabled:opacity-50"
|
||||||
|
>
|
||||||
|
<X className="h-3 w-3" />
|
||||||
|
Close Tunnel
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<p className="text-xs text-muted-foreground">
|
||||||
|
Tunnel closes after 5 min of inactivity
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return null
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user