mirror of
https://github.com/tailscale/tailscale.git
synced 2025-01-10 01:53:49 +00:00
256da8dfb5
The new menu delay added to fix libdbusmenu systrays causes problems with KDE. Given the state of wildly varying systray implementations, I suspect we may need more desktop-specific hacks, so I'm setting this up to accommodate that. Updates #1708 Updates #14431 Change-Id: Ia101a4a3005adb9118051b3416f5a64a4a45987d Signed-off-by: Will Norris <will@tailscale.com>
382 lines
9.5 KiB
Go
382 lines
9.5 KiB
Go
// Copyright (c) Tailscale Inc & AUTHORS
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
//go:build cgo || !darwin
|
|
|
|
// The systray command is a minimal Tailscale systray application for Linux.
|
|
package main
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"log"
|
|
"net/http"
|
|
"os"
|
|
"runtime"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
"fyne.io/systray"
|
|
"github.com/atotto/clipboard"
|
|
dbus "github.com/godbus/dbus/v5"
|
|
"github.com/toqueteos/webbrowser"
|
|
"tailscale.com/client/tailscale"
|
|
"tailscale.com/ipn"
|
|
"tailscale.com/ipn/ipnstate"
|
|
)
|
|
|
|
var (
|
|
localClient tailscale.LocalClient
|
|
chState chan ipn.State // tailscale state changes
|
|
|
|
appIcon *os.File
|
|
|
|
// newMenuDelay is the amount of time to sleep after creating a new menu,
|
|
// but before adding items to it. This works around a bug in some dbus implementations.
|
|
newMenuDelay time.Duration
|
|
)
|
|
|
|
func main() {
|
|
systray.Run(onReady, onExit)
|
|
}
|
|
|
|
// Menu represents the systray menu, its items, and the current Tailscale state.
|
|
type Menu struct {
|
|
mu sync.Mutex // protects the entire Menu
|
|
status *ipnstate.Status
|
|
|
|
connect *systray.MenuItem
|
|
disconnect *systray.MenuItem
|
|
|
|
self *systray.MenuItem
|
|
more *systray.MenuItem
|
|
quit *systray.MenuItem
|
|
|
|
accountsCh chan ipn.ProfileID
|
|
|
|
eventCancel func() // cancel eventLoop
|
|
}
|
|
|
|
func init() {
|
|
if runtime.GOOS != "linux" {
|
|
// so far, these tweaks are only needed on Linux
|
|
return
|
|
}
|
|
|
|
desktop := strings.ToLower(os.Getenv("XDG_CURRENT_DESKTOP"))
|
|
switch desktop {
|
|
case "kde":
|
|
// KDE doesn't need a delay, and actually won't render submenus
|
|
// if we delay for more than about 400µs.
|
|
newMenuDelay = 0
|
|
default:
|
|
// Add a slight delay to ensure the menu is created before adding items.
|
|
//
|
|
// Systray implementations that use libdbusmenu sometimes process messages out of order,
|
|
// resulting in errors such as:
|
|
// (waybar:153009): LIBDBUSMENU-GTK-WARNING **: 18:07:11.551: Children but no menu, someone's been naughty with their 'children-display' property: 'submenu'
|
|
//
|
|
// See also: https://github.com/fyne-io/systray/issues/12
|
|
newMenuDelay = 100 * time.Millisecond
|
|
}
|
|
}
|
|
|
|
func onReady() {
|
|
log.Printf("starting")
|
|
ctx := context.Background()
|
|
|
|
setAppIcon(disconnected)
|
|
|
|
// dbus wants a file path for notification icons, so copy to a temp file.
|
|
appIcon, _ = os.CreateTemp("", "tailscale-systray.png")
|
|
io.Copy(appIcon, connected.renderWithBorder(3))
|
|
|
|
chState = make(chan ipn.State, 1)
|
|
|
|
menu := new(Menu)
|
|
menu.rebuild(fetchState(ctx))
|
|
|
|
go watchIPNBus(ctx)
|
|
}
|
|
|
|
type state struct {
|
|
status *ipnstate.Status
|
|
curProfile ipn.LoginProfile
|
|
allProfiles []ipn.LoginProfile
|
|
}
|
|
|
|
func fetchState(ctx context.Context) state {
|
|
status, err := localClient.Status(ctx)
|
|
if err != nil {
|
|
log.Print(err)
|
|
}
|
|
curProfile, allProfiles, err := localClient.ProfileStatus(ctx)
|
|
if err != nil {
|
|
log.Print(err)
|
|
}
|
|
return state{
|
|
status: status,
|
|
curProfile: curProfile,
|
|
allProfiles: allProfiles,
|
|
}
|
|
}
|
|
|
|
// rebuild the systray menu based on the current Tailscale state.
|
|
//
|
|
// We currently rebuild the entire menu because it is not easy to update the existing menu.
|
|
// You cannot iterate over the items in a menu, nor can you remove some items like separators.
|
|
// So for now we rebuild the whole thing, and can optimize this later if needed.
|
|
func (menu *Menu) rebuild(state state) {
|
|
menu.mu.Lock()
|
|
defer menu.mu.Unlock()
|
|
|
|
if menu.eventCancel != nil {
|
|
menu.eventCancel()
|
|
}
|
|
ctx := context.Background()
|
|
ctx, menu.eventCancel = context.WithCancel(ctx)
|
|
|
|
menu.status = state.status
|
|
systray.ResetMenu()
|
|
|
|
menu.connect = systray.AddMenuItem("Connect", "")
|
|
menu.disconnect = systray.AddMenuItem("Disconnect", "")
|
|
menu.disconnect.Hide()
|
|
systray.AddSeparator()
|
|
|
|
account := "Account"
|
|
if pt := profileTitle(state.curProfile); pt != "" {
|
|
account = pt
|
|
}
|
|
accounts := systray.AddMenuItem(account, "")
|
|
setRemoteIcon(accounts, state.curProfile.UserProfile.ProfilePicURL)
|
|
time.Sleep(newMenuDelay)
|
|
// Aggregate all clicks into a shared channel.
|
|
menu.accountsCh = make(chan ipn.ProfileID)
|
|
for _, profile := range state.allProfiles {
|
|
title := profileTitle(profile)
|
|
var item *systray.MenuItem
|
|
if profile.ID == state.curProfile.ID {
|
|
item = accounts.AddSubMenuItemCheckbox(title, "", true)
|
|
} else {
|
|
item = accounts.AddSubMenuItem(title, "")
|
|
}
|
|
setRemoteIcon(item, profile.UserProfile.ProfilePicURL)
|
|
go func(profile ipn.LoginProfile) {
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
case <-item.ClickedCh:
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
case menu.accountsCh <- profile.ID:
|
|
}
|
|
}
|
|
}
|
|
}(profile)
|
|
}
|
|
|
|
if state.status != nil && state.status.Self != nil {
|
|
title := fmt.Sprintf("This Device: %s (%s)", state.status.Self.HostName, state.status.Self.TailscaleIPs[0])
|
|
menu.self = systray.AddMenuItem(title, "")
|
|
}
|
|
systray.AddSeparator()
|
|
|
|
menu.more = systray.AddMenuItem("More settings", "")
|
|
menu.more.Enable()
|
|
|
|
menu.quit = systray.AddMenuItem("Quit", "Quit the app")
|
|
menu.quit.Enable()
|
|
|
|
go menu.eventLoop(ctx)
|
|
}
|
|
|
|
// profileTitle returns the title string for a profile menu item.
|
|
func profileTitle(profile ipn.LoginProfile) string {
|
|
title := profile.Name
|
|
if profile.NetworkProfile.DomainName != "" {
|
|
title += "\n" + profile.NetworkProfile.DomainName
|
|
}
|
|
return title
|
|
}
|
|
|
|
var (
|
|
cacheMu sync.Mutex
|
|
httpCache = map[string][]byte{} // URL => response body
|
|
)
|
|
|
|
// setRemoteIcon sets the icon for menu to the specified remote image.
|
|
// Remote images are fetched as needed and cached.
|
|
func setRemoteIcon(menu *systray.MenuItem, urlStr string) {
|
|
if menu == nil || urlStr == "" {
|
|
return
|
|
}
|
|
|
|
cacheMu.Lock()
|
|
b, ok := httpCache[urlStr]
|
|
if !ok {
|
|
resp, err := http.Get(urlStr)
|
|
if err == nil && resp.StatusCode == http.StatusOK {
|
|
b, _ = io.ReadAll(resp.Body)
|
|
httpCache[urlStr] = b
|
|
resp.Body.Close()
|
|
}
|
|
}
|
|
cacheMu.Unlock()
|
|
|
|
if len(b) > 0 {
|
|
menu.SetIcon(b)
|
|
}
|
|
}
|
|
|
|
// eventLoop is the main event loop for handling click events on menu items
|
|
// and responding to Tailscale state changes.
|
|
// This method does not return until ctx.Done is closed.
|
|
func (menu *Menu) eventLoop(ctx context.Context) {
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
case state := <-chState:
|
|
switch state {
|
|
case ipn.Running:
|
|
setAppIcon(loading)
|
|
menu.rebuild(fetchState(ctx))
|
|
setAppIcon(connected)
|
|
menu.connect.SetTitle("Connected")
|
|
menu.connect.Disable()
|
|
menu.disconnect.Show()
|
|
menu.disconnect.Enable()
|
|
case ipn.NoState, ipn.Stopped:
|
|
menu.connect.SetTitle("Connect")
|
|
menu.connect.Enable()
|
|
menu.disconnect.Hide()
|
|
setAppIcon(disconnected)
|
|
case ipn.Starting:
|
|
setAppIcon(loading)
|
|
}
|
|
case <-menu.connect.ClickedCh:
|
|
_, err := localClient.EditPrefs(ctx, &ipn.MaskedPrefs{
|
|
Prefs: ipn.Prefs{
|
|
WantRunning: true,
|
|
},
|
|
WantRunningSet: true,
|
|
})
|
|
if err != nil {
|
|
log.Print(err)
|
|
continue
|
|
}
|
|
|
|
case <-menu.disconnect.ClickedCh:
|
|
_, err := localClient.EditPrefs(ctx, &ipn.MaskedPrefs{
|
|
Prefs: ipn.Prefs{
|
|
WantRunning: false,
|
|
},
|
|
WantRunningSet: true,
|
|
})
|
|
if err != nil {
|
|
log.Printf("disconnecting: %v", err)
|
|
continue
|
|
}
|
|
|
|
case <-menu.self.ClickedCh:
|
|
copyTailscaleIP(menu.status.Self)
|
|
|
|
case <-menu.more.ClickedCh:
|
|
webbrowser.Open("http://100.100.100.100/")
|
|
|
|
case id := <-menu.accountsCh:
|
|
if err := localClient.SwitchProfile(ctx, id); err != nil {
|
|
log.Printf("failed switching to profile ID %v: %v", id, err)
|
|
}
|
|
|
|
case <-menu.quit.ClickedCh:
|
|
systray.Quit()
|
|
}
|
|
}
|
|
}
|
|
|
|
// watchIPNBus subscribes to the tailscale event bus and sends state updates to chState.
|
|
// This method does not return.
|
|
func watchIPNBus(ctx context.Context) {
|
|
for {
|
|
if err := watchIPNBusInner(ctx); err != nil {
|
|
log.Println(err)
|
|
if errors.Is(err, context.Canceled) {
|
|
// If the context got canceled, we will never be able to
|
|
// reconnect to IPN bus, so exit the process.
|
|
log.Fatalf("watchIPNBus: %v", err)
|
|
}
|
|
}
|
|
// If our watch connection breaks, wait a bit before reconnecting. No
|
|
// reason to spam the logs if e.g. tailscaled is restarting or goes
|
|
// down.
|
|
time.Sleep(3 * time.Second)
|
|
}
|
|
}
|
|
|
|
func watchIPNBusInner(ctx context.Context) error {
|
|
watcher, err := localClient.WatchIPNBus(ctx, ipn.NotifyInitialState|ipn.NotifyNoPrivateKeys)
|
|
if err != nil {
|
|
return fmt.Errorf("watching ipn bus: %w", err)
|
|
}
|
|
defer watcher.Close()
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return nil
|
|
default:
|
|
n, err := watcher.Next()
|
|
if err != nil {
|
|
return fmt.Errorf("ipnbus error: %w", err)
|
|
}
|
|
if n.State != nil {
|
|
chState <- *n.State
|
|
log.Printf("new state: %v", n.State)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// copyTailscaleIP copies the first Tailscale IP of the given device to the clipboard
|
|
// and sends a notification with the copied value.
|
|
func copyTailscaleIP(device *ipnstate.PeerStatus) {
|
|
if device == nil || len(device.TailscaleIPs) == 0 {
|
|
return
|
|
}
|
|
name := strings.Split(device.DNSName, ".")[0]
|
|
ip := device.TailscaleIPs[0].String()
|
|
err := clipboard.WriteAll(ip)
|
|
if err != nil {
|
|
log.Printf("clipboard error: %v", err)
|
|
}
|
|
|
|
sendNotification(fmt.Sprintf("Copied Address for %v", name), ip)
|
|
}
|
|
|
|
// sendNotification sends a desktop notification with the given title and content.
|
|
func sendNotification(title, content string) {
|
|
conn, err := dbus.SessionBus()
|
|
if err != nil {
|
|
log.Printf("dbus: %v", err)
|
|
return
|
|
}
|
|
timeout := 3 * time.Second
|
|
obj := conn.Object("org.freedesktop.Notifications", "/org/freedesktop/Notifications")
|
|
call := obj.Call("org.freedesktop.Notifications.Notify", 0, "Tailscale", uint32(0),
|
|
appIcon.Name(), title, content, []string{}, map[string]dbus.Variant{}, int32(timeout.Milliseconds()))
|
|
if call.Err != nil {
|
|
log.Printf("dbus: %v", call.Err)
|
|
}
|
|
}
|
|
|
|
func onExit() {
|
|
log.Printf("exiting")
|
|
os.Remove(appIcon.Name())
|
|
}
|