mirror of
https://github.com/AvengeMedia/DankMaterialShell.git
synced 2026-06-08 12:13:31 -04:00
8eb23bcc29
- Bring up Mango to parity with niri/hyprland via a native JSON-IPC w/Native MangoServic., replaces the legacy dwl/`mmsg` path and recent breaking changes - Dankinstall: mango supported installer, config/binds templates, and packaging (Arch AUR, Fedora Terra auto-enable, Gentoo GURU) - Window rules: Go provider + CLI + Settings GUI editor - Keybinds + config reload on edit (mmsg dispatch reload_config) - Misc new supported options in DMS settings
894 lines
29 KiB
Go
894 lines
29 KiB
Go
package config
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"regexp"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/AvengeMedia/DankMaterialShell/core/internal/deps"
|
|
)
|
|
|
|
const hyprlandBackupDirName = ".dms-backups"
|
|
|
|
type ConfigDeployer struct {
|
|
logChan chan<- string
|
|
}
|
|
|
|
type DeploymentResult struct {
|
|
ConfigType string
|
|
Path string
|
|
BackupPath string
|
|
Deployed bool
|
|
Error error
|
|
}
|
|
|
|
func NewConfigDeployer(logChan chan<- string) *ConfigDeployer {
|
|
return &ConfigDeployer{
|
|
logChan: logChan,
|
|
}
|
|
}
|
|
|
|
func (cd *ConfigDeployer) log(message string) {
|
|
if cd.logChan != nil {
|
|
cd.logChan <- message
|
|
}
|
|
}
|
|
|
|
// DeployConfigurations deploys all necessary configurations based on the chosen window manager
|
|
func (cd *ConfigDeployer) DeployConfigurations(ctx context.Context, wm deps.WindowManager) ([]DeploymentResult, error) {
|
|
return cd.DeployConfigurationsWithTerminal(ctx, wm, deps.TerminalGhostty)
|
|
}
|
|
|
|
// DeployConfigurationsWithTerminal deploys all necessary configurations based on chosen window manager and terminal
|
|
func (cd *ConfigDeployer) DeployConfigurationsWithTerminal(ctx context.Context, wm deps.WindowManager, terminal deps.Terminal) ([]DeploymentResult, error) {
|
|
return cd.DeployConfigurationsSelective(ctx, wm, terminal, nil, nil)
|
|
}
|
|
|
|
// DeployConfigurationsWithSystemd deploys configurations with systemd option
|
|
func (cd *ConfigDeployer) DeployConfigurationsWithSystemd(ctx context.Context, wm deps.WindowManager, terminal deps.Terminal, useSystemd bool) ([]DeploymentResult, error) {
|
|
return cd.deployConfigurationsInternal(ctx, wm, terminal, nil, nil, nil, useSystemd)
|
|
}
|
|
|
|
func (cd *ConfigDeployer) DeployConfigurationsSelective(ctx context.Context, wm deps.WindowManager, terminal deps.Terminal, installedDeps []deps.Dependency, replaceConfigs map[string]bool) ([]DeploymentResult, error) {
|
|
return cd.DeployConfigurationsSelectiveWithReinstalls(ctx, wm, terminal, installedDeps, replaceConfigs, nil)
|
|
}
|
|
|
|
func (cd *ConfigDeployer) DeployConfigurationsSelectiveWithReinstalls(ctx context.Context, wm deps.WindowManager, terminal deps.Terminal, installedDeps []deps.Dependency, replaceConfigs map[string]bool, reinstallItems map[string]bool) ([]DeploymentResult, error) {
|
|
return cd.deployConfigurationsInternal(ctx, wm, terminal, installedDeps, replaceConfigs, reinstallItems, true)
|
|
}
|
|
|
|
func (cd *ConfigDeployer) deployConfigurationsInternal(ctx context.Context, wm deps.WindowManager, terminal deps.Terminal, installedDeps []deps.Dependency, replaceConfigs map[string]bool, reinstallItems map[string]bool, useSystemd bool) ([]DeploymentResult, error) {
|
|
var results []DeploymentResult
|
|
|
|
// Primary config file paths used to detect fresh installs.
|
|
configPrimaryPaths := map[string][]string{
|
|
"Niri": {
|
|
filepath.Join(os.Getenv("HOME"), ".config", "niri", "config.kdl"),
|
|
},
|
|
"Hyprland": {
|
|
filepath.Join(os.Getenv("HOME"), ".config", "hypr", "hyprland.lua"),
|
|
filepath.Join(os.Getenv("HOME"), ".config", "hypr", "hyprland.conf"),
|
|
},
|
|
"Mango": {
|
|
filepath.Join(os.Getenv("HOME"), ".config", "mango", "config.conf"),
|
|
filepath.Join(os.Getenv("HOME"), ".config", "mango", "mango.conf"),
|
|
},
|
|
"Ghostty": {
|
|
filepath.Join(os.Getenv("HOME"), ".config", "ghostty", "config"),
|
|
},
|
|
"Kitty": {
|
|
filepath.Join(os.Getenv("HOME"), ".config", "kitty", "kitty.conf"),
|
|
},
|
|
"Alacritty": {
|
|
filepath.Join(os.Getenv("HOME"), ".config", "alacritty", "alacritty.toml"),
|
|
},
|
|
}
|
|
|
|
shouldReplaceConfig := func(configType string) bool {
|
|
if replaceConfigs == nil {
|
|
return true
|
|
}
|
|
replace, exists := replaceConfigs[configType]
|
|
if !exists || replace {
|
|
return true
|
|
}
|
|
// Config is explicitly set to "don't replace" — but still deploy
|
|
// if the config file doesn't exist yet (fresh install scenario).
|
|
if primaryPaths, ok := configPrimaryPaths[configType]; ok {
|
|
exists := false
|
|
for _, primaryPath := range primaryPaths {
|
|
if _, err := os.Stat(primaryPath); err == nil {
|
|
exists = true
|
|
break
|
|
}
|
|
}
|
|
if !exists {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
switch wm {
|
|
case deps.WindowManagerNiri:
|
|
if shouldReplaceConfig("Niri") {
|
|
result, err := cd.deployNiriConfig(terminal, useSystemd)
|
|
results = append(results, result)
|
|
if err != nil {
|
|
return results, fmt.Errorf("failed to deploy Niri config: %w", err)
|
|
}
|
|
}
|
|
case deps.WindowManagerHyprland:
|
|
if shouldReplaceConfig("Hyprland") {
|
|
result, err := cd.deployHyprlandConfig(terminal, useSystemd)
|
|
results = append(results, result)
|
|
if err != nil {
|
|
return results, fmt.Errorf("failed to deploy Hyprland config: %w", err)
|
|
}
|
|
}
|
|
case deps.WindowManagerMango:
|
|
if shouldReplaceConfig("Mango") {
|
|
result, err := cd.deployMangoConfig(terminal, useSystemd)
|
|
results = append(results, result)
|
|
if err != nil {
|
|
return results, fmt.Errorf("failed to deploy Mango config: %w", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
switch terminal {
|
|
case deps.TerminalGhostty:
|
|
if shouldReplaceConfig("Ghostty") {
|
|
ghosttyResults, err := cd.deployGhosttyConfig()
|
|
results = append(results, ghosttyResults...)
|
|
if err != nil {
|
|
return results, fmt.Errorf("failed to deploy Ghostty config: %w", err)
|
|
}
|
|
}
|
|
case deps.TerminalKitty:
|
|
if shouldReplaceConfig("Kitty") {
|
|
kittyResults, err := cd.deployKittyConfig()
|
|
results = append(results, kittyResults...)
|
|
if err != nil {
|
|
return results, fmt.Errorf("failed to deploy Kitty config: %w", err)
|
|
}
|
|
}
|
|
case deps.TerminalAlacritty:
|
|
if shouldReplaceConfig("Alacritty") {
|
|
alacrittyResults, err := cd.deployAlacrittyConfig()
|
|
results = append(results, alacrittyResults...)
|
|
if err != nil {
|
|
return results, fmt.Errorf("failed to deploy Alacritty config: %w", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
return results, nil
|
|
}
|
|
|
|
func (cd *ConfigDeployer) deployNiriConfig(terminal deps.Terminal, useSystemd bool) (DeploymentResult, error) {
|
|
result := DeploymentResult{
|
|
ConfigType: "Niri",
|
|
Path: filepath.Join(os.Getenv("HOME"), ".config", "niri", "config.kdl"),
|
|
}
|
|
|
|
configDir := filepath.Dir(result.Path)
|
|
if err := os.MkdirAll(configDir, 0o755); err != nil {
|
|
result.Error = fmt.Errorf("failed to create config directory: %w", err)
|
|
return result, result.Error
|
|
}
|
|
|
|
dmsDir := filepath.Join(configDir, "dms")
|
|
if err := os.MkdirAll(dmsDir, 0o755); err != nil {
|
|
result.Error = fmt.Errorf("failed to create dms directory: %w", err)
|
|
return result, result.Error
|
|
}
|
|
|
|
var existingConfig string
|
|
if _, err := os.Stat(result.Path); err == nil {
|
|
cd.log("Found existing Niri configuration")
|
|
|
|
existingData, err := os.ReadFile(result.Path)
|
|
if err != nil {
|
|
result.Error = fmt.Errorf("failed to read existing config: %w", err)
|
|
return result, result.Error
|
|
}
|
|
existingConfig = string(existingData)
|
|
|
|
timestamp := time.Now().Format("2006-01-02_15-04-05")
|
|
result.BackupPath = result.Path + ".backup." + timestamp
|
|
if err := os.WriteFile(result.BackupPath, existingData, 0o644); err != nil {
|
|
result.Error = fmt.Errorf("failed to create backup: %w", err)
|
|
return result, result.Error
|
|
}
|
|
cd.log(fmt.Sprintf("Backed up existing config to %s", result.BackupPath))
|
|
}
|
|
|
|
var terminalCommand string
|
|
switch terminal {
|
|
case deps.TerminalGhostty:
|
|
terminalCommand = "ghostty"
|
|
case deps.TerminalKitty:
|
|
terminalCommand = "kitty"
|
|
case deps.TerminalAlacritty:
|
|
terminalCommand = "alacritty"
|
|
default:
|
|
terminalCommand = "ghostty"
|
|
}
|
|
|
|
newConfig := strings.ReplaceAll(NiriConfig, "{{TERMINAL_COMMAND}}", terminalCommand)
|
|
|
|
if !useSystemd {
|
|
newConfig = cd.transformNiriConfigForNonSystemd(newConfig, terminalCommand)
|
|
}
|
|
|
|
if existingConfig != "" {
|
|
mergedConfig, err := cd.mergeNiriOutputSections(newConfig, existingConfig, dmsDir)
|
|
if err != nil {
|
|
cd.log(fmt.Sprintf("Warning: Failed to merge output sections: %v", err))
|
|
} else {
|
|
newConfig = mergedConfig
|
|
cd.log("Successfully merged existing output sections")
|
|
}
|
|
}
|
|
|
|
if err := os.WriteFile(result.Path, []byte(newConfig), 0o644); err != nil {
|
|
result.Error = fmt.Errorf("failed to write config: %w", err)
|
|
return result, result.Error
|
|
}
|
|
|
|
if err := cd.deployNiriDmsConfigs(dmsDir, terminalCommand); err != nil {
|
|
result.Error = fmt.Errorf("failed to deploy dms configs: %w", err)
|
|
return result, result.Error
|
|
}
|
|
|
|
result.Deployed = true
|
|
cd.log("Successfully deployed Niri configuration")
|
|
return result, nil
|
|
}
|
|
|
|
func (cd *ConfigDeployer) deployNiriDmsConfigs(dmsDir, terminalCommand string) error {
|
|
configs := []struct {
|
|
name string
|
|
content string
|
|
}{
|
|
{"colors.kdl", NiriColorsConfig},
|
|
{"layout.kdl", NiriLayoutConfig},
|
|
{"alttab.kdl", NiriAlttabConfig},
|
|
{"binds.kdl", strings.ReplaceAll(NiriBindsConfig, "{{TERMINAL_COMMAND}}", terminalCommand)},
|
|
{"outputs.kdl", ""},
|
|
{"cursor.kdl", ""},
|
|
{"windowrules.kdl", ""},
|
|
}
|
|
|
|
for _, cfg := range configs {
|
|
path := filepath.Join(dmsDir, cfg.name)
|
|
// Skip if file already exists and is not empty to preserve user modifications
|
|
if info, err := os.Stat(path); err == nil && info.Size() > 0 {
|
|
cd.log(fmt.Sprintf("Skipping %s (already exists)", cfg.name))
|
|
continue
|
|
}
|
|
if err := os.WriteFile(path, []byte(cfg.content), 0o644); err != nil {
|
|
return fmt.Errorf("failed to write %s: %w", cfg.name, err)
|
|
}
|
|
cd.log(fmt.Sprintf("Deployed %s", cfg.name))
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (cd *ConfigDeployer) deployMangoConfig(terminal deps.Terminal, useSystemd bool) (DeploymentResult, error) {
|
|
result := DeploymentResult{
|
|
ConfigType: "Mango",
|
|
Path: filepath.Join(os.Getenv("HOME"), ".config", "mango", "config.conf"),
|
|
}
|
|
|
|
configDir := filepath.Dir(result.Path)
|
|
if err := os.MkdirAll(configDir, 0o755); err != nil {
|
|
result.Error = fmt.Errorf("failed to create config directory: %w", err)
|
|
return result, result.Error
|
|
}
|
|
|
|
dmsDir := filepath.Join(configDir, "dms")
|
|
if err := os.MkdirAll(dmsDir, 0o755); err != nil {
|
|
result.Error = fmt.Errorf("failed to create dms directory: %w", err)
|
|
return result, result.Error
|
|
}
|
|
|
|
var terminalCommand string
|
|
switch terminal {
|
|
case deps.TerminalGhostty:
|
|
terminalCommand = "ghostty"
|
|
case deps.TerminalKitty:
|
|
terminalCommand = "kitty"
|
|
case deps.TerminalAlacritty:
|
|
terminalCommand = "alacritty"
|
|
default:
|
|
terminalCommand = "ghostty"
|
|
}
|
|
|
|
// DMS owns config.conf for mango (like niri/hyprland): back up and replace.
|
|
if existingData, err := os.ReadFile(result.Path); err == nil {
|
|
cd.log("Found existing Mango configuration")
|
|
timestamp := time.Now().Format("2006-01-02_15-04-05")
|
|
result.BackupPath = result.Path + ".backup." + timestamp
|
|
if err := os.WriteFile(result.BackupPath, existingData, 0o644); err != nil {
|
|
result.Error = fmt.Errorf("failed to create backup: %w", err)
|
|
return result, result.Error
|
|
}
|
|
cd.log(fmt.Sprintf("Backed up existing config to %s", result.BackupPath))
|
|
}
|
|
|
|
newConfig := strings.ReplaceAll(MangoConfig, "{{TERMINAL_COMMAND}}", terminalCommand)
|
|
if err := os.WriteFile(result.Path, []byte(newConfig), 0o644); err != nil {
|
|
result.Error = fmt.Errorf("failed to write config: %w", err)
|
|
return result, result.Error
|
|
}
|
|
|
|
if err := cd.deployMangoDmsConfigs(dmsDir, terminalCommand); err != nil {
|
|
result.Error = fmt.Errorf("failed to deploy dms configs: %w", err)
|
|
return result, result.Error
|
|
}
|
|
|
|
result.Deployed = true
|
|
cd.log("Successfully deployed Mango configuration")
|
|
return result, nil
|
|
}
|
|
|
|
func (cd *ConfigDeployer) deployMangoDmsConfigs(dmsDir, terminalCommand string) error {
|
|
configs := []struct {
|
|
name string
|
|
content string
|
|
overwrite bool
|
|
}{
|
|
// binds.conf is DMS-owned (overwrite); the rest are runtime/user-managed.
|
|
{"binds.conf", strings.ReplaceAll(MangoBindsConfig, "{{TERMINAL_COMMAND}}", terminalCommand), true},
|
|
{"colors.conf", MangoColorsConfig, false},
|
|
{"layout.conf", MangoLayoutConfig, false},
|
|
{"outputs.conf", "", false},
|
|
{"cursor.conf", "", false},
|
|
{"windowrules.conf", "", false},
|
|
}
|
|
|
|
for _, cfg := range configs {
|
|
path := filepath.Join(dmsDir, cfg.name)
|
|
if !cfg.overwrite {
|
|
if info, err := os.Stat(path); err == nil && info.Size() > 0 {
|
|
cd.log(fmt.Sprintf("Skipping %s (already exists)", cfg.name))
|
|
continue
|
|
}
|
|
}
|
|
if err := os.WriteFile(path, []byte(cfg.content), 0o644); err != nil {
|
|
return fmt.Errorf("failed to write %s: %w", cfg.name, err)
|
|
}
|
|
cd.log(fmt.Sprintf("Deployed %s", cfg.name))
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (cd *ConfigDeployer) deployGhosttyConfig() ([]DeploymentResult, error) {
|
|
var results []DeploymentResult
|
|
|
|
mainResult := DeploymentResult{
|
|
ConfigType: "Ghostty",
|
|
Path: filepath.Join(os.Getenv("HOME"), ".config", "ghostty", "config"),
|
|
}
|
|
|
|
configDir := filepath.Dir(mainResult.Path)
|
|
if err := os.MkdirAll(configDir, 0o755); err != nil {
|
|
mainResult.Error = fmt.Errorf("failed to create config directory: %w", err)
|
|
return []DeploymentResult{mainResult}, mainResult.Error
|
|
}
|
|
|
|
if _, err := os.Stat(mainResult.Path); err == nil {
|
|
cd.log("Found existing Ghostty configuration")
|
|
|
|
existingData, err := os.ReadFile(mainResult.Path)
|
|
if err != nil {
|
|
mainResult.Error = fmt.Errorf("failed to read existing config: %w", err)
|
|
return []DeploymentResult{mainResult}, mainResult.Error
|
|
}
|
|
|
|
timestamp := time.Now().Format("2006-01-02_15-04-05")
|
|
mainResult.BackupPath = mainResult.Path + ".backup." + timestamp
|
|
if err := os.WriteFile(mainResult.BackupPath, existingData, 0o644); err != nil {
|
|
mainResult.Error = fmt.Errorf("failed to create backup: %w", err)
|
|
return []DeploymentResult{mainResult}, mainResult.Error
|
|
}
|
|
cd.log(fmt.Sprintf("Backed up existing config to %s", mainResult.BackupPath))
|
|
}
|
|
|
|
if err := os.WriteFile(mainResult.Path, []byte(GhosttyConfig), 0o644); err != nil {
|
|
mainResult.Error = fmt.Errorf("failed to write config: %w", err)
|
|
return []DeploymentResult{mainResult}, mainResult.Error
|
|
}
|
|
|
|
mainResult.Deployed = true
|
|
cd.log("Successfully deployed Ghostty configuration")
|
|
results = append(results, mainResult)
|
|
|
|
colorResult := DeploymentResult{
|
|
ConfigType: "Ghostty Colors",
|
|
Path: filepath.Join(os.Getenv("HOME"), ".config", "ghostty", "themes", "dankcolors"),
|
|
}
|
|
|
|
themesDir := filepath.Dir(colorResult.Path)
|
|
if err := os.MkdirAll(themesDir, 0o755); err != nil {
|
|
mainResult.Error = fmt.Errorf("failed to create themes directory: %w", err)
|
|
return []DeploymentResult{mainResult}, mainResult.Error
|
|
}
|
|
|
|
if err := os.WriteFile(colorResult.Path, []byte(GhosttyColorConfig), 0o644); err != nil {
|
|
colorResult.Error = fmt.Errorf("failed to write color config: %w", err)
|
|
return results, colorResult.Error
|
|
}
|
|
|
|
colorResult.Deployed = true
|
|
cd.log("Successfully deployed Ghostty color configuration")
|
|
results = append(results, colorResult)
|
|
|
|
return results, nil
|
|
}
|
|
|
|
func (cd *ConfigDeployer) deployKittyConfig() ([]DeploymentResult, error) {
|
|
var results []DeploymentResult
|
|
|
|
mainResult := DeploymentResult{
|
|
ConfigType: "Kitty",
|
|
Path: filepath.Join(os.Getenv("HOME"), ".config", "kitty", "kitty.conf"),
|
|
}
|
|
|
|
configDir := filepath.Dir(mainResult.Path)
|
|
if err := os.MkdirAll(configDir, 0o755); err != nil {
|
|
mainResult.Error = fmt.Errorf("failed to create config directory: %w", err)
|
|
return []DeploymentResult{mainResult}, mainResult.Error
|
|
}
|
|
|
|
if _, err := os.Stat(mainResult.Path); err == nil {
|
|
cd.log("Found existing Kitty configuration")
|
|
|
|
existingData, err := os.ReadFile(mainResult.Path)
|
|
if err != nil {
|
|
mainResult.Error = fmt.Errorf("failed to read existing config: %w", err)
|
|
return []DeploymentResult{mainResult}, mainResult.Error
|
|
}
|
|
|
|
timestamp := time.Now().Format("2006-01-02_15-04-05")
|
|
mainResult.BackupPath = mainResult.Path + ".backup." + timestamp
|
|
if err := os.WriteFile(mainResult.BackupPath, existingData, 0o644); err != nil {
|
|
mainResult.Error = fmt.Errorf("failed to create backup: %w", err)
|
|
return []DeploymentResult{mainResult}, mainResult.Error
|
|
}
|
|
cd.log(fmt.Sprintf("Backed up existing config to %s", mainResult.BackupPath))
|
|
}
|
|
|
|
if err := os.WriteFile(mainResult.Path, []byte(KittyConfig), 0o644); err != nil {
|
|
mainResult.Error = fmt.Errorf("failed to write config: %w", err)
|
|
return []DeploymentResult{mainResult}, mainResult.Error
|
|
}
|
|
|
|
mainResult.Deployed = true
|
|
cd.log("Successfully deployed Kitty configuration")
|
|
results = append(results, mainResult)
|
|
|
|
themeResult := DeploymentResult{
|
|
ConfigType: "Kitty Theme",
|
|
Path: filepath.Join(os.Getenv("HOME"), ".config", "kitty", "dank-theme.conf"),
|
|
}
|
|
|
|
if err := os.WriteFile(themeResult.Path, []byte(KittyThemeConfig), 0o644); err != nil {
|
|
themeResult.Error = fmt.Errorf("failed to write theme config: %w", err)
|
|
return results, themeResult.Error
|
|
}
|
|
|
|
themeResult.Deployed = true
|
|
cd.log("Successfully deployed Kitty theme configuration")
|
|
results = append(results, themeResult)
|
|
|
|
tabsResult := DeploymentResult{
|
|
ConfigType: "Kitty Tabs",
|
|
Path: filepath.Join(os.Getenv("HOME"), ".config", "kitty", "dank-tabs.conf"),
|
|
}
|
|
|
|
if err := os.WriteFile(tabsResult.Path, []byte(KittyTabsConfig), 0o644); err != nil {
|
|
tabsResult.Error = fmt.Errorf("failed to write tabs config: %w", err)
|
|
return results, tabsResult.Error
|
|
}
|
|
|
|
tabsResult.Deployed = true
|
|
cd.log("Successfully deployed Kitty tabs configuration")
|
|
results = append(results, tabsResult)
|
|
|
|
return results, nil
|
|
}
|
|
|
|
func (cd *ConfigDeployer) deployAlacrittyConfig() ([]DeploymentResult, error) {
|
|
var results []DeploymentResult
|
|
|
|
mainResult := DeploymentResult{
|
|
ConfigType: "Alacritty",
|
|
Path: filepath.Join(os.Getenv("HOME"), ".config", "alacritty", "alacritty.toml"),
|
|
}
|
|
|
|
configDir := filepath.Dir(mainResult.Path)
|
|
if err := os.MkdirAll(configDir, 0o755); err != nil {
|
|
mainResult.Error = fmt.Errorf("failed to create config directory: %w", err)
|
|
return []DeploymentResult{mainResult}, mainResult.Error
|
|
}
|
|
|
|
if _, err := os.Stat(mainResult.Path); err == nil {
|
|
cd.log("Found existing Alacritty configuration")
|
|
|
|
existingData, err := os.ReadFile(mainResult.Path)
|
|
if err != nil {
|
|
mainResult.Error = fmt.Errorf("failed to read existing config: %w", err)
|
|
return []DeploymentResult{mainResult}, mainResult.Error
|
|
}
|
|
|
|
timestamp := time.Now().Format("2006-01-02_15-04-05")
|
|
mainResult.BackupPath = mainResult.Path + ".backup." + timestamp
|
|
if err := os.WriteFile(mainResult.BackupPath, existingData, 0o644); err != nil {
|
|
mainResult.Error = fmt.Errorf("failed to create backup: %w", err)
|
|
return []DeploymentResult{mainResult}, mainResult.Error
|
|
}
|
|
cd.log(fmt.Sprintf("Backed up existing config to %s", mainResult.BackupPath))
|
|
}
|
|
|
|
if err := os.WriteFile(mainResult.Path, []byte(AlacrittyConfig), 0o644); err != nil {
|
|
mainResult.Error = fmt.Errorf("failed to write config: %w", err)
|
|
return []DeploymentResult{mainResult}, mainResult.Error
|
|
}
|
|
|
|
mainResult.Deployed = true
|
|
cd.log("Successfully deployed Alacritty configuration")
|
|
results = append(results, mainResult)
|
|
|
|
themeResult := DeploymentResult{
|
|
ConfigType: "Alacritty Theme",
|
|
Path: filepath.Join(os.Getenv("HOME"), ".config", "alacritty", "dank-theme.toml"),
|
|
}
|
|
|
|
if err := os.WriteFile(themeResult.Path, []byte(AlacrittyThemeConfig), 0o644); err != nil {
|
|
themeResult.Error = fmt.Errorf("failed to write theme config: %w", err)
|
|
return results, themeResult.Error
|
|
}
|
|
|
|
themeResult.Deployed = true
|
|
cd.log("Successfully deployed Alacritty theme configuration")
|
|
results = append(results, themeResult)
|
|
|
|
return results, nil
|
|
}
|
|
|
|
func (cd *ConfigDeployer) mergeNiriOutputSections(newConfig, existingConfig, dmsDir string) (string, error) {
|
|
outputRegex := regexp.MustCompile(`(?m)^(/-)?\s*output\s+"[^"]+"\s*\{[^{}]*(?:\{[^{}]*\}[^{}]*)*\}`)
|
|
existingOutputs := outputRegex.FindAllString(existingConfig, -1)
|
|
|
|
if len(existingOutputs) == 0 {
|
|
return newConfig, nil
|
|
}
|
|
|
|
outputsPath := filepath.Join(dmsDir, "outputs.kdl")
|
|
if _, err := os.Stat(outputsPath); err != nil {
|
|
var outputsContent strings.Builder
|
|
for _, output := range existingOutputs {
|
|
outputsContent.WriteString(output)
|
|
outputsContent.WriteString("\n\n")
|
|
}
|
|
if err := os.WriteFile(outputsPath, []byte(outputsContent.String()), 0o644); err != nil {
|
|
cd.log(fmt.Sprintf("Warning: Failed to migrate outputs to %s: %v", outputsPath, err))
|
|
} else {
|
|
cd.log("Migrated output sections to dms/outputs.kdl")
|
|
}
|
|
}
|
|
|
|
exampleOutputRegex := regexp.MustCompile(`(?m)^/-output "eDP-2" \{[^{}]*(?:\{[^{}]*\}[^{}]*)*\}`)
|
|
mergedConfig := exampleOutputRegex.ReplaceAllString(newConfig, "")
|
|
|
|
inputEndRegex := regexp.MustCompile(`(?m)^}$`)
|
|
inputMatches := inputEndRegex.FindAllStringIndex(newConfig, -1)
|
|
|
|
if len(inputMatches) < 1 {
|
|
return "", fmt.Errorf("could not find insertion point for output sections")
|
|
}
|
|
|
|
insertPos := inputMatches[0][1]
|
|
|
|
var builder strings.Builder
|
|
builder.WriteString(mergedConfig[:insertPos])
|
|
builder.WriteString("\n// Outputs from existing configuration\n")
|
|
|
|
for _, output := range existingOutputs {
|
|
builder.WriteString(output)
|
|
builder.WriteString("\n")
|
|
}
|
|
|
|
builder.WriteString(mergedConfig[insertPos:])
|
|
|
|
return builder.String(), nil
|
|
}
|
|
|
|
// deployHyprlandConfig handles Hyprland configuration deployment with backup and merging
|
|
func (cd *ConfigDeployer) deployHyprlandConfig(terminal deps.Terminal, useSystemd bool) (DeploymentResult, error) {
|
|
result := DeploymentResult{
|
|
ConfigType: "Hyprland",
|
|
Path: filepath.Join(os.Getenv("HOME"), ".config", "hypr", "hyprland.lua"),
|
|
}
|
|
|
|
configDir := filepath.Dir(result.Path)
|
|
if err := os.MkdirAll(configDir, 0o755); err != nil {
|
|
result.Error = fmt.Errorf("failed to create config directory: %w", err)
|
|
return result, result.Error
|
|
}
|
|
|
|
dmsDir := filepath.Join(configDir, "dms")
|
|
if err := os.MkdirAll(dmsDir, 0o755); err != nil {
|
|
result.Error = fmt.Errorf("failed to create dms directory: %w", err)
|
|
return result, result.Error
|
|
}
|
|
|
|
timestamp := time.Now().Format("2006-01-02_15-04-05")
|
|
backupDir := filepath.Join(configDir, hyprlandBackupDirName, timestamp)
|
|
var existingConfig string
|
|
existingData, existingPath, err := readExistingHyprlandConfig(configDir)
|
|
if err != nil {
|
|
result.Error = err
|
|
return result, result.Error
|
|
}
|
|
if existingData != "" {
|
|
existingConfig = existingData
|
|
cd.log(fmt.Sprintf("Found existing Hyprland configuration at %s", existingPath))
|
|
|
|
result.BackupPath = filepath.Join(backupDir, filepath.Base(existingPath))
|
|
if err := backupHyprlandConfigFile(existingPath, result.BackupPath, []byte(existingData), strings.EqualFold(filepath.Ext(existingPath), ".conf")); err != nil {
|
|
result.Error = fmt.Errorf("failed to create backup: %w", err)
|
|
return result, result.Error
|
|
}
|
|
cd.log(fmt.Sprintf("Backed up existing config to %s", result.BackupPath))
|
|
}
|
|
|
|
var terminalCommand string
|
|
switch terminal {
|
|
case deps.TerminalGhostty:
|
|
terminalCommand = "ghostty"
|
|
case deps.TerminalKitty:
|
|
terminalCommand = "kitty"
|
|
case deps.TerminalAlacritty:
|
|
terminalCommand = "alacritty"
|
|
default:
|
|
terminalCommand = "ghostty"
|
|
}
|
|
|
|
newConfig := strings.ReplaceAll(HyprlandLuaConfig, "{{TERMINAL_COMMAND}}", terminalCommand)
|
|
|
|
if !useSystemd {
|
|
newConfig = transformHyprlandLuaForNonSystemd(newConfig, terminalCommand)
|
|
}
|
|
|
|
if existingConfig != "" {
|
|
mergedConfig, err := cd.mergeHyprlandMonitorSections(newConfig, existingConfig, dmsDir)
|
|
if err != nil {
|
|
cd.log(fmt.Sprintf("Warning: Failed to merge monitor sections: %v", err))
|
|
} else {
|
|
newConfig = mergedConfig
|
|
cd.log("Successfully merged existing monitor sections")
|
|
}
|
|
}
|
|
|
|
if err := os.WriteFile(result.Path, []byte(newConfig), 0o644); err != nil {
|
|
result.Error = fmt.Errorf("failed to write config: %w", err)
|
|
return result, result.Error
|
|
}
|
|
|
|
movedLegacy, err := backupLegacyHyprlandConfFiles(configDir, dmsDir, backupDir)
|
|
if err != nil {
|
|
result.Error = fmt.Errorf("failed to back up legacy hyprlang configs: %w", err)
|
|
return result, result.Error
|
|
}
|
|
if movedLegacy > 0 {
|
|
if result.BackupPath == "" {
|
|
result.BackupPath = backupDir
|
|
}
|
|
cd.log(fmt.Sprintf("Moved %d legacy hyprlang config(s) to %s", movedLegacy, backupDir))
|
|
}
|
|
|
|
if err := cd.deployHyprlandDmsConfigs(dmsDir, terminalCommand); err != nil {
|
|
result.Error = fmt.Errorf("failed to deploy dms configs: %w", err)
|
|
return result, result.Error
|
|
}
|
|
|
|
CleanupStrayHyprlandConfFile(func(format string, v ...any) {
|
|
cd.log(fmt.Sprintf(format, v...))
|
|
})
|
|
|
|
result.Deployed = true
|
|
cd.log("Successfully deployed Hyprland configuration")
|
|
return result, nil
|
|
}
|
|
|
|
func backupHyprlandConfigFile(src, dst string, data []byte, removeSource bool) error {
|
|
if err := os.MkdirAll(filepath.Dir(dst), 0o755); err != nil {
|
|
return err
|
|
}
|
|
if err := os.WriteFile(dst, data, 0o644); err != nil {
|
|
return err
|
|
}
|
|
if removeSource {
|
|
if err := os.Remove(src); err != nil && !os.IsNotExist(err) {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func backupLegacyHyprlandConfFiles(configDir, dmsDir, backupDir string) (int, error) {
|
|
legacyPaths := []string{filepath.Join(configDir, "hyprland.conf")}
|
|
dmsConfPaths, err := filepath.Glob(filepath.Join(dmsDir, "*.conf"))
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
legacyPaths = append(legacyPaths, dmsConfPaths...)
|
|
backupPaths, err := adjacentHyprlandBackupFiles(configDir, dmsDir)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
legacyPaths = append(legacyPaths, backupPaths...)
|
|
|
|
moved := 0
|
|
for _, src := range legacyPaths {
|
|
info, err := os.Lstat(src)
|
|
if os.IsNotExist(err) {
|
|
continue
|
|
}
|
|
if err != nil {
|
|
return moved, err
|
|
}
|
|
if info.IsDir() {
|
|
continue
|
|
}
|
|
|
|
rel, err := filepath.Rel(configDir, src)
|
|
if err != nil {
|
|
rel = filepath.Base(src)
|
|
}
|
|
dst := filepath.Join(backupDir, rel)
|
|
if err := moveHyprlandConfigFile(src, dst); err != nil {
|
|
return moved, err
|
|
}
|
|
moved++
|
|
}
|
|
|
|
return moved, nil
|
|
}
|
|
|
|
func moveHyprlandConfigFile(src, dst string) error {
|
|
if err := os.MkdirAll(filepath.Dir(dst), 0o755); err != nil {
|
|
return err
|
|
}
|
|
return os.Rename(src, dst)
|
|
}
|
|
|
|
func adjacentHyprlandBackupFiles(configDir, dmsDir string) ([]string, error) {
|
|
var paths []string
|
|
patterns := []string{
|
|
filepath.Join(configDir, "hyprland.conf.backup.*"),
|
|
filepath.Join(configDir, "hyprland.lua.backup.*"),
|
|
filepath.Join(dmsDir, "*.conf.backup.*"),
|
|
filepath.Join(dmsDir, "*.lua.backup.*"),
|
|
}
|
|
for _, pattern := range patterns {
|
|
matches, err := filepath.Glob(pattern)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
paths = append(paths, matches...)
|
|
}
|
|
return paths, nil
|
|
}
|
|
|
|
func (cd *ConfigDeployer) deployHyprlandDmsConfigs(dmsDir string, terminalCommand string) error {
|
|
configs := []struct {
|
|
name string
|
|
content string
|
|
overwrite bool
|
|
}{
|
|
{name: "colors.lua", content: DMSColorsLuaConfig},
|
|
{name: "layout.lua", content: DMSLayoutLuaConfig},
|
|
{name: "binds.lua", content: strings.ReplaceAll(DMSBindsLuaConfig, "{{TERMINAL_COMMAND}}", terminalCommand), overwrite: true},
|
|
{name: "binds-user.lua", content: DMSBindsUserLuaConfig},
|
|
{name: "outputs.lua", content: DMSOutputsLuaConfig},
|
|
{name: "cursor.lua", content: DMSCursorLuaConfig},
|
|
{name: "windowrules.lua", content: DMSWindowRulesLuaConfig},
|
|
}
|
|
|
|
for _, cfg := range configs {
|
|
path := filepath.Join(dmsDir, cfg.name)
|
|
existed := false
|
|
if info, err := os.Stat(path); err == nil && info.Size() > 0 {
|
|
existed = true
|
|
}
|
|
if existed && !cfg.overwrite {
|
|
cd.log(fmt.Sprintf("Skipping %s (already exists)", cfg.name))
|
|
continue
|
|
}
|
|
if err := os.WriteFile(path, []byte(cfg.content), 0o644); err != nil {
|
|
return fmt.Errorf("failed to write %s: %w", cfg.name, err)
|
|
}
|
|
if existed {
|
|
cd.log(fmt.Sprintf("Updated %s", cfg.name))
|
|
continue
|
|
}
|
|
cd.log(fmt.Sprintf("Deployed %s", cfg.name))
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (cd *ConfigDeployer) mergeHyprlandMonitorSections(newConfig, existingConfig, dmsDir string) (string, error) {
|
|
_ = newConfig
|
|
lines := extractHyprlangMonitorLines(existingConfig)
|
|
if len(lines) == 0 {
|
|
return newConfig, nil
|
|
}
|
|
|
|
outputsPath := filepath.Join(dmsDir, "outputs.lua")
|
|
if info, err := os.Stat(outputsPath); err == nil && info.Size() > 0 {
|
|
cd.log("Skipping monitor migration: dms/outputs.lua already exists")
|
|
return newConfig, nil
|
|
}
|
|
|
|
var b strings.Builder
|
|
b.WriteString("-- Migrated from existing hyprlang monitor lines\n\n")
|
|
ok := 0
|
|
for _, line := range lines {
|
|
lua, err := hyprlangMonitorLineToLua(line)
|
|
if err != nil {
|
|
cd.log(fmt.Sprintf("Warning: could not migrate monitor line %q: %v", line, err))
|
|
continue
|
|
}
|
|
b.WriteString(lua)
|
|
b.WriteByte('\n')
|
|
ok++
|
|
}
|
|
if ok == 0 {
|
|
return newConfig, nil
|
|
}
|
|
b.WriteByte('\n')
|
|
b.WriteString("-- Default fallback\n")
|
|
b.WriteString("hl.monitor({ output = \"\", mode = \"preferred\", position = \"auto\", scale = \"auto\" })\n")
|
|
if err := os.WriteFile(outputsPath, []byte(b.String()), 0o644); err != nil {
|
|
return newConfig, err
|
|
}
|
|
cd.log("Migrated monitor sections to dms/outputs.lua")
|
|
return newConfig, nil
|
|
}
|
|
|
|
func (cd *ConfigDeployer) transformNiriConfigForNonSystemd(config, terminalCommand string) string {
|
|
envVars := fmt.Sprintf(`environment {
|
|
XDG_CURRENT_DESKTOP "niri"
|
|
QT_QPA_PLATFORM "wayland;xcb"
|
|
ELECTRON_OZONE_PLATFORM_HINT "auto"
|
|
QT_QPA_PLATFORMTHEME "gtk3"
|
|
QT_QPA_PLATFORMTHEME_QT6 "gtk3"
|
|
TERMINAL "%s"
|
|
}`, terminalCommand)
|
|
|
|
config = regexp.MustCompile(`environment \{[^}]*\}`).ReplaceAllString(config, envVars)
|
|
|
|
spawnDms := `spawn-at-startup "dms" "run"`
|
|
if !strings.Contains(config, spawnDms) {
|
|
// Insert spawn-at-startup for dms after the environment block
|
|
envBlockEnd := regexp.MustCompile(`environment \{[^}]*\}`)
|
|
if loc := envBlockEnd.FindStringIndex(config); loc != nil {
|
|
config = config[:loc[1]] + "\n" + spawnDms + config[loc[1]:]
|
|
}
|
|
}
|
|
|
|
return config
|
|
}
|