cmd/systray: add a basic linux systray app
This adds a systray app for linux, similar to the apps for macOS and
windows. There are already a number of community-developed systray apps,
but most of them are either long abandoned, are built for a specific
desktop environment, or simply wrap the tailscale CLI.
This uses fyne.io/systray (a fork of github.com/getlantern/systray)
which uses newer D-Bus specifications to render the tray icon and menu.
This results in a pretty broad support for modern desktop environments.
This initial commit lacks a number of features like profile switching,
device listing, and exit node selection. This is really focused on the
application structure, the interaction with LocalAPI, and some system
integration pieces like the app icon, notifications, and the clipboard.
Updates #1708
Signed-off-by: Will Norris <will@tailscale.com>
2024-07-10 13:45:10 -07:00
// 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"
2024-09-05 12:11:05 -05:00
"errors"
cmd/systray: add a basic linux systray app
This adds a systray app for linux, similar to the apps for macOS and
windows. There are already a number of community-developed systray apps,
but most of them are either long abandoned, are built for a specific
desktop environment, or simply wrap the tailscale CLI.
This uses fyne.io/systray (a fork of github.com/getlantern/systray)
which uses newer D-Bus specifications to render the tray icon and menu.
This results in a pretty broad support for modern desktop environments.
This initial commit lacks a number of features like profile switching,
device listing, and exit node selection. This is really focused on the
application structure, the interaction with LocalAPI, and some system
integration pieces like the app icon, notifications, and the clipboard.
Updates #1708
Signed-off-by: Will Norris <will@tailscale.com>
2024-07-10 13:45:10 -07:00
"fmt"
"io"
"log"
"os"
"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
)
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
2024-12-10 13:54:31 -08:00
accountsCh chan ipn . ProfileID
cmd/systray: add a basic linux systray app
This adds a systray app for linux, similar to the apps for macOS and
windows. There are already a number of community-developed systray apps,
but most of them are either long abandoned, are built for a specific
desktop environment, or simply wrap the tailscale CLI.
This uses fyne.io/systray (a fork of github.com/getlantern/systray)
which uses newer D-Bus specifications to render the tray icon and menu.
This results in a pretty broad support for modern desktop environments.
This initial commit lacks a number of features like profile switching,
device listing, and exit node selection. This is really focused on the
application structure, the interaction with LocalAPI, and some system
integration pieces like the app icon, notifications, and the clipboard.
Updates #1708
Signed-off-by: Will Norris <will@tailscale.com>
2024-07-10 13:45:10 -07:00
eventCancel func ( ) // cancel eventLoop
}
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 . render ( ) )
chState = make ( chan ipn . State , 1 )
2024-12-10 13:54:31 -08:00
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 {
cmd/systray: add a basic linux systray app
This adds a systray app for linux, similar to the apps for macOS and
windows. There are already a number of community-developed systray apps,
but most of them are either long abandoned, are built for a specific
desktop environment, or simply wrap the tailscale CLI.
This uses fyne.io/systray (a fork of github.com/getlantern/systray)
which uses newer D-Bus specifications to render the tray icon and menu.
This results in a pretty broad support for modern desktop environments.
This initial commit lacks a number of features like profile switching,
device listing, and exit node selection. This is really focused on the
application structure, the interaction with LocalAPI, and some system
integration pieces like the app icon, notifications, and the clipboard.
Updates #1708
Signed-off-by: Will Norris <will@tailscale.com>
2024-07-10 13:45:10 -07:00
status , err := localClient . Status ( ctx )
if err != nil {
log . Print ( err )
}
2024-12-10 13:54:31 -08:00
curProfile , allProfiles , err := localClient . ProfileStatus ( ctx )
if err != nil {
log . Print ( err )
}
return state {
status : status ,
curProfile : curProfile ,
allProfiles : allProfiles ,
}
cmd/systray: add a basic linux systray app
This adds a systray app for linux, similar to the apps for macOS and
windows. There are already a number of community-developed systray apps,
but most of them are either long abandoned, are built for a specific
desktop environment, or simply wrap the tailscale CLI.
This uses fyne.io/systray (a fork of github.com/getlantern/systray)
which uses newer D-Bus specifications to render the tray icon and menu.
This results in a pretty broad support for modern desktop environments.
This initial commit lacks a number of features like profile switching,
device listing, and exit node selection. This is really focused on the
application structure, the interaction with LocalAPI, and some system
integration pieces like the app icon, notifications, and the clipboard.
Updates #1708
Signed-off-by: Will Norris <will@tailscale.com>
2024-07-10 13:45:10 -07:00
}
// 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.
2024-12-10 13:54:31 -08:00
func ( menu * Menu ) rebuild ( state state ) {
cmd/systray: add a basic linux systray app
This adds a systray app for linux, similar to the apps for macOS and
windows. There are already a number of community-developed systray apps,
but most of them are either long abandoned, are built for a specific
desktop environment, or simply wrap the tailscale CLI.
This uses fyne.io/systray (a fork of github.com/getlantern/systray)
which uses newer D-Bus specifications to render the tray icon and menu.
This results in a pretty broad support for modern desktop environments.
This initial commit lacks a number of features like profile switching,
device listing, and exit node selection. This is really focused on the
application structure, the interaction with LocalAPI, and some system
integration pieces like the app icon, notifications, and the clipboard.
Updates #1708
Signed-off-by: Will Norris <will@tailscale.com>
2024-07-10 13:45:10 -07:00
menu . mu . Lock ( )
defer menu . mu . Unlock ( )
if menu . eventCancel != nil {
menu . eventCancel ( )
}
2024-12-10 13:54:31 -08:00
ctx := context . Background ( )
ctx , menu . eventCancel = context . WithCancel ( ctx )
menu . status = state . status
cmd/systray: add a basic linux systray app
This adds a systray app for linux, similar to the apps for macOS and
windows. There are already a number of community-developed systray apps,
but most of them are either long abandoned, are built for a specific
desktop environment, or simply wrap the tailscale CLI.
This uses fyne.io/systray (a fork of github.com/getlantern/systray)
which uses newer D-Bus specifications to render the tray icon and menu.
This results in a pretty broad support for modern desktop environments.
This initial commit lacks a number of features like profile switching,
device listing, and exit node selection. This is really focused on the
application structure, the interaction with LocalAPI, and some system
integration pieces like the app icon, notifications, and the clipboard.
Updates #1708
Signed-off-by: Will Norris <will@tailscale.com>
2024-07-10 13:45:10 -07:00
systray . ResetMenu ( )
menu . connect = systray . AddMenuItem ( "Connect" , "" )
menu . disconnect = systray . AddMenuItem ( "Disconnect" , "" )
menu . disconnect . Hide ( )
systray . AddSeparator ( )
2024-12-10 13:54:31 -08:00
account := "Account"
if state . curProfile . Name != "" {
account += fmt . Sprintf ( " (%s)" , state . curProfile . Name )
}
accounts := systray . AddMenuItem ( account , "" )
// The dbus message about this menu item must propagate to the receiving
// end before we attach any submenu items. Otherwise the receiver may not
// yet record the parent menu item and error out.
//
// On waybar with libdbusmenu-gtk, this manifests as the following warning:
// (waybar:153009): LIBDBUSMENU-GTK-WARNING **: 18:07:11.551: Children but no menu, someone's been naughty with their 'children-display' property: 'submenu'
time . Sleep ( 100 * time . Millisecond )
// Aggregate all clicks into a shared channel.
menu . accountsCh = make ( chan ipn . ProfileID )
for _ , profile := range state . allProfiles {
title := fmt . Sprintf ( "%s (%s)" , profile . Name , profile . NetworkProfile . DomainName )
// Note: we could use AddSubMenuItemCheckbox instead of this formatting
// hack, but checkboxes don't work across all desktops unfortunately.
if profile . ID == state . curProfile . ID {
title = "* " + title
}
item := accounts . AddSubMenuItem ( title , "" )
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 ] )
cmd/systray: add a basic linux systray app
This adds a systray app for linux, similar to the apps for macOS and
windows. There are already a number of community-developed systray apps,
but most of them are either long abandoned, are built for a specific
desktop environment, or simply wrap the tailscale CLI.
This uses fyne.io/systray (a fork of github.com/getlantern/systray)
which uses newer D-Bus specifications to render the tray icon and menu.
This results in a pretty broad support for modern desktop environments.
This initial commit lacks a number of features like profile switching,
device listing, and exit node selection. This is really focused on the
application structure, the interaction with LocalAPI, and some system
integration pieces like the app icon, notifications, and the clipboard.
Updates #1708
Signed-off-by: Will Norris <will@tailscale.com>
2024-07-10 13:45:10 -07:00
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 )
}
// 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 )
2024-12-10 13:54:31 -08:00
menu . rebuild ( fetchState ( ctx ) )
cmd/systray: add a basic linux systray app
This adds a systray app for linux, similar to the apps for macOS and
windows. There are already a number of community-developed systray apps,
but most of them are either long abandoned, are built for a specific
desktop environment, or simply wrap the tailscale CLI.
This uses fyne.io/systray (a fork of github.com/getlantern/systray)
which uses newer D-Bus specifications to render the tray icon and menu.
This results in a pretty broad support for modern desktop environments.
This initial commit lacks a number of features like profile switching,
device listing, and exit node selection. This is really focused on the
application structure, the interaction with LocalAPI, and some system
integration pieces like the app icon, notifications, and the clipboard.
Updates #1708
Signed-off-by: Will Norris <will@tailscale.com>
2024-07-10 13:45:10 -07:00
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/" )
2024-12-10 13:54:31 -08:00
case id := <- menu . accountsCh :
if err := localClient . SwitchProfile ( ctx , id ) ; err != nil {
log . Printf ( "failed switching to profile ID %v: %v" , id , err )
}
cmd/systray: add a basic linux systray app
This adds a systray app for linux, similar to the apps for macOS and
windows. There are already a number of community-developed systray apps,
but most of them are either long abandoned, are built for a specific
desktop environment, or simply wrap the tailscale CLI.
This uses fyne.io/systray (a fork of github.com/getlantern/systray)
which uses newer D-Bus specifications to render the tray icon and menu.
This results in a pretty broad support for modern desktop environments.
This initial commit lacks a number of features like profile switching,
device listing, and exit node selection. This is really focused on the
application structure, the interaction with LocalAPI, and some system
integration pieces like the app icon, notifications, and the clipboard.
Updates #1708
Signed-off-by: Will Norris <will@tailscale.com>
2024-07-10 13:45:10 -07:00
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 ) {
2024-09-05 12:11:05 -05:00
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 {
2024-08-23 08:34:36 -07:00
watcher , err := localClient . WatchIPNBus ( ctx , ipn . NotifyInitialState | ipn . NotifyNoPrivateKeys )
cmd/systray: add a basic linux systray app
This adds a systray app for linux, similar to the apps for macOS and
windows. There are already a number of community-developed systray apps,
but most of them are either long abandoned, are built for a specific
desktop environment, or simply wrap the tailscale CLI.
This uses fyne.io/systray (a fork of github.com/getlantern/systray)
which uses newer D-Bus specifications to render the tray icon and menu.
This results in a pretty broad support for modern desktop environments.
This initial commit lacks a number of features like profile switching,
device listing, and exit node selection. This is really focused on the
application structure, the interaction with LocalAPI, and some system
integration pieces like the app icon, notifications, and the clipboard.
Updates #1708
Signed-off-by: Will Norris <will@tailscale.com>
2024-07-10 13:45:10 -07:00
if err != nil {
2024-09-05 12:11:05 -05:00
return fmt . Errorf ( "watching ipn bus: %w" , err )
cmd/systray: add a basic linux systray app
This adds a systray app for linux, similar to the apps for macOS and
windows. There are already a number of community-developed systray apps,
but most of them are either long abandoned, are built for a specific
desktop environment, or simply wrap the tailscale CLI.
This uses fyne.io/systray (a fork of github.com/getlantern/systray)
which uses newer D-Bus specifications to render the tray icon and menu.
This results in a pretty broad support for modern desktop environments.
This initial commit lacks a number of features like profile switching,
device listing, and exit node selection. This is really focused on the
application structure, the interaction with LocalAPI, and some system
integration pieces like the app icon, notifications, and the clipboard.
Updates #1708
Signed-off-by: Will Norris <will@tailscale.com>
2024-07-10 13:45:10 -07:00
}
defer watcher . Close ( )
for {
select {
case <- ctx . Done ( ) :
2024-09-05 12:11:05 -05:00
return nil
cmd/systray: add a basic linux systray app
This adds a systray app for linux, similar to the apps for macOS and
windows. There are already a number of community-developed systray apps,
but most of them are either long abandoned, are built for a specific
desktop environment, or simply wrap the tailscale CLI.
This uses fyne.io/systray (a fork of github.com/getlantern/systray)
which uses newer D-Bus specifications to render the tray icon and menu.
This results in a pretty broad support for modern desktop environments.
This initial commit lacks a number of features like profile switching,
device listing, and exit node selection. This is really focused on the
application structure, the interaction with LocalAPI, and some system
integration pieces like the app icon, notifications, and the clipboard.
Updates #1708
Signed-off-by: Will Norris <will@tailscale.com>
2024-07-10 13:45:10 -07:00
default :
n , err := watcher . Next ( )
if err != nil {
2024-09-05 12:11:05 -05:00
return fmt . Errorf ( "ipnbus error: %w" , err )
cmd/systray: add a basic linux systray app
This adds a systray app for linux, similar to the apps for macOS and
windows. There are already a number of community-developed systray apps,
but most of them are either long abandoned, are built for a specific
desktop environment, or simply wrap the tailscale CLI.
This uses fyne.io/systray (a fork of github.com/getlantern/systray)
which uses newer D-Bus specifications to render the tray icon and menu.
This results in a pretty broad support for modern desktop environments.
This initial commit lacks a number of features like profile switching,
device listing, and exit node selection. This is really focused on the
application structure, the interaction with LocalAPI, and some system
integration pieces like the app icon, notifications, and the clipboard.
Updates #1708
Signed-off-by: Will Norris <will@tailscale.com>
2024-07-10 13:45:10 -07:00
}
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 ( ) )
}