#!/bin/sh
-if [ 0 -eq 1 ]; then
- alias mkdir="echo mkdir"
- alias ln="echo ln"
- alias rm="echo rm"
- alias cap_mkdb="echo cap_mkdb"
- alias unlink="echo unlink"
+# install location and source files
+: ${DEST:=$(realpath "$HOME")}
+: ${SRC:=$(dirname "$(realpath "$0")")}
+
+# confirm XDG vars we care about are set up
+: ${LOCAL:=.local} # NOTE: needed for bin, which has no xdg var
+: ${XDG_DATA_HOME:=$DEST/$LOCAL/share}
+: ${XDG_CONFIG_HOME:=$DEST/.config}
+: ${XDG_STATE_HOME:=$DEST/$LOCAL/state}
+: ${XDG_CACHE_HOME:=$DEST/.cache}
+
+# these are the vars we actually use
+share="${XDG_DATA_HOME#$DEST/}"
+etc="${XDG_CONFIG_HOME#$DEST/}"
+state="${XDG_STATE_HOME#$DEST/}"
+cache="${XDG_CACHE_HOME#$DEST/}"
+
+# test
+devnull=/dev/null
+if [ "$1" = "test" ]; then
+ echo "Running test -- no changes will be applied"
+ devnull=/dev/stdout
+ _cd() {
+ echo "-- cd $* --> $PWD";
+ cd $*;
+ }
+ alias cd="_cd"
+ alias mkdir="echo -- mkdir"
+ alias rmdir="echo -- rmdir"
+ alias ln="echo -- ln"
+ alias rm="echo -- rm"
+ alias mv="echo -- mv"
+ alias sh="echo -- sh"
fi
-#fix permissions from git (TODO: should probably move this to git hook)
-chmod go= base/.zshenv base/.Xresources
-
-#create preferred folder structure
-cd "$HOME"
-mkdir -p bin etc share .urxvt/ext > /dev/null 2>&1
-mkdir -m 700 tmp tmp/vim > /dev/null 2>&1
-cd -
-
-#accomodate linuxisms
-fex='-perm -ugo=x'
-# TODO: should probably do this better
-case "`uname`" in
- *BSD*) lnargs=-sFhvw
- d1='-depth 1'
- bsd=yes;;
- *) lnargs=-sfnv
- d1='-mindepth 1 -maxdepth 1'
- #android busybox find doesn't have -execute
- #and its sh doesn't set OSTYPE :DD
- if [ -z "$ANDROID_ROOT" ]; then
- fex=-executable
- fi;;
+# ensure ostype
+# NOTE: copied from .zshenv
+if [ -z "$OSTYPE" ]; then
+ OSTYPE=$(uname -s | tr '[:upper:]' '[:lower:]')
+fi
+
+# accomodate multiple userlands
+# XXX: assuming GNU by default
+# TODO: test for GNU, fallback to POSIX instead (no -n)
+lnargs=-sfnv
+case "$OSTYPE" in
+ freebsd*) lnargs=-sFhv;;
+ netbsd|openbsd*) lnargs=-sfhv;;
esac
+# save pwd
+old_pwd="$PWD"
+
+cd "$DEST"
+
+# create preferred folder structure
+mkdir -pm 760 "$share" "$etc" "$state" "$cache"
+ln $lnargs "$DEST/$etc" "$LOCAL/etc"
+mkdir -p "$LOCAL/bin" "$share/fonts" "$share/themes" "$share/icons"
+
+# clean out old links
+# TODO: doesn't remove dead links if realpath fails; this happens if the link
+# target doesn't exist under busybox
+find . "$share" "$etc" \
+ "$LOCAL/bin" "$share/fonts" "$share/themes" "$share/icons" \
+ "$etc/gnupg" .ssh .termux \
+ -xdev -mindepth 1 -maxdepth 1 -type l -exec sh -c '
+ r="$(realpath "{}")"
+ [ "${r#'"$SRC"'/}" != "$r" ] &&
+ '"$([ "$1" = test ] && echo 'echo +++ ')"'unlink "{}"' \;
+
+cd "$SRC" >$devnull
+
l() {
- # TODO: use install?
- [ -L "$1" -o \! -e "$1" ] && ln $lnargs "${2:-$x}" "$1"
+ # TODO: use install?
+ d="$DEST/${1:-$x}"
+ [ -L "$d" -o \! -e "$d" ] && ln $lnargs "$SRC/${2:-$x}" "$d" ||
+ echo "!!! File exists and is not a link: $1"
}
-find "$PWD/base" $d1 | while read -r x; do
- l "$HOME/`basename "$x"`"
-done
-find bin share $d1 | while read -r x; do
- l "$HOME/$x" "$PWD/$x"
-done
-find "$PWD/xdg" $d1 | while read -r x; do
- l "${XDG_CONFIG_HOME:-$HOME/etc}/`basename "$x"`"
-done
-find "$PWD/urxvt-ext" -type f $fex -mindepth 1 -maxdepth 2 | while read -r x; do
- l "$HOME/.urxvt/ext/`basename $x`"
+# generic links
+find base -mindepth 1 -maxdepth 1 | while read -r x; do l "${x#base/}"; done
+find bin -mindepth 1 -maxdepth 1 | while read -r x; do l "$LOCAL/$x"; done
+find share -mindepth 1 -maxdepth 1 | while read -r x; do l "$share${x#share}"; done
+find etc -mindepth 1 -maxdepth 1 | while read -r x; do l "$etc${x#etc}"; done
+
+# xorg/wayland resources
+find gui -mindepth 1 -maxdepth 1 | while read -r y; do
+ case "${y#gui/}" in
+ icons|themes)
+ find "$y" -mindepth 1 -maxdepth 2 -type d | while read -r x; do
+ [ ! -e "$x/index.theme" ] && continue
+ command -v gtk-update-icon-cache >$devnull 2>&1 &&
+ gtk-update-icon-cache -f "$x"
+ l "$share${y#gui}/$(basename "$x")"
+ done
+ ;;
+ fonts)
+ find "$y" -mindepth 1 -maxdepth 2 -type d | while read -r x; do
+ command -v mkfontscale >$devnull 2>&1 && mkfontscale "$x"
+ command -v mkfontdir >$devnull 2>&1 && mkfontdir "$x"
+ [ "$(head -1 "$x/fonts.scale" >$devnull 2>&1)" = 0 ] &&
+ rm "$x/fonts.scale"
+ [ "$(head -1 "$x/fonts.dir" >$devnull 2>&1)" = 0 ] &&
+ rm "$x/fonts.dir"
+ l "$share${y#gui}/$(basename "$x")"
+ done
+ ;;
+ *)
+ find "$y" -mindepth 1 -maxdepth 1 | while read -r x; do
+ l "$share${x#gui}"
+ done
+ ;;
+ esac
done
-cd ~
-if [ -n "$bsd" ]; then
- touch .hushlogin
+# gpg
+if command -v gpg >$devnull 2>&1; then
+ mkdir -pm 700 "$DEST/$etc/gnupg"
+ find gnupg -mindepth 1 -maxdepth 1 \! -name '*.gpg' | while read -r x; do
+ l "$etc/$x"
+ done
+ find gnupg -mindepth 1 -maxdepth 1 -name '*.gpg' | while read -r x; do
+ gpg --import "$x"
+ done
+fi
+
+# ssh
+if command -v ssh >$devnull 2>&1; then
+ mkdir -pm 700 "$DEST/.ssh"
+ find ssh -mindepth 1 -maxdepth 1 | while read -r x; do
+ l ".$x"; done
+fi
- #FreeBSD tries `_secure_path' on `.login_conf' before reading the database,
- #so it needs to be compiled and unlinked for it to actually take effect.
- rm .login_conf.db
- cap_mkdb .login_conf
+# termux, assume it if android
+if [ $(uname -o) = 'Android' ]; then
+ mkdir -p "$DEST/.termux"
+ find termux -mindepth 1 -maxdepth 1 | while read -r x; do l ".$x"; done
+ l '.termux/font.ttf' "gui/fonts/Dina-ttf/Dina.ttf"
fi
-unlink .login_conf
-cd -
+
+cd "$DEST" >$devnull
+
+# ensure xdg user dirs, and move old to new while we're at it
+source $etc/user-dirs.dirs
+function fixup_xdg_home {
+ if [ -z "$1" -o "${1#/}" = "$1" -o "$(realpath "$1")" = "$(realpath "$HOME")" ]; then
+ return;
+ fi
+ mkdir -p "$1"
+ if [ $? -eq 0 -a -n "$2" -a -d "$2" ]; then
+ mv "$2"/* $1
+ rmdir "$2"
+ fi
+}
+fixup_xdg_home $XDG_DESKTOP_DIR Desktop
+fixup_xdg_home $XDG_DOWNLOAD_DIR Downloads
+fixup_xdg_home $XDG_TEMPLATES_DIR
+fixup_xdg_home $XDG_PUBLICSHARE_DIR Public
+fixup_xdg_home $XDG_DOCUMENTS_DIR Documents
+fixup_xdg_home $XDG_MUSIC_DIR Music
+fixup_xdg_home $XDG_PICTURES_DIR Pictures
+fixup_xdg_home $XDG_VIDEOS_DIR Videos
+
+cd "$old_pwd"