Files
drover-go/cmd/drover/main.go
T
root 19f851afb0
Build / test (push) Successful in 1m15s
Build / build-windows (push) Successful in 57s
Release / release (push) Successful in 3m16s
Auto-update on startup: prompt + apply + auto-restart
Double-clicking drover.exe now silently checks for updates first
(8s timeout, ignores network failures). If a newer release is
published, a YES/NO message box asks the user; on YES we download,
verify SHA256, apply via selfupdate, and re-launch the binary so
the user immediately sees the new version's window.

NO or no update → straight to the smoke-test window as before.
Update errors show an error dialog, then continue.

Implementation:
- internal/updater is reused as-is — the existing CheckForUpdate +
  ApplyUpdate API is enough.
- autoupdate_windows.go owns the dialog and re-launch logic
  (golang.org/x/sys/windows for MessageBoxW, os/exec for fresh
  process). autoupdate_other.go is the no-op stub for Linux CI.
- relaunchSelf uses cmd.Start (fire and forget) — no Wait. Current
  process os.Exit(0) immediately so the OS doesn't keep both
  generations of drover running.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-01 03:21:59 +03:00

151 lines
4.3 KiB
Go

// Command drover is the entry point for the Discord proxy CLI.
package main
import (
"fmt"
"os"
"github.com/spf13/cobra"
"git.okcu.io/root/drover-go/internal/updater"
)
// Build-time variables, populated via -ldflags "-X main.Version=... -X main.Commit=... -X main.BuildDate=...".
var (
Version = "dev"
Commit = "dev"
BuildDate = "dev"
)
// configPath is the path to the TOML config file, set via the --config global flag.
// Reserved for use in later phases.
var configPath string
func main() {
// On Windows the binary is linked with -H=windowsgui so a double-click
// doesn't flash a console window. When the user runs us from cmd or
// PowerShell we still want stdout/stderr to land in their terminal —
// AttachConsole(ATTACH_PARENT_PROCESS) wires that up. No-op elsewhere.
attachToParentConsole()
// Inject our build version so the updater package can stamp it on the
// User-Agent header it sends to git.okcu.io.
updater.SetVersion(Version)
if err := newRootCmd().Execute(); err != nil {
// Cobra already prints the error; just exit non-zero.
os.Exit(1)
}
}
func newRootCmd() *cobra.Command {
root := &cobra.Command{
Use: "drover",
Short: "Discord proxy via SOCKS5 + WinDivert",
Version: fmt.Sprintf("%s (commit %s, built %s)", Version, Commit, BuildDate),
SilenceUsage: true,
SilenceErrors: false,
// No subcommand and no flags = end-user double-clicked the exe.
// First do a quick update check (silent if no network or already
// current); if an update is available we prompt, apply, and
// re-launch ourselves. Then show the smoke-test window.
RunE: func(cmd *cobra.Command, args []string) error {
autoUpdateOnStartup()
showTestWindow()
return nil
},
}
// Custom version template: "drover-go vX.Y.Z (commit abc1234, built 2026-05-01)".
root.SetVersionTemplate(fmt.Sprintf("drover-go v%s (commit %s, built %s)\n", Version, Commit, BuildDate))
root.PersistentFlags().StringVar(&configPath, "config", "", "path to TOML config file (reserved)")
root.AddCommand(newCheckCmd())
root.AddCommand(newUpdateCmd())
root.AddCommand(newServiceCmd())
root.AddCommand(newGUICmd())
return root
}
func newGUICmd() *cobra.Command {
return &cobra.Command{
Use: "gui",
Short: "Show a test window (smoke check that the binary launches on this machine)",
RunE: func(cmd *cobra.Command, args []string) error {
showTestWindow()
return nil
},
}
}
func newCheckCmd() *cobra.Command {
return &cobra.Command{
Use: "check",
Short: "Run the 7-step proxy diagnostic",
RunE: func(cmd *cobra.Command, args []string) error {
fmt.Fprintln(cmd.OutOrStdout(), "TODO: 7-step diagnostic")
return nil
},
}
}
func newUpdateCmd() *cobra.Command {
var checkOnly bool
cmd := &cobra.Command{
Use: "update",
Short: "Self-update via the Forgejo Releases API",
RunE: func(cmd *cobra.Command, args []string) error {
ctx := cmd.Context()
out := cmd.OutOrStdout()
src := updater.NewForgejoSource("git.okcu.io", "root", "drover-go", "windows-amd64.exe")
rel, hasUpdate, err := updater.CheckForUpdate(ctx, src, Version)
if err != nil {
return fmt.Errorf("check for update: %w", err)
}
if !hasUpdate {
fmt.Fprintln(out, "No updates available")
return nil
}
fmt.Fprintf(out, "Update available: %s (current v%s)\n", rel.TagName, Version)
if checkOnly {
return nil
}
fmt.Fprintln(out, "Downloading...")
if err := updater.ApplyUpdate(ctx, rel, func(d, t int64) {
if t > 0 {
fmt.Fprintf(out, "\r%d/%d bytes", d, t)
}
}); err != nil {
return fmt.Errorf("apply update: %w", err)
}
fmt.Fprintln(out, "\nUpdate applied. Restart drover.")
return nil
},
}
cmd.Flags().BoolVar(&checkOnly, "check-only", false, "only check for an update, do not apply")
return cmd
}
func newServiceCmd() *cobra.Command {
cmd := &cobra.Command{
Use: "service",
Short: "Manage the drover Windows service",
}
for _, name := range []string{"install", "uninstall", "start", "stop"} {
name := name
cmd.AddCommand(&cobra.Command{
Use: name,
Short: fmt.Sprintf("%s the drover Windows service", name),
RunE: func(cmd *cobra.Command, args []string) error {
fmt.Fprintf(cmd.OutOrStdout(), "TODO: service %s\n", name)
return nil
},
})
}
return cmd
}