config

A simple configuration manager

NameSizeMode
..
config 4081B -rwxr-xr-x
001
002
003
004
005
006
007
008
009
010
011
012
013
014
015
016
017
018
019
020
021
022
023
024
025
026
027
028
029
030
031
032
033
034
035
036
037
038
039
040
041
042
043
044
045
046
047
048
049
050
051
052
053
054
055
056
057
058
059
060
061
062
063
064
065
066
067
068
069
070
071
072
073
074
075
076
077
078
079
080
081
082
083
084
085
086
087
088
089
090
091
092
093
094
095
096
097
098
099
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
#!/bin/bash
#                   __ _       
#   ___ ___  _ __  / _(_) __ _ 
#  / __/ _ \| '_ \| |_| |/ _` |
# | (_| (_) | | | |  _| | (_| |
#  \___\___/|_| |_|_| |_|\__, |
#                        |___/ 
#
# A simple configuration manager
# 2021 (C) Pablo
# Free use of this software is granted under the terms of the GPL-3.0 License

print_files() 
{
  IFS='' read -r file

  if [ -n "$file" ]
  then
    printf '\033[0;%sm%s files:\033[0m\n' "$2" "$1"
    while true
    do
      printf '\033[0;%sm%s\033[0m\n' "$2" "$file"
      IFS='' read -r file
      [ -z "$file" ] && break
    done
  fi
}

# Get the path to the git repository that should be used for storing stuff
if [ -n "$DOTFILES_REPO" ]
then
  dotfiles_repo="$DOTFILES_REPO"
elif [ -n "$XDG_DATA_HOME" ]
then
  dotfiles_repo="$XDG_DATA_HOME/dotfiles"
else
  dotfiles_repo="$HOME/.local/share/dotfiles"
fi

# Get the name of the files that holds the list of all dotfiles that should be
# tracked
if [ -n "$DOTFILES_LIST" ]
then
  dotfiles_list="$DOTFILES_LIST"
elif [ -n "$XDG_CONFIG_HOME" ]
then
  dotfiles_list="$XDG_CONFIG_HOME/dotfiles.list"
else
  dotfiles_list="$HOME/.config/dotfiles.list"
fi

# Calls git in the appropriate repository
call_git() 
{
  git --git-dir="$dotfiles_repo" --work-tree="$HOME" "$@"
}

# Given a path, print the files and directories matching this path that are
# included in the list of dotfiles
get_pathspec()
{
  path="$1"

  if [ -f "$path" ]
  then
    tmp_path="$(realpath "$path")"

    # Check if the file is contained in any of the directories listed in the
    # dotfiles list are the parents of path. If this is the case, print path
    while [ "$tmp_path" != '/' ]
    do
      if grep -Fx "$tmp_path" "$dotfiles_list" > /dev/null
      then
        echo "$path"
        return
      fi

      tmp_path="$(dirname "$tmp_path")"
    done
  elif [ -d "$path" ]
  then
    # Print all of the files and directories from the list of dotfiles
    # contained in path

    # TODO: The regex should only math occurances of path starting at the
    # beggining of the line
    grep -F "$(realpath "$path")" "$dotfiles_list"
  fi

  # TODO: Print a warning if we get to here: No such file or directory
}

# Parse the commands
case "$1" in
  # Add files to the list of tracked files
  track)
    shift

    case "$1" in
      # Remove files from the list of dotfiles
      -d|--delete)
        shift

        # Remove files from the list of dotfiles
        for item in "$@"
        do
          # TODO: Properly escape pattern
          echo "Removing '$item' from the list of dotfile"
          pattern="$(realpath "$item")"
          pattern="${pattern//\//\\\/}"

          # Remove the files from the git repository and from the list of files
          # Awk is used because 'item' may be a directory
          awk -i inplace "!/^$pattern/ { print }" "$dotfiles_list"
          call_git rm "$item" --cached -r
        done
        ;;
      *)
        # Add files to the list of dotfiles
        for item in "$@"
        do
          echo "Adding '$item' to the list of dotfiles"
          realpath "$item" >> "$dotfiles_list"
        done
        ;;
    esac
    ;;

  # Add the files
  add)
    shift
    for item in "$@"
    do
      get_pathspec "$item" | call_git add --pathspec-from-file -
    done
    ;;

  # List the dotfiles specified in configurations list
  list)
    cat "$dotfiles_list"
    ;;

  # Show the configuration files that have been modified or deleted
  status)
    call_git status --porcelain \
      | awk '/^A / { print "✔️  ~/"$2 }; /^ A/ { print "   ~/"$2 }' \
      | print_files 'new' '30'

    call_git status --porcelain \
      | awk '/^M / { print "✔️  ~/"$2 }; /^ M/ { print "   ~/"$2 }' \
      | print_files 'modified' '33'

    call_git status --porcelain \
      | awk '/^D / { print "✔️  ~/"$2 }; /^ D/ { print "   ~/"$2 }' \
      | print_files 'deleted' '31'
    ;;

  # Just pass the following arguments to git
  --)
    shift
    call_git "$@"
    ;;

  # Just pass the arguments to git
  *)
    call_git "$@"
    ;;
esac