Skip to content
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

Symlink config to multiple locations? #186

Open
jack-mil opened this issue Sep 17, 2024 · 6 comments
Open

Symlink config to multiple locations? #186

jack-mil opened this issue Sep 17, 2024 · 6 comments
Labels

Comments

@jack-mil
Copy link

jack-mil commented Sep 17, 2024

Environment

  • OS: Ubuntu / Debian / Windows 10
  • Dotter version: 0.13.3

Multiple destinations for same package file?

From what I have tried, it doesn't seem like this is possible at the moment. What I am trying to achieve is something like:

[pwsh.files]
  pwsh = '~\Documents\PowerShell'

[powershell.files]
  pwsh = '~\Documents\WindowsPowerShell'

with the package folder pwsh/ looking like

pwsh\
└── Microsoft.PowerShell_profile.ps1

This works fine if only "pwsh" or "powershell" package is chosen (mutually exclusive, as seen in examples), but I would like to pick both. I want to link the profile.ps1 file to both locations. Is it possible right now? The only potential workaround I could think of is have one profile symlinked to the other in the dotfiles repository. But that could get messy with doubly linked symlinks (which I don't know if it is even possible? Symlink to a symlink?)

The error I get is (unsurprisingly):

[ERROR] Failed to deploy
Caused by:
    get a configuration
    merge configuration files
    merge package "pwsh"
    file "pwsh" already encountered
@quintrino
Copy link

quintrino commented Sep 22, 2024

So I'm not sure if this would solve your problem, but just in case it's helpful I've dealt with something similar by having two seperate files and having them both load the same template.

Just note that this would mean that the powershell file in each would be dynamically generated and therefore any changes made wouldn't flow back to the original. That may or may not matter to you.

If you need help setting this up, just let me know.

@hartigan43
Copy link

hartigan43 commented Oct 9, 2024

Similar issue and for me here:

[vim.files]
"vim/vimrc"  = { target = "~/.vimrc", type = "symbolic" }
"vim/vimrc"  = { target = "~/.config/nvim/init.vim", type = "symbolic", if = '(is_executable "nvim")' }

I should just move on from vim and wholly onto neovim, but I think there might be other valid use cases for this pattern?

EDIT:
In my case, a simple manual symlink to ~/.vimrc will suffice, as dotter deploys the last one read based on dotter/cache.toml

@jack-mil
Copy link
Author

jack-mil commented Oct 10, 2024

I keep running into other cases where it might be useful to have a configuration file shared across multiple dotter "packages". Today it was because I want the same MPV config for regular mpv-git and for an different mpv distribution that has a different config location.

How complex would implementing this feature be? I am guessing there is some architectural reason that this doesn't work out of the box. In toml, a table cannot have duplicate keys (so your example @hartigan43 is invalid toml), but the powershell example in my original post is valid. I would expect that to work, but there must be something that dotter is doing that requires (all?) keys to be unique, regardless of which table they are in?

I'm still a Rust beginner, but I will take a look myself when I get the chance.

Addendum: A similar restriction requires all variable names to be unique, across all packages. This is actually something that confused me starting out, because I had the (mistaken) assumption that variables would be scoped to packages.

@serkonda7
Copy link

I encountered this problem with Visual Studio code as I use the Code and Code - OSS builds. My workaround is to symlink on directory to the other in the post deploy script.

However I would really like to see an official feature supporting this.

@SuperCuber
Copy link
Owner

I see it's a pretty highly requested feature and I see the uses, but I don't have the time and motivation right now to implement it. There's a couple design restrictions here

  1. TOML does not support duplicated keys, so the multiple targets would have to live as the value of a singular key
  2. There's an existing behavior of overriding file targets and disabling files with "" target which needs to still be supported

I can't think of the top of my head of a satisfying design for how to specify it in the configuration file, and I'm pretty sure the implementation assumes that there's one target per source so it might need some sweeping changes to support this.

I'm welcoming PRs on this :)

@jack-mil
Copy link
Author

Some thoughts on the user configuration side for supporting this feature.
For example, this configuration is invalid toml, and would not be supported:

# global.toml
# INVALID
[code.files]
vscode = "~/.config/Code/User"
vscode = "~/.config/VSCodium/User"

What makes more sense is to scope each app to a seperate package, like this:
This configuration would be supported with keys scoped to the table names.

# global.toml
# OK, cannot deploy both 'vscode' and 'codium' packages (currently)
[vscode.files]
vscode = "~/.config/Code/User"

[codium.files]
vscode = "~/.config/VSCodium/User"

I think this is what most people would need. I believe it would be simpler than allowing a list of targets, as that could complicate the specification quite a bit.
This configuration works currently, as long as you only pick to deploy either 'vscode' or 'codium' packages. The issue arrises when packages are merged together into one global namespace.

When a local.toml configuration requests packages = ["vscode", "codium"], the implementation trys to merge all packages into one, and runs into a duplicate key error. This is done so that a local.toml config can override individual file targets at the filename level with the [files] table. At the local level, all filenames and variable package scope is discarded, and merged into one "package". I found this confusing starting out, as I expected to be able to override targets at the local.toml level with additional [<package>.files] tables.

(Consequently, allowing defining packages at the local (untracked) level might be a useful feature)

In summary, a proposed local.toml format that would work with the above global definition could look like:

# local.toml
packages = ["codium", "vscode"]

# additional overrides can be specified like

[codium.files]
vscode = "~/my/local/location"
[vscode.files]
vscode = ""

# could even specify package level variable overrides...
[vscode.variables]
special_var = "value"

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

No branches or pull requests

5 participants