This repository contains the source code for dotfilers
, a dotfile management utility written in Rust.
You can either grab the latest release or build it yourself:
$ cargo build --release
You will find the binary in target/release/dotfilers
.
Add the binary to your path, and you are ready to go!
By default, when invoking dotfilers
it will look for a dotfilers.yaml
file in the current directory. However, you can specify a custom file by using -c/--config PATH
. Keep in mind that the current working directory will be used for calculating relative paths.
When you are testing some configurations, you can pass -d/--dry-run
in order not to perform any actual operation. If invoked in dry-run mode, dotfilers will print the operations that would be executed, but won't actually perform any operation.
Also, in case you only want to apply some of your dotfilers.yaml
sections, you can pass the section names as arguments. Let's say you only want to execute your nvim
and ssh
sections. In order to do so, you can run dotfilers nvim ssh
.
You can have a .dotfilers
section in your dotfilers.yaml
that will configure behaviours for the dotfilers
binary.
Here you can find an example with the default values:
.dotfilers:
# Log level for the dotfilers binary
# Must be one of:
# - trace
# - debug
# - info
# - warn
# - error
log_level: info
# Strategy to use in case of conflict (the file that would be created already exists)
# Must be one of:
# - abort (the program will stop)
# - overwrite (the already existing file/directory will be removed)
# - rename-old (the already existing file/directory will be renamed to NAME.bak, and in case it also exists, .bak1, .bak2...)
conflict_strategy: rename-old
# Shell that will be used for 'run' directives
shell: /bin/bash -c
In the root of your dotfilers.yaml
you can specify the different sections you want to manage.
A section starts with a name, and then a list of operations to be performed. Here you can find an example of all the available operations:
# Section for zsh files
zsh:
# Create a symlink from zsh/.zshrc to ~/.zshrc
- link_from: zsh/.zshrc
link_to: ~/.zshrc
# Only if the os is linux, create a symlink from zsh/.zshrc.linux to ~/.zshrc.local
- if_os: linux
link_from: zsh/.zshrc.linux
link_to: ~/.zshrc.local
# Only if the os is darwin, create a symlink from zsh/.zshrc.linux to ~/.zshrc.local
- if_os: darwin
link_from: zsh/.zshrc.darwin
link_to: ~/.zshrc.local
# Section for ssh files
ssh:
# Only if the os is linux:
# - use the template on ssh/config.tpl
# - fill it using the vars from ssh/vars_linux
# - store the result at ~/.ssh/config
- if_os: linux
template: ssh/config.tpl
template_to: ~/.ssh/config
vars: ssh/vars_linux
# Only if the os is darwin:
# - use the template on ssh/config.tpl
# - fill it using the vars from ssh/vars_darwin
# - store the result at ~/.ssh/config
- if_os: darwin
template: ssh/config.tpl
template_to: ~/.ssh/config
vars: ssh/vars_darwin
# Copy all files that match id_rsa* from the ssh folder into ~/.ssh/
- copy_from: ssh/id_rsa*
copy_to: ~/.ssh/
# Section for nvim files
nvim:
# As this section is too long, run the contents of nvim/deploy.yaml
- include: nvim/deploy.yaml
# Section for mash files
mash:
# This will recursively link all files inside the mash directory into ~/mash
# However, any existing directories will be created rather than symlinked
- link_from: mash
link_to: ~/mash
link_directory_behaviour: create
other:
# This will symlink all files found inside the 'other' directory,
# but will ignore any directory found inside the 'other' directory
- link_from: other/*
link_to: ~/.config/other
link_directory_behaviour: ignore
# Section for extra things
extra:
# Run inline code
- run: echo "abc" > /tmp/abc
# Run multiple commands
- run: |
echo "some_contents" >> /tmp/abc
echo "Other contents" >> /tmp/abc
# You can even invoke other programs/scripts
- run: /usr/bin/python3 -c "print('a' * 127)" > /tmp/lotsofas
Here you can find a detailed list of all the directives that are supported.
Please notice that if you declare multiple options in the same directive entry (such as link_from
, link_to
, copy_from
, copy_to
) the resulting operation does not have any guarantee of being consistent, so please avoid doing so.
Copy files or directories from one location to another. This command supports globs in the copy_from
section.
Sections:
copy_from
: Which file / directory to be copied.- It supports globs, such as
ssh/id_rsa*
ordirectory/*.txt
.
- It supports globs, such as
copy_to
: Where to copy the files.- If the
copy_from
is a single file, please also write the desired destination filename (such as:copy_to: ~/.ssh/authorized_keys
). - If the
copy_from
is a glob, you should use the path to the destination directory (such as:copy_from: directory/*.txt
andcopy_to: ~/data
).
- If the
Link files or directories from one location to another. This command supports globs in the link_from
section.
Sections:
link_from
: Which file / directory to be symlinked.- It supports globs, such as
ssh/id_rsa*
ordirectory/*.txt
.
- It supports globs, such as
link_to
: Where to symlink the files.- If the
link_from
is a single file, please also write the desired destination filename (such as:link_to: ~/.ssh/authorized_keys
). - If the
link_from
is a glob, you should use the path to the destination directory (such as:link_from: directory/*.txt
andlink_to: ~/data
).
- If the
link_directory_behaviour
: What to do with directories. If not specified defaults tolink
.- If set to
create
, all the directories inside the dir will be created as directories rather than symlinked to the directory. Then, the files inside the original directory will be recursively symlinked. - If set to
link
, any directories found will just be symlinked. - If set to
ignore
any directories that match thelink_from
(specially if it's a glob) will be ignored.
- If set to
You can also generate files on the fly by filling templates. dotfilers
uses Tera as a templating engine, so please refer to the Tera documentation for templates.
In order to define variables, you can create a file with any name you want, and fill the variables with the following format:
variable1=value1
# Comments are ignored, but are only supported
# at the beginning of the line
variable2=value2
There are some variables that are filled by dotfilers
itself. For now these variables are:
dotfilers_os
: The current OS. May either belinux
ordarwin
.
You can run arbitrary commands with dotfilers
.
There is no better documentation that some examples:
# Run inline code
- run: echo "abc" > /tmp/abc
# Run multiple commands
- run: |
echo "some_contents" >> /tmp/abc
echo "Other contents" >> /tmp/abc
# You can even invoke other programs/scripts
- run: /usr/bin/python3 -c "print('a' * 127)" > /tmp/lotsofas
Keep in mind that if the exit status code is not 0, the execution will abort.
For very long sections it may be handy to delegate the directives into another file. dotfilers
supports doing so by using the include
directive.
The included files must have the same structure as the dotfilers.yaml
file, so that means you will need to wrap your directives into a section.
# dotfilers.yaml
nvim:
- include: nvim/directives.yaml
# nvim/directives.yaml
nvim:
- if_os: linux
run: echo "This is linux"
MIT License
Copyright (c) 2022 Carlos Quintana
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.