Skip to content

yaklabco/dot

Repository files navigation

dot logo

dot

CI Release Version Go Report Card License: MIT

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.26 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:

  1. Explicit flag: --dir /path/to/dotfiles (highest priority)
  2. Environment variable: DOT_PACKAGE_DIR=/path/to/dotfiles
  3. Current directory: If current directory contains .dotbootstrap.yaml
  4. Parent search: Searches up directory tree for .dotbootstrap.yaml
  5. Configuration file: From directories.package in config
  6. 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-bashrcconfig/.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

dot CLI help

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
dot adopt interactive UI

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

Tool Upgrade

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):

  1. System-wide: /etc/dot/config.yaml
  2. User global: ~/.config/dot/config.yaml (XDG) or ~/.dotrc
  3. Project local: ./.dotrc
  4. Environment variables: DOT_* prefix
  5. 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

  1. Domain Layer: Pure domain model with phantom-typed paths for compile-time safety
  2. Core Layer: Pure functional logic for scanning, planning, and resolution
  3. Pipeline Layer: Composable pipeline stages with generic type parameters
  4. Executor Layer: Side-effecting operations with two-phase commit and rollback
  5. API Layer: Clean public Go library interface for embedding
  6. 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

  1. Fork the repository
  2. Create a feature branch
  3. Write tests for new functionality
  4. Implement the feature
  5. Ensure all tests and linters pass
  6. 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.26
  • 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.

About

A cli for managing dotfiles

Resources

License

Contributing

Security policy

Stars

Watchers

Forks

Packages

 
 
 

Contributors

Languages