Dotfiles management with Copier
Introduction
Dotfiles are the configuration files to customize the shell and other command-line tools. They are usually hidden in the user’s home directory, and they are called “dotfiles” because their filenames start with a dot (.
).
Dotfiles are present on every Unix-like system, the most popular ones are .bashrc
or .zshrc
(for zsh shell), who are loaded when the shell starts. They can contain aliases, functions, environment variables, and other settings that customize the shell environment to the user’s preferences.
On Windows they also exist, for example the profile file for PowerShell
, which is located in the $PROFILE
environment variable.
As someone who works on multiple machines and operating systems, I find it important to have a consistent environment across all of them, especially when I found and tried random configurations regularly. After some time it’s hard to keep track of all the changes and customizations, and even harder to replicate them across machines.
This is why I started to look for a solution to manage my dotfiles. This is not a new problem, and there are many tools and methods readily available, and I will discuss some of them in this post.
But first, let’s show my current context and what I want to achieve with a dotfiles management solution.
Context
Currently I find myself dabbling with multiple machines on a daily basis, each one with a different OS and shell environment.
- My personal PC runs
Windows 11
, withPowerShell v7
as the default shell. - I also use
Debian
viaWSL2
on my personal PC, withzsh
as the default shell. - I have a mini PC running
Ubuntu
, withzsh
as the default shell. - My work laptop runs
macOS
, withzsh
as the default shell. - I also work with multiple
Ubuntu
servers, withbash
as the default shell. Normally I am not allowed to customize these servers, but I include them anyway for the complete picture.
As you can see, the situation is quite complex, and it is hard to switch contexts without slowing down my workflow. A consistent environment across all machines won’t solve all the problems, but it will make the transition between them much smoother.
The ideal solution I am looking for should have the following features:
- Backup my current dotfiles to a central repository with version control.
- Install the dotfiles to any machine with minimal effort.
- Support for custom scripts and configurations during the installation process.
- Support for different operating systems and shells, including
Windows
andPowerShell
. - Allow to customize profiles on each machine (personal, work, etc.).
- Allow to sync updates across all machines without erasing local customizations.
- Should be able to sync outside of the
$HOME
directory.
Existing solutions and their limitations
There are many tools and methods available to manage dotfiles, where the most common approaches can be summarized into the following categories:
- Git bare repository
- Symlink: for example GNU Stow
- Template and overwrite target files: for example Chezmoi
These solutions are elegant and should work well for most users. I have read and tried some of them, and almost settled with chezmoi
for its flexible features.
However, they all share 2 common limitations that doesn’t satisfy my requirements:
- Unable to manage files outside of the
$HOME
directory. This is related to myPowerShell
configuration, where$PROFILE
is located inD:\
, not in$HOME
(C:\Users\username\
). - Unable to sync updates without erasing local customizations. This one is quite specific, and kinda defeats the purpose of a homogeneous environment. But let’s consider the following scenario: I use Ruby on a specific machine to manage my blogs (this one), so I need to extend my
$PATH
for the Ruby binaries in~/.zshrc
:
1
export PATH="$HOME/gems/bin:$PATH"
I don’t need this line anywhere else, nor do I want to add it to my dotfiles permanently. However, any current solution will erase this line when syncing updates, which is also not I want. What I want is similar to the git cherry-pick
feature, where changes are applied selectively, and not just blindly overwritten.
I know my life would be much easier if I just add this line to the central repository and forget about it, but I was stubborn and took it as a challenge to make it work, so here we are.
Solution using Copier
What is Copier?
I came across Copier at work, where I use it to create templates to synchronize projects structure across my organization’s repositories. It offers a lot of flexibility and customization using templates and task execution, which is very similar to chezmoi
.
However, one of its most powerful features is to update and synchronize the project to the latest template version, using a sophisticated workflow based on git diff
to apply the changes selectively.
This was the missing piece of the puzzle I was looking for when I was using chezmoi
, where the only way to sync updates was to overwrite local changes.
Manage dotfiles with Copier
The main idea is a combination of existing solutions, with multiple components:
- A central repository to store the dotfiles as template. It is hosted here on my dotfiles repository.
- A local repository to render the dotfiles on the target machine from the central template.
- Python scripts to backup then symlink the local files to the correct location. The script is executed automatically when the local repository is updated.
A simplified routines for initial setup and updating the dotfiles would look like this:
Prerequisites
Copier is not the only required component for this solution, it is combined with another powerful tool, uv
. The advantages of using uv
are numerous:
- It can install Copier independently as a CLI tool.
- It is cross-platform
- It does not require an existing Python installation.
The 2nd point is particularly important. It also allows me to write Python scripts to automate the dotfiles management, instead of writing separate shell scripts for each platform, which is quite a pain (looking at you, PowerShell
). The power of uv
is not limited to this, but I will not cover it in this post, maybe in a future post dedicated to it.
To install uv
:
1
curl -LsSf https://astral.sh/uv/install.sh | sh
For other platforms, please refer to the official documentation.
To install Copier as a CLI tool:
1
uv tool install copier
This allows me to launch Copier directly with copier
command, or through uv with uvx copier
(equivalent to uv tool copier
).
Initial setup
The initial setup is straightforward, just clone the central repository using the copier
command:
1
uvx copier copy gh:ldkv/dotfiles.git /path/to/local/repository --trust
You will be prompted to answer some questions, such as the user name and email, or any other questions defined in the copier.yml
file. The template will be then rendered to the folder based on the configurations, and a oneshot script will be executed to install anything I define, such as oh-my-zsh
and its plugins. The --trust
flag is required to allow the script to be executed.
The script will also initialize a git repository in the local folder, which is required by Copier
for its update workflow.
Update routine
For the update routine to work normally, there are 2 requirements:
- Generate a new tag in Github for each release. This is also a good practice to have a history of changes.
- The local repository doesn’t have uncommited changes. This should not be a problem since all local changes are committed automatically by the script.
To execute the update routine:
1
uvx copier update /path/to/local/repository -A --trust
This will apply the changes selectively to the local repository, you will be able to review and resolve any conflicts. Finally, an update script will be executed to do the following:
- Detect all managed dotfiles based on local folder structure. The script find the managed dotfiles from 2 sources in the local repository:
- All files in the
common
folder. - All files in
unix
orwindows
folder, based on the OS. If we are on Windows, theunix
folder won’t be rendered, and vice versa.
- All files in the
- Backup all existing dotfiles to a
backups
subfolder. This allows to restore the previous dotfiles if there were any problems during the update. - Create symlinks to the new dotfiles.
- Commit all local changes to the repository, including the backups.
Customizations
At this point you will wonder how does this solve my 1st requirement mentioned earlier, about managing files outside of the $HOME
directory?
In fact, the script takes into account a mappings defined in the configs.json
file, where I can specify a custom target to any path outside of the $HOME
directory for a given dotfile.
If no mapping is defined, the dotfile will be linked to the $HOME
folder by default, based on its relative path in the local repository. For example, a file located in common/.config/.abc
will be linked to $HOME/.config/.abc
.
The Python script can manipulate files using native Pathlib
package, but it can also execute any command using subprocess.run
. It also allows me to adapt the script to the target platform easily, should the need arise.
With this approach, the possibilities for customizations are endless. For example, in the same configs JSON, I also define a list of external plugins for oh-my-zsh
to install during the initial setup, but only triggered on Unix systems.
Final thoughts
This concludes my dotfiles management solution. It is not perfect, but it is a good compromise between flexibility and ease of use. Above all, it fits all my needs, and I enjoyed tackling the challenge with multiple iterations, learning a lot during the process.
I am sure there are many improvements that can be made, such as automating the updates whenever a new version is released, or script to revert the changes using the backups, etc. I will probably work on it in the future, and I will share the progress here.
Finally, as this post is dedicated to the management solution, I didn’t go into details about my dotfiles preferences, and I will save that for another post.