254 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			254 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable File
		
	
	
	
	
| #! /bin/bash
 | |
| #
 | |
| # bootleg-stow
 | |
| # Copyright (C) 2021 Vintage Salt <rehashedsalt@cock.li>
 | |
| #
 | |
| # Distributed under terms of the MIT license.
 | |
| #
 | |
| set -e
 | |
| 
 | |
| # Read-only set-once variables
 | |
| declare -r _name="$(basename -- "$0")"
 | |
| declare -i _opthelp
 | |
| declare -i _optverbose
 | |
| # Modes
 | |
| declare -i _optstow=1
 | |
| declare -i _optunstow
 | |
| # Working variables
 | |
| declare -a _args
 | |
| declare _return
 | |
| declare _files
 | |
| declare _directories
 | |
| 
 | |
| # Helper functions
 | |
| log() {
 | |
| 	# Print a line to the terminal if _optverbose is greater than $2
 | |
| 	# $2 defaults to 0
 | |
| 	# loglevel 0: Daily-use messages
 | |
| 	# loglevel 1: Detailed but not quite debugging
 | |
| 	# loglevel 2: Definitely debugging
 | |
| 	[ -z "$1" ] && return 1
 | |
| 	if (( _optverbose >= ${2:-0} )); then
 | |
| 		printf "%s\\n" "$1"
 | |
| 	fi
 | |
| }
 | |
| warn() {
 | |
| 	# Print a yellow line to the terminal, respecting _optverbose
 | |
| 	[ -z "$1" ] && return 1
 | |
| 	if (( _optverbose >= ${2:-0} )); then
 | |
| 		if [ -t 1 ]; then
 | |
| 			printf "\\e[33m%s\\e[0m\\n" "$1"
 | |
| 		else
 | |
| 			printf "WARN: %s\\n" "$1"
 | |
| 		fi
 | |
| 	fi
 | |
| }
 | |
| error() {
 | |
| 	# Print a red line to the terminal, exit if $2 is specified
 | |
| 	[ -z "$1" ] && return 1
 | |
| 	if [ -t 2 ]; then
 | |
| 		printf "\\e[31m%s\\e[0m\\n" "$1" 1>&2
 | |
| 	else
 | |
| 		printf "ERROR: %s\\n" "$1" 1>&2
 | |
| 	fi
 | |
| 	[ -z "$2" ] && return
 | |
| 	exit "${2:-1}"
 | |
| }
 | |
| has() {
 | |
| 	# Parse out all arguments and try to find them in path
 | |
| 	# If an argument cannot be found, set _return and fail
 | |
| 	for prog in "$@"; do
 | |
| 		if ! command -v "$prog" > /dev/null 2>&1; then
 | |
| 			_return="$prog"
 | |
| 			return 1
 | |
| 		fi
 | |
| 	done
 | |
| 	return 0
 | |
| }
 | |
| checkconflict() {
 | |
| 	# Take a directory as argument 1 and stow its contents in ..
 | |
| 	[ -z "$1" ] && return 1
 | |
| 	if ! [ -d "$1" ]; then
 | |
| 		error "Could not find directory: $1" 2
 | |
| 	fi
 | |
| 	if ! [ -r "$1" ]; then
 | |
| 		error "Could not read directory: $1" 2
 | |
| 	fi
 | |
| 	# Get our list of files
 | |
| 	local files="$(find "$1" ! -name "*.swp" -type f -o -type l)"
 | |
| 	local directories="$(find "$1" -type d)"
 | |
| 	local -a conflict
 | |
| 	# Iterate over them
 | |
| 	for file in $files; do
 | |
| 		# Get the basename of the file
 | |
| 		filename="${file#"$1"/}"
 | |
| 		if [ -f ../"$filename" ]; then
 | |
| 			if [ -h ../"$filename" ]; then
 | |
| 				continue
 | |
| 			else
 | |
| 				conflict+=("$filename")
 | |
| 			fi
 | |
| 		fi
 | |
| 	done
 | |
| 	if [ -n "${conflict[*]}" ]; then
 | |
| 		warn "The following non-symlinks would be touched by this operation:"
 | |
| 		for file in "${conflict[@]}"; do
 | |
| 			echo "$file"
 | |
| 		done
 | |
| 		error "Please resolve these conflicts manually." 3
 | |
| 	fi
 | |
| 	_files="$files"
 | |
| 	_directories="$directories"
 | |
| }
 | |
| stow() {
 | |
| 	# Stow all of _files and _directories in $1 as a package name
 | |
| 	# Note that you should checkconflict first
 | |
| 	[ -z "$1" ] && return 1
 | |
| 	stowdir="$(basename -- "$PWD")"
 | |
| 	log "Stowing package: $stowdir" 1
 | |
| 	pushd .. > /dev/null 2>&1
 | |
| 	for dir in $_directories; do
 | |
| 		dirname="${dir#"$1"/}"
 | |
| 		if [ "$dir" == "$1" ]; then
 | |
| 			log "Skipping package directory: $dir" 2
 | |
| 			continue
 | |
| 		fi
 | |
| 		if ! [ -d "$dirname" ]; then
 | |
| 			log "Creating directory: $dirname" 2
 | |
| 			mkdir -p "$dirname"
 | |
| 		fi
 | |
| 	done
 | |
| 	for file in $_files; do
 | |
| 		filename="${file#"$1"/}"
 | |
| 		if has realpath dirname; then
 | |
| 			path="$(realpath --relative-to="$(dirname "$filename")" "$PWD/$stowdir/$1/$filename")"
 | |
| 		else
 | |
| 			path="$PWD/$stowdir/$1/$filename"
 | |
| 		fi
 | |
| 		if [ -h "$filename" ]; then
 | |
| 			rm "$filename"
 | |
| 		fi
 | |
| 		if [ -h "$PWD/$stowdir/$1/$filename" ]; then
 | |
| 			log "Copying symlink: $filename" 2
 | |
| 			cp -d "$PWD/$stowdir/$1/$filename" "$filename"
 | |
| 		else
 | |
| 			log "Linking file: $filename to $path" 2
 | |
| 			ln -s "$path" "$filename"
 | |
| 		fi
 | |
| 	done
 | |
| 	log "Done stowing package: $1" 1
 | |
| 	popd > /dev/null 2>&1
 | |
| }
 | |
| unstow() {
 | |
| 	# Unstow all of _files and _directories
 | |
| 	# Takes a packagename as $1
 | |
| 	[ -z "$1" ] && return 1
 | |
| 	pushd .. > /dev/null 2>&1
 | |
| 	for file in $_files; do
 | |
| 		filename="${file#"$1"/}"
 | |
| 		if [ -h "$filename" ]; then
 | |
| 			rm "$filename"
 | |
| 		elif ! [ -e "$filename" ]; then
 | |
| 			warn "File does not exist, skipping: $filename" 1
 | |
| 		else
 | |
| 			warn "File is not a symlink, skipping: $filename"
 | |
| 		fi
 | |
| 	done
 | |
| 	_directories="$(echo "$_directories" | tac)"
 | |
| 	log "Removing empty directories" 2
 | |
| 	for dir in $_directories; do
 | |
| 		dirname="${dir#"$1"/}"
 | |
| 		# We silently ignore errors here so that rmdiring a directory with stuff in
 | |
| 		# it doesn't break our loop.
 | |
| 		rmdir -p "$dirname" > /dev/null 2>&1 || continue
 | |
| 	done
 | |
| 	popd > /dev/null 2>&1
 | |
| }
 | |
| 
 | |
| # Core program functions
 | |
| printhelp() {
 | |
| 	cat << EOF
 | |
| Usage: $_name [OPTION]... PACKAGE...
 | |
| Bootleg stow for the poor sods who can't get it.
 | |
| 
 | |
|   -h			Print this help text
 | |
|   -R			Unstow a directory.
 | |
|   -S			Stow a directory. Default operation.
 | |
|   -v			Print more status messages. Stacks
 | |
| 
 | |
| Copyright (c) 2021 rehashedsalt@cock.li
 | |
| Licensed under the MIT license
 | |
| EOF
 | |
| }
 | |
| 
 | |
| # Main
 | |
| main() {
 | |
| 	# Parse out arguments
 | |
| 	while [ -n "$1" ]; do
 | |
| 		# Parse out flags
 | |
| 		while getopts ":hRSv" opt; do
 | |
| 			case $opt in
 | |
| 				h)
 | |
| 					_opthelp=1
 | |
| 					;;
 | |
| 				R)
 | |
| 					_optunstow=1
 | |
| 					;;
 | |
| 				S)
 | |
| 					_optstow=1
 | |
| 					;;
 | |
| 				v)
 | |
| 					_optverbose+=1
 | |
| 					;;
 | |
| 				:)
 | |
| 					error "Option requires argument: -$OPTARG" 2
 | |
| 					;;
 | |
| 				*)
 | |
| 					error "Invalid option: -$OPTARG" 2
 | |
| 					;;
 | |
| 			esac
 | |
| 		done
 | |
| 		# Store arguments
 | |
| 		shift $((OPTIND - 1))
 | |
| 		if [ -n "$1" ]; then
 | |
| 			_args+=("$1")
 | |
| 			shift
 | |
| 		fi
 | |
| 		unset OPTIND
 | |
| 	done
 | |
| 	# Early hook for help
 | |
| 	[ -n "$_opthelp" ] && printhelp && exit 0
 | |
| 	# Validate critical options
 | |
| 	# TODO: That
 | |
| 	# Validate core program dependencies
 | |
| 	log "Validating dependencies" 2
 | |
| 	if ! has basename find tac; then
 | |
| 		error "Failed to find program: $_return" 1
 | |
| 	fi
 | |
| 
 | |
| 	# Do the do
 | |
| 	if [ -n "$_optunstow" ]; then
 | |
| 		if [ -n "${_args[*]}" ]; then
 | |
| 			for package in "${_args[@]}"; do
 | |
| 				checkconflict "$package"
 | |
| 				unstow "$package"
 | |
| 			done
 | |
| 		else
 | |
| 			error "No package specified" 1
 | |
| 		fi
 | |
| 	elif [ -n "$_optstow" ]; then
 | |
| 		if [ -n "${_args[*]}" ]; then
 | |
| 			for package in "${_args[@]}"; do
 | |
| 				checkconflict "$package"
 | |
| 				stow "$package"
 | |
| 			done
 | |
| 		else
 | |
| 			error "No package specified" 1
 | |
| 		fi
 | |
| 	fi
 | |
| 	exit 0
 | |
| }
 | |
| 
 | |
| main "$@"
 | |
| 
 |