-
-
Notifications
You must be signed in to change notification settings - Fork 1.3k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[v3] Start At Login #3910
base: v3-alpha
Are you sure you want to change the base?
[v3] Start At Login #3910
Changes from 2 commits
adb17f1
81d670b
c25cb02
730610a
623382b
7157952
11fdcc1
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -161,12 +161,19 @@ static const char* serializationNSDictionary(void *dict) { | |
import "C" | ||
import ( | ||
"encoding/json" | ||
"fmt" | ||
"os" | ||
"os/exec" | ||
"path/filepath" | ||
"strings" | ||
"unsafe" | ||
|
||
"github.com/pkg/errors" | ||
"github.com/wailsapp/wails/v3/internal/operatingsystem" | ||
|
||
"github.com/wailsapp/wails/v3/internal/assetserver/webview" | ||
"github.com/wailsapp/wails/v3/pkg/events" | ||
"github.com/wailsapp/wails/v3/pkg/mac" | ||
) | ||
|
||
type macosApp struct { | ||
|
@@ -377,3 +384,42 @@ func HandleOpenFile(filePath *C.char) { | |
ctx: eventContext, | ||
} | ||
} | ||
|
||
func (m *macosApp) setStartAtLogin(enabled bool) error { | ||
exe, err := os.Executable() | ||
if err != nil { | ||
return errors.Wrap(err, "Error running os.Executable:") | ||
} | ||
|
||
binName := filepath.Base(exe) | ||
if !strings.HasSuffix(exe, "/Contents/MacOS/"+binName) { | ||
return fmt.Errorf("app needs to be running as package.app file to start at login") | ||
} | ||
|
||
appPath := strings.TrimSuffix(exe, "/Contents/MacOS/"+binName) | ||
var command string | ||
if enabled { | ||
command = fmt.Sprintf("tell application \"System Events\" to make login item at end with properties {name: \"%s\",path:\"%s\", hidden:false}", binName, appPath) | ||
} else { | ||
command = fmt.Sprintf("tell application \"System Events\" to delete login item \"%s\"", binName) | ||
} | ||
|
||
cmd := exec.Command("osascript", "-e", command) | ||
_, err = cmd.CombinedOutput() | ||
return err | ||
} | ||
|
||
func (m *macosApp) canStartAtLogin() bool { | ||
bundleID := mac.GetBundleID() | ||
if bundleID == "" { | ||
return false | ||
} | ||
|
||
exe, err := os.Executable() | ||
if err != nil { | ||
return false | ||
} | ||
|
||
binName := filepath.Base(exe) | ||
return strings.HasSuffix(exe, "/Contents/MacOS/"+binName) | ||
} | ||
Comment on lines
+388
to
+425
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 💡 Codebase verification Missing required permissions for AppleScript automation The implementation lacks the required
🔗 Analysis chainVerify security implications and permissions The implementation uses AppleScript to modify login items, which requires appropriate permissions. We should verify:
Let's check for similar implementations and security considerations: Let me gather more information about error handling and permission requirements. 🏁 Scripts executedThe following scripts were executed for the analysis: Script: #!/bin/bash
# Search for similar login item implementations and permission handling
rg -g '*.go' -A 5 'System Events.*login item'
# Look for permission-related code
rg -g '*.go' 'NSAppleEventsUsageDescription'
# Check for security-related comments or documentation
rg -g '*.md' -i 'login item|start at login|permission|security'
Length of output: 4515 Script: #!/bin/bash
# Check Info.plist for permission requirements
fd -e plist -x cat {}
# Look for error handling patterns around osascript
rg -g '*.go' -B 3 -A 3 'osascript.*err'
# Search for documentation about permissions
fd -g 'README*.md' -x cat {}
Length of output: 109485 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @ansxuman this is important, if true... There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @leaanthony Most likely it's not needed. I had tested it on my machine. I'll do one thing to be double sure, I'll build and sign it using GitHub Actions and then will check how it behaves on my machine. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It may be needed on later versions of macOS. The rabbit may be on to something..... There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @leaanthony I build the package on github action and tested on my machine, it seems to be working without
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. If it works, then that's all good |
Original file line number | Diff line number | Diff line change | ||||||||||||||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
|
@@ -16,8 +16,10 @@ import "C" | |||||||||||||||||||||||||||||||
import ( | ||||||||||||||||||||||||||||||||
"fmt" | ||||||||||||||||||||||||||||||||
"os" | ||||||||||||||||||||||||||||||||
"path/filepath" | ||||||||||||||||||||||||||||||||
"strings" | ||||||||||||||||||||||||||||||||
"sync" | ||||||||||||||||||||||||||||||||
"text/template" | ||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||
"github.com/godbus/dbus/v5" | ||||||||||||||||||||||||||||||||
"github.com/wailsapp/wails/v3/internal/operatingsystem" | ||||||||||||||||||||||||||||||||
|
@@ -260,3 +262,68 @@ func fatalHandler(errFunc func(error)) { | |||||||||||||||||||||||||||||||
// Stub for windows function | ||||||||||||||||||||||||||||||||
return | ||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||
func (l *linuxApp) setStartAtLogin(enabled bool) error { | ||||||||||||||||||||||||||||||||
homedir, err := os.UserHomeDir() | ||||||||||||||||||||||||||||||||
if err != nil { | ||||||||||||||||||||||||||||||||
return err | ||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||
bin, err := os.Executable() | ||||||||||||||||||||||||||||||||
if err != nil { | ||||||||||||||||||||||||||||||||
return err | ||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||
Comment on lines
+272
to
+275
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 🛠️ Refactor suggestion Validate executable path The executable path should be validated to ensure it exists and is executable. bin, err := os.Executable()
if err != nil {
return err
}
+
+ // Validate executable
+ if info, err := os.Stat(bin); err != nil {
+ return fmt.Errorf("invalid executable path: %w", err)
+ } else if info.Mode()&0111 == 0 {
+ return fmt.Errorf("file is not executable: %s", bin)
+ } 📝 Committable suggestion
Suggested change
|
||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||
name := filepath.Base(bin) | ||||||||||||||||||||||||||||||||
autostartFile := fmt.Sprintf("%s-autostart.desktop", name) | ||||||||||||||||||||||||||||||||
autostartPath := filepath.Join(homedir, ".config", "autostart", autostartFile) | ||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||
if !enabled { | ||||||||||||||||||||||||||||||||
err := os.Remove(autostartPath) | ||||||||||||||||||||||||||||||||
if os.IsNotExist(err) { | ||||||||||||||||||||||||||||||||
return nil | ||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||
return err | ||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||
const tpl = `[Desktop Entry] | ||||||||||||||||||||||||||||||||
Name={{.Name}} | ||||||||||||||||||||||||||||||||
Comment=Autostart service for {{.Name}} | ||||||||||||||||||||||||||||||||
Type=Application | ||||||||||||||||||||||||||||||||
Exec={{.Cmd}} | ||||||||||||||||||||||||||||||||
Hidden=true | ||||||||||||||||||||||||||||||||
X-GNOME-Autostart-enabled=true | ||||||||||||||||||||||||||||||||
` | ||||||||||||||||||||||||||||||||
if err := os.MkdirAll(filepath.Dir(autostartPath), 0755); err != nil { | ||||||||||||||||||||||||||||||||
return err | ||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||
file, err := os.OpenFile(autostartPath, os.O_RDWR|os.O_CREATE, 0600) | ||||||||||||||||||||||||||||||||
if err != nil { | ||||||||||||||||||||||||||||||||
return err | ||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||
defer file.Close() | ||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||
t := template.Must(template.New("autostart").Parse(tpl)) | ||||||||||||||||||||||||||||||||
return t.Execute(file, struct { | ||||||||||||||||||||||||||||||||
Name string | ||||||||||||||||||||||||||||||||
Cmd string | ||||||||||||||||||||||||||||||||
}{ | ||||||||||||||||||||||||||||||||
Name: name, | ||||||||||||||||||||||||||||||||
Cmd: exe, | ||||||||||||||||||||||||||||||||
ansxuman marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||||||||||||||||||
}) | ||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||
func (l *linuxApp) canStartAtLogin() bool { | ||||||||||||||||||||||||||||||||
homedir, err := os.UserHomeDir() | ||||||||||||||||||||||||||||||||
if err != nil { | ||||||||||||||||||||||||||||||||
return false | ||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||
autostartDir := filepath.Join(homedir, ".config", "autostart") | ||||||||||||||||||||||||||||||||
if err := os.MkdirAll(autostartDir, 0755); err != nil { | ||||||||||||||||||||||||||||||||
return false | ||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||
return true | ||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||
Comment on lines
+317
to
+329
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 🛠️ Refactor suggestion Refactor capability check to be non-mutating The func (l *linuxApp) canStartAtLogin() bool {
homedir, err := os.UserHomeDir()
if err != nil {
+ l.parent.error("Failed to get user home directory: %v", err)
return false
}
autostartDir := filepath.Join(homedir, ".config", "autostart")
- if err := os.MkdirAll(autostartDir, 0755); err != nil {
+ _, err = os.Stat(autostartDir)
+ if os.IsNotExist(err) {
+ return true
+ } else if err != nil {
+ l.parent.error("Failed to check autostart directory: %v", err)
return false
}
return true
}
+
+// ensureAutostartDir ensures the autostart directory exists
+func (l *linuxApp) ensureAutostartDir() error {
+ homedir, err := os.UserHomeDir()
+ if err != nil {
+ return fmt.Errorf("failed to get user home directory: %w", err)
+ }
+
+ autostartDir := filepath.Join(homedir, ".config", "autostart")
+ return os.MkdirAll(autostartDir, 0755)
+} Then update func (l *linuxApp) setStartAtLogin(enabled bool) error {
+ if err := l.ensureAutostartDir(); err != nil {
+ return err
+ }
|
Original file line number | Diff line number | Diff line change | ||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
|
@@ -7,6 +7,7 @@ import ( | |||||||||||||
"os" | ||||||||||||||
"path/filepath" | ||||||||||||||
"slices" | ||||||||||||||
"strings" | ||||||||||||||
"sync" | ||||||||||||||
"syscall" | ||||||||||||||
"unsafe" | ||||||||||||||
|
@@ -16,6 +17,7 @@ import ( | |||||||||||||
|
||||||||||||||
"github.com/wailsapp/wails/v3/pkg/events" | ||||||||||||||
"github.com/wailsapp/wails/v3/pkg/w32" | ||||||||||||||
"golang.org/x/sys/windows/registry" | ||||||||||||||
) | ||||||||||||||
|
||||||||||||||
type windowsApp struct { | ||||||||||||||
|
@@ -373,3 +375,33 @@ func fatalHandler(errFunc func(error)) { | |||||||||||||
w32.Fatal = errFunc | ||||||||||||||
return | ||||||||||||||
} | ||||||||||||||
|
||||||||||||||
func (m *windowsApp) setStartAtLogin(enabled bool) error { | ||||||||||||||
exePath, err := os.Executable() | ||||||||||||||
if err != nil { | ||||||||||||||
return fmt.Errorf("failed to get executable path: %s", err) | ||||||||||||||
} | ||||||||||||||
|
||||||||||||||
registryKey := strings.Split(filepath.Base(exePath), ".")[0] | ||||||||||||||
|
||||||||||||||
key, err := registry.OpenKey(registry.CURRENT_USER, `Software\Microsoft\Windows\CurrentVersion\Run`, registry.ALL_ACCESS) | ||||||||||||||
Comment on lines
+381
to
+383
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 🛠️ Refactor suggestion Improve registry key name handling and access rights The current implementation has potential issues with registry key handling:
Consider these improvements: -registryKey := strings.Split(filepath.Base(exePath), ".")[0]
+registryKey := strings.TrimSuffix(filepath.Base(exePath), filepath.Ext(exePath))
-key, err := registry.OpenKey(registry.CURRENT_USER, `Software\Microsoft\Windows\CurrentVersion\Run`, registry.ALL_ACCESS)
+key, err := registry.OpenKey(registry.CURRENT_USER, `Software\Microsoft\Windows\CurrentVersion\Run`, registry.SET_VALUE|registry.QUERY_VALUE) 📝 Committable suggestion
Suggested change
|
||||||||||||||
if err != nil { | ||||||||||||||
return fmt.Errorf("failed to open registry key: %s", err) | ||||||||||||||
} | ||||||||||||||
defer key.Close() | ||||||||||||||
|
||||||||||||||
if enabled { | ||||||||||||||
err = key.SetStringValue(registryKey, exePath) | ||||||||||||||
} else { | ||||||||||||||
err = key.DeleteValue(registryKey) | ||||||||||||||
if err == registry.ErrNotExist { | ||||||||||||||
return nil | ||||||||||||||
} | ||||||||||||||
} | ||||||||||||||
return err | ||||||||||||||
} | ||||||||||||||
|
||||||||||||||
func (m *windowsApp) canStartAtLogin() bool { | ||||||||||||||
// Windows generally supports start at login via registry | ||||||||||||||
return true | ||||||||||||||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Enhance error handling and add input validation
While the implementation is generally sound, there are a few areas that could be improved:
cmd.CombinedOutput()
might contain valuable error information that's currently being discardedbinName
should be validated/escaped to prevent AppleScript injectionHere's a suggested improvement:
📝 Committable suggestion