dot

A type-safe symbolic link manager for configuration files and dotfiles, written in Go.
Overview
dot manages configuration files through symbolic links, providing a centralized approach to dotfile management with strong safety guarantees. The tool creates and maintains symlinks from a directory (package repository) to a target directory (typically home directory), enabling version control and synchronization of configuration files across multiple machines.
Key Capabilities
- Repository Cloning: Single-command setup on new machines with optional bootstrap configuration
- Package Management: Install, remove, and update packages containing configuration files
- Conflict Resolution: Detect and resolve conflicts with configurable resolution policies
- Incremental Operations: Content-based change detection for efficient updates
- Transactional Safety: Two-phase commit with automatic rollback on failure
- State Tracking: Manifest-based state management for fast status queries
- Parallel Execution: Concurrent operation processing for improved performance
- Cross-Platform: Supports Linux, macOS, BSD, and Windows (with limitations)
Installation
Homebrew (macOS/Linux)
brew tap yaklabco/dot
brew install dot
From Binary Releases
Download the latest release for your platform from GitHub Releases.
# Linux/macOS
curl -L https://github.com/yaklabco/dot/releases/latest/download/dot-$(uname -s)-$(uname -m).tar.gz | tar xz
sudo mv dot /usr/local/bin/
From Source
Requires Go 1.25.4 or later:
go install github.com/yaklabco/dot/cmd/dot@latest
Verification
dot --version
Quick Start
New Machine Setup
Clone an existing dotfiles repository:
dot clone https://github.com/user/dotfiles
Like git clone, this creates a subdirectory named after the repository (e.g., ./dotfiles). This single command:
- Clones the repository to the local directory
- Selects packages to install (via profile or interactively)
- Creates all symlinks
- Tracks repository information for updates
- Offers to save the package directory location to your config
The repository can include:
- Configuration (
.config/dot/config.yaml): Repository-specific dot configuration that's automatically used after clone
- Bootstrap (
.dotbootstrap.yaml): Package selection profiles, platform requirements, and installation policies
Repository configuration allows your repository to define how it should be managed without circular dependency. See Repository Configuration for details.
Bootstrap configuration enables installation profiles and platform-specific package selection. See Bootstrap Specification for details.
Package Directory Resolution
dot intelligently locates your dotfiles repository using the following precedence:
- Explicit flag:
--dir /path/to/dotfiles (highest priority)
- Environment variable:
DOT_PACKAGE_DIR=/path/to/dotfiles
- Current directory: If current directory contains
.dotbootstrap.yaml
- Parent search: Searches up directory tree for
.dotbootstrap.yaml
- Configuration file: From
directories.package in config
- Default fallback:
~/.dotfiles (lowest priority)
This means you can cd into your dotfiles directory and run dot commands without flags, or set DOT_PACKAGE_DIR for project-wide consistency.
Initial Setup
Create a package directory to store your packages:
mkdir -p ~/dotfiles/dot-vim
echo "set number" > ~/dotfiles/dot-vim/vimrc
Manage a Package
Install the package by creating symlinks:
cd ~/dotfiles
dot manage dot-vim
This creates ~/.vim/vimrc pointing to ~/dotfiles/dot-vim/vimrc.
Package names determine target directories:
dot-vim → ~/.vim/
dot-gnupg → ~/.gnupg/
vim → ~/vim/
Check Status
View installed packages and their status:
dot status
Unmanage a Package
Remove symlinks for a package:
dot unmanage dot-vim
Core Concepts
Package Directory
The source directory containing packages. Each subdirectory represents a package. Default: current directory.
Target Directory
The destination directory where symlinks are created. Default: $HOME.
Package
A directory within the package directory containing configuration files.
Package Name Mapping
Package names determine target directories (default behavior):
- Package
dot-vim → files installed to ~/.vim/
- Package
dot-gnupg → files installed to ~/.gnupg/
- Package
config → files installed to ~/config/
This eliminates redundant nesting and makes package naming intuitive.
Dotfile Translation
Files prefixed with dot- are translated to dotfiles (leading .):
- Within
dot-vim/: dot-vimrc → .vim/.vimrc
- Within
config/: dot-bashrc → config/.bashrc
Both package-level and file-level translation work together.
Directory Folding
When all files in a directory belong to a single package, dot creates a single directory-level symlink instead of individual file symlinks, reducing symlink count and improving performance.
Usage
Package Management Commands
Manage (Install)
Create symlinks for packages:
# Single package
dot manage dot-vim
# Multiple packages
dot manage dot-vim dot-tmux dot-zsh
# With options
dot --dir ~/dotfiles --target $HOME manage dot-vim
dot --dry-run manage dot-vim # Preview changes
Unmanage (Remove)
Remove symlinks for packages:
# Single package
dot unmanage dot-vim
# Multiple packages
dot unmanage dot-vim dot-tmux
# Preview removal
dot --dry-run unmanage dot-vim
Remanage (Update)
Update packages (remove and reinstall with incremental detection):
# Update packages efficiently
dot remanage dot-vim
# Updates only changed packages
dot remanage dot-vim dot-tmux dot-zsh
Adopt (Import)
Move existing files into a package and replace with symlinks.
Interactive Mode
Run without arguments to interactively discover and select dotfiles. This mode scans your home directory for potential dotfiles and presents a TUI for selection.
dot adopt
Command Line Mode
Use for scripting or known files:
# Auto-naming: single file (package name derived from filename)
dot adopt ~/.vimrc
# Creates package: dot-vimrc
# Auto-naming: single directory (package name derived from directory)
dot adopt ~/.ssh
# Creates package: dot-ssh
# Explicit package: specify package name for multiple files
dot adopt vim ~/.vimrc ~/.vim
dot adopt zsh ~/.zshrc ~/.zprofile ~/.zshenv
# Shell glob expansion: specify package name
dot adopt git .git*
# Package "git" with all .git* files
Note: When adopting multiple files via command line, you must provide an explicit package name as the first argument.
Interactive Options
dot adopt --scan-dirs ~/.config # Scan additional directories
dot adopt --max-size 100MB # Increase file size limit
Repository Commands
Clone
Clone a dotfiles repository and set up on a new machine:
# Basic clone (creates ./dotfiles directory)
dot clone https://github.com/user/dotfiles
# Clone creates ./my-dotfiles directory based on repo name
dot clone https://github.com/user/my-dotfiles
# Clone with specific profile
dot clone https://github.com/user/dotfiles --profile minimal
# Clone specific branch
dot clone https://github.com/user/dotfiles --branch develop
# Clone to specific directory (overrides default)
dot clone --dir ~/packages https://github.com/user/dotfiles
The clone command:
- Clones the Git repository into a subdirectory (named after the repo, like
git clone)
- Reads
.dotbootstrap.yaml if present for configuration
- Prompts for package selection (or uses specified profile)
- Creates all symlinks for selected packages
- Tracks repository information in manifest for future updates
Clone Bootstrap
Generate a .dotbootstrap.yaml configuration file from your current installation:
# Generate in package directory
dot clone bootstrap
# Specify output location
dot clone bootstrap --output ~/dotfiles/.dotbootstrap.yaml
# Preview without writing
dot clone bootstrap --dry-run
# Only include currently installed packages
dot clone bootstrap --from-manifest
# Set default conflict policy
dot clone bootstrap --conflict-policy backup
# Overwrite existing file
dot clone bootstrap --force
The generated configuration includes:
- All discovered packages (or only installed ones with
--from-manifest)
- Default conflict resolution policy
- Commented template for defining profiles
- Helpful documentation links
After generation, customize the file to:
- Mark packages as required
- Define installation profiles (minimal, full, development)
- Add platform-specific package restrictions
- Set per-package conflict policies
See Bootstrap Configuration Specification for details.
Query Commands
Status
Display installation status:
# All packages
dot status
# Specific packages
dot status dot-vim dot-tmux
# Different formats
dot status --format json
dot status --format yaml
dot status --format table
Doctor
Validate installation consistency and detect issues:
# Health check
dot doctor
# With detailed output
dot doctor -v
# JSON output for scripting
dot doctor --format json
Exit codes:
- 0: No issues detected
- 1: Issues found
List
Show installed packages:
# List all packages
dot list
# Sort by various fields
dot list --sort name # Alphabetical (default)
dot list --sort links # By link count
dot list --sort date # By installation date
# Different formats
dot list --format table
dot list --format json
Global Options
-d, --dir PATH Package directory (default: current directory)
-t, --target PATH Target directory (default: $HOME)
-n, --dry-run Preview changes without applying
-v, --verbose Increase verbosity (repeatable: -v, -vv, -vvv)
--quiet Suppress non-error output
--log-json Output logs in JSON format
--no-folding Disable directory folding optimization
--absolute Use absolute symlinks instead of relative
--ignore PATTERN Add ignore pattern (repeatable)
Maintenance Commands
Update dot to the latest version:
# Check for and install updates
dot upgrade
# Check without installing
dot upgrade --check-only
Configuration
dot supports configuration files in YAML, JSON, or TOML formats.
Configuration Management
Manage configuration settings directly from the CLI:
# Initialize a new configuration file
dot config init
# View current configuration
dot config list
# Get a specific value
dot config get directories.package
# Set a value
dot config set logging.level debug
# Show configuration file path
dot config path
# Upgrade configuration format
dot config upgrade
Configuration Locations
Searched in order (later sources override earlier):
- System-wide:
/etc/dot/config.yaml
- User global:
~/.config/dot/config.yaml (XDG) or ~/.dotrc
- Project local:
./.dotrc
- Environment variables:
DOT_* prefix
- Command-line flags (highest priority)
Configuration Example
# ~/.config/dot/config.yaml
packageDir: ~/dotfiles
targetDir: ~
linkMode: relative
folding: true
verbosity: 0
ignore:
- "*.log"
- ".git"
- ".DS_Store"
- "*.swp"
override: []
backupDir: ~/.dot-backups
Configuration Options
| Option |
Type |
Default |
Description |
packageDir |
string |
. |
Source directory containing packages |
targetDir |
string |
$HOME |
Destination for symlinks |
linkMode |
string |
relative |
Link mode: relative or absolute |
folding |
boolean |
true |
Enable directory folding optimization |
verbosity |
integer |
0 |
Logging verbosity (0-3) |
ignore |
array |
(defaults) |
File patterns to exclude |
override |
array |
[] |
Patterns to force include |
backupDir |
string |
(none) |
Directory for conflict backups |
See Configuration Reference for complete documentation.
Conflict Resolution
dot detects conflicts when existing files or symlinks prevent package installation.
Resolution Policies
Configure via --on-conflict flag or configuration file:
- fail (default): Stop and report conflicts
- backup: Move conflicting files to backup location
- overwrite: Replace conflicting files with symlinks
- skip: Skip conflicting files and continue
Example:
# Backup existing files
dot --on-conflict backup manage vim
# Skip conflicts and continue
dot --on-conflict skip manage vim tmux
See User Guide - Workflows for conflict resolution strategies.
System Requirements
Operating Systems
- Linux (all distributions)
- macOS 10.15 or later
- FreeBSD, OpenBSD, NetBSD
- Windows 10 or later (with symlink support enabled)
Filesystems
Full support:
- ext4, btrfs, xfs (Linux)
- APFS, HFS+ (macOS)
- ZFS (all platforms)
Limited support:
- FAT32, exFAT (no symlink support)
- Network filesystems (NFS, SMB) with caveats
Architectures
- amd64 (x86-64)
- arm64 (aarch64)
- 386 (x86)
- arm (32-bit ARM)
Documentation
Complete documentation index available at docs/README.md.
User Documentation
Developer Documentation
Examples
Development
Building
make build
Testing
# Run all tests
make test
# Run with coverage
make test-coverage
# Run integration tests
make test-integration
Linting
# Run all linters
make lint
# Run specific checks
make lint-go
make lint-docs
Quality Checks
# Run complete quality suite
make check
This runs tests, linting, and builds in sequence.
Architecture
dot follows a layered architecture with functional programming principles:
Layers
- Domain Layer: Pure domain model with phantom-typed paths for compile-time safety
- Core Layer: Pure functional logic for scanning, planning, and resolution
- Pipeline Layer: Composable pipeline stages with generic type parameters
- Executor Layer: Side-effecting operations with two-phase commit and rollback
- API Layer: Clean public Go library interface for embedding
- CLI Layer: Cobra-based command-line interface
Design Principles
- Functional Core, Imperative Shell: Pure planning with isolated side effects
- Type Safety: Phantom types prevent path-related bugs at compile time
- Explicit Errors: Result types and error aggregation, never silent failures
- Transactional: Two-phase commit with automatic rollback on errors
- Observable: Structured logging, distributed tracing, metrics collection
- Testable: Pure core enables property-based testing of algebraic laws
See User Guide - Advanced Features for implementation details.
Library Usage
dot can be embedded as a Go library:
package main
import (
"context"
"github.com/yaklabco/dot/pkg/dot"
)
func main() {
cfg := dot.Config{
PackageDir: "/home/user/dotfiles",
TargetDir: "/home/user",
LinkMode: dot.LinkRelative,
Folding: true,
}
client, err := dot.New(cfg)
if err != nil {
panic(err)
}
ctx := context.Background()
if err := client.Manage(ctx, "vim", "tmux"); err != nil {
panic(err)
}
}
Contributing
Contributions are welcome. All contributions must follow project standards:
Requirements
- Test-driven development: write tests before implementation
- Minimum 75% test coverage for new code
- All linters must pass without warnings
- Conventional Commits specification for commit messages
- Atomic commits: one logical change per commit
- Academic documentation style: factual, precise, no hyperbole
Process
- Fork the repository
- Create a feature branch
- Write tests for new functionality
- Implement the feature
- Ensure all tests and linters pass
- Submit a pull request
See Contributing Guide for detailed guidelines.
License
MIT License - see LICENSE file for details.
Project Standards
This project adheres to strict development standards:
- Language: Go 1.25.4
- Development: Test-Driven Development (TDD) mandatory
- Testing: Minimum 75% coverage, property-based tests for core logic
- Commits: Atomic commits with Conventional Commits format
- Code Style: golangci-lint v2 with comprehensive linter set
- Documentation: Academic style, factual, technically precise
- Versioning: Semantic Versioning 2.0.0
- Changelog: Keep a Changelog format
Comparison with GNU Stow
dot is inspired by GNU Stow. dot provides feature parity with GNU Stow plus additional capabilities:
| Feature |
dot |
GNU Stow |
| Basic stow/unstow |
Yes |
Yes |
| Conflict detection |
Yes |
Yes |
| Directory folding |
Yes |
Yes |
| Incremental updates |
Yes |
No |
| Transactional operations |
Yes |
No |
| Parallel execution |
Yes |
No |
| Adopt existing files |
Yes |
No |
| Status/health checks |
Yes |
No |
| Multiple output formats |
Yes |
No |
| Type safety |
Yes |
No |
| Cross-platform |
Yes |
Limited |
See Migration Guide for transitioning from GNU Stow.
Support
Project Status
Current Version: v0.6.0
Stability: Stable
See CHANGELOG for release history.
Acknowledgments
Inspired by GNU Stow, reimagined with modern language features and safety guarantees. Made with ❤️ in Go.