omegafox/launcher/exec.go
daijro 8aadd3e1f2 Load default addons through launcher
- Default addons are now downloaded through the launcher to avoid redownloading when new profiles are created
- Updated README
2024-08-13 22:14:43 -05:00

150 lines
3 KiB
Go

package main
import (
"bufio"
"fmt"
"io"
"os"
"os/exec"
"os/signal"
"path/filepath"
"runtime"
"syscall"
)
func getExecutableName() string {
// Get the executable name based on the OS
switch normalizeOS(runtime.GOOS) {
case "linux":
return "./camoufox-bin"
case "macos":
return "./Camoufox.app"
case "windows":
return "./camoufox.exe"
default:
// This should never be reached due to the check in normalizeOS
return ""
}
}
func setExecutablePermissions(execPath string) error {
// Set executable permissions if needed
switch normalizeOS(runtime.GOOS) {
case "macos":
return filepath.Walk(execPath, func(path string, info os.FileInfo, err error) error {
if err != nil {
return err
}
return maybeSetPermission(path, 0755)
})
case "linux":
return maybeSetPermission(execPath, 0755)
}
return nil
}
func maybeSetPermission(path string, mode os.FileMode) error {
info, err := os.Stat(path)
if err != nil {
return err
}
currentMode := info.Mode().Perm()
if currentMode != mode {
return os.Chmod(path, mode)
}
return nil
}
func filterOutput(r io.Reader, w io.Writer) {
scanner := bufio.NewScanner(r)
for scanner.Scan() {
line := scanner.Text()
if !ExclusionRegex.MatchString(line) {
fmt.Fprintln(w, line)
}
}
}
// Run Camoufox
func runCamoufox(execName string, args []string, addonsList []string) {
// If addons are specified, get the debug port
var debugPortInt int
if len(addonsList) > 0 {
debugPortInt = getDebugPort(&args)
}
// Print args
cmd := exec.Command(execName, args...)
setProcessGroupID(cmd)
stdout, err := cmd.StdoutPipe()
if err != nil {
fmt.Printf("Error creating stdout pipe: %v\n", err)
os.Exit(1)
}
stderr, err := cmd.StderrPipe()
if err != nil {
fmt.Printf("Error creating stderr pipe: %v\n", err)
os.Exit(1)
}
// Set up signal handling
sigChan := make(chan os.Signal, 1)
signal.Notify(sigChan, os.Interrupt, syscall.SIGTERM)
if err := cmd.Start(); err != nil {
fmt.Printf("Error starting %s: %v\n", execName, err)
os.Exit(1)
}
if len(addonsList) > 0 {
go tryLoadAddons(debugPortInt, addonsList)
}
// Channel to signal when the subprocess has finished
subprocessDone := make(chan struct{})
// Start a goroutine to handle signals
go func() {
select {
case <-sigChan:
killProcessGroup(cmd)
case <-subprocessDone:
// Subprocess has finished, exit the Go process
os.Exit(0)
}
}()
done := make(chan bool)
go func() {
filterOutput(stdout, os.Stdout)
done <- true
}()
go func() {
filterOutput(stderr, os.Stderr)
done <- true
}()
<-done
<-done
// Wait for the command to finish
if err := cmd.Wait(); err != nil {
if exitErr, ok := err.(*exec.ExitError); ok {
// If the subprocess exited with an error, use its exit code
os.Exit(exitErr.ExitCode())
} else {
fmt.Printf("Error running %s: %v\n", execName, err)
os.Exit(1)
}
}
// Signal that the subprocess has finished
close(subprocessDone)
// Wait here to allow the signal handling goroutine to exit the process
select {}
}