dotfiles/script_helpers/file_ops.sh

383 lines
12 KiB
Bash
Raw Normal View History

2020-07-25 23:39:09 +00:00
#!/usr/bin/env bash
2019-07-03 21:08:59 +00:00
2019-12-31 05:38:46 +00:00
# Requires the printing.sh helper to be sourced.
# Requires the platform.sh helper to be sourced.
2019-07-03 21:08:59 +00:00
#---------------------------------------------------------------------------------------------------
# API
#---------------------------------------------------------------------------------------------------
2021-02-18 03:38:53 +00:00
strip_trailing_slashes() {
local ret="$1"
shopt -s extglob
ret=$(echo "${ret%%+(/)}")
echo $ret
}
2019-12-31 05:38:46 +00:00
2021-02-07 20:11:28 +00:00
# Will return a symlink path in its expanded form. If the path's root is the
2019-12-31 05:38:46 +00:00
# home directory symbol "~" then it'll be replaced by the full home path.
2019-07-03 21:08:59 +00:00
expand_path() {
2020-07-26 17:58:29 +00:00
local ret="$1"
IFS="/" read -ra parts <<< "$ret"
if [[ "${parts[0]}" == "~" ]]; then
ret="$HOME"
for ((i=1; i < ${#parts[@]}; i++))
do
ret="$ret/${parts[$i]}"
done
fi
ret=$(readlink -m "$ret")
echo $ret
2019-07-03 21:08:59 +00:00
}
2021-02-07 20:11:28 +00:00
# Returned value does not have a trailing '\'.
2019-12-31 05:38:46 +00:00
unix_to_windows_path() {
2021-02-18 03:38:53 +00:00
local ret="$1"
2021-02-07 20:11:28 +00:00
if [[ $(is_windows_path "$ret") -eq 0 ]]; then
if [[ $(is_absolute_unix_path "$ret") -eq 1 ]]; then
2019-12-31 05:38:46 +00:00
ret="${ret/\//}"
# Fix the drive name, e.g. c\foo becomes c:\foo
ret=$(sed 's,\([a-zA-Z]*\),\1:,' <<< "$ret")
fi
ret="${ret////\\}" # Replace Unix slashes.
ret="${ret//\\\(/\(}" # Remove backslash before (.
ret="${ret//\\\)/\)}" # Remove backslash before ).
2019-12-31 05:38:46 +00:00
fi
2021-02-18 03:38:53 +00:00
ret=$(strip_trailing_slashes "$ret")
2021-02-07 20:11:28 +00:00
echo $ret
2019-07-03 21:08:59 +00:00
}
2021-02-07 20:11:28 +00:00
# Returned value does not have a trailing '/'.
2019-07-03 21:08:59 +00:00
windows_to_unix_path() {
2021-02-18 03:38:53 +00:00
local ret="$1"
2020-07-26 17:58:29 +00:00
ret="/${ret/:/}" # Remove drive ':'.
ret="${ret//\\//}" # Replace Windows slashes.
ret="${ret// /\\ }" # Add a backslash before spaces.
ret="${ret//\(/\\(}" # Add a backslash before (.
ret="${ret//\)/\\)}" # Add a backslash before ).
2021-02-18 03:38:53 +00:00
ret="${ret/\/\//\/}" # If the passed in path was a unix path then we'll have two leading '/'; strip if it exists.
ret=$(strip_trailing_slashes "$ret")
2021-02-07 20:11:28 +00:00
echo "$ret"
}
# Returns a Unix path without escaped spaces, e.g. "/x/some folder" instead of "/x/some\ folder"
windows_to_unix_path_unescaped() {
2021-02-18 03:38:53 +00:00
local ret=$(windows_to_unix_path "$1")
2021-02-07 20:11:28 +00:00
ret="${ret/\\ / }" # Remove '\' that appears before spaces.
echo "$ret"
}
# Returns a Unix path with spaces escaped with a '\'.
escape_unix_path() {
2021-02-18 03:38:53 +00:00
local ret="$1"
2021-02-07 20:11:28 +00:00
ret="${ret/ /\\ }"
2020-07-26 17:58:29 +00:00
echo "$ret"
2019-07-03 21:08:59 +00:00
}
2021-02-18 03:38:53 +00:00
# Returns a Windows path with backslashes escaped so that you can print the path and see the slashes.
escape_backslashes() {
local ret="$1"
ret="${ret/\\/\\\\}"
echo "$ret"
}
# Returns the last part of a path without leading or trailing slashes.
2021-02-07 20:11:28 +00:00
strip_path() {
2021-02-18 03:38:53 +00:00
local result=$(basename "$1")
2021-02-07 20:11:28 +00:00
echo "$result"
}
2021-02-18 03:38:53 +00:00
# Returns a path without the last part. Does not end in a slash.
2021-02-07 20:11:28 +00:00
strip_filename() {
2021-02-18 03:38:53 +00:00
local result=$(dirname "$1")
2021-02-07 20:11:28 +00:00
echo "$result"
}
2021-02-18 03:38:53 +00:00
is_absolute_unix_path() {
if [[ $1 =~ ^/ ]]; then echo 1; else echo 0; fi
}
# Check if the first part of a path is a symlink.
is_first_dir_a_sym_link() {
local path="$1"
IFS="/" parts=( ${path//\/" "} )
local first_dir="${parts[0]}" # will be empty string if path started with slash.
if [[ $first_dir == "" || $first_dir == "." || ! -L $first_dir ]]; then
echo 0
else
echo 1
fi
}
# Check if any part of the path is a symlink. Stops at the first symlink that is found.
is_any_part_of_path_a_symlink() {
local path="$1"
if [[ $(is_absolute_unix_path "$path") -eq 1 ]]; then
echo 0
return
fi
IFS="/" parts=( ${path//\/" "} )
if [[ ${parts[0]} == "" ]]; then
echo 0
return
fi
local at=0
local len=${#parts[@]}
if [[ ${parts[0]} == "." ]]; then
if [[ $len -gt 1 ]]; then
at=1 # Skip the period.
else
echo 0
return
fi
fi
local test_path="${parts[$at]}"
if [[ -L "$test_path" ]]; then
echo 1
return
fi
((at=at+1))
until [ $at -eq $len ]
do
local part=${parts[$at]}
test_path="$test_path/$part"
if [[ -L $test_path ]]; then
echo 1
return
fi
((at=at+1))
done
echo 0
}
# Check if the first part of a path is a dotfile.
is_dotfile() {
if [[ $1 =~ ^\.{1} ]]; then echo 1; else echo 0; fi
}
# We're treating symlinks as Unix paths. This may give us trouble but we'll
# deal with it later should an edge case come up.
is_windows_path() {
if [[ $(is_any_part_of_path_a_symlink "$1") -eq 0 && $1 =~ \\+ ]]; then echo 1; else echo 0; fi
}
is_unix_path() {
echo $(! is_windows_path "$1")
}
path_has_a_space() {
local regexp="[[:blank:]]+"
if [[ $1 =~ $regexp ]]; then echo 1; else echo 0; fi
}
expand_path_if_not_symlink_or_absolute() {
local path="$1"
if [[ $(is_absolute_unix_path "$path") -eq 0 && $(is_any_part_of_path_a_symlink "$path") -eq 0 ]]; then
path=$(expand_path "$path")
fi
echo $path
}
2019-07-03 21:08:59 +00:00
move_file() {
2020-07-26 17:58:29 +00:00
local src="$1"
2021-02-07 20:11:28 +00:00
local src_expanded=$(expand_path "$src")
local dest_path=$(windows_to_unix_path_unescaped "$2")
local dest_filename="$3"
2020-07-26 17:58:29 +00:00
2021-02-07 20:11:28 +00:00
if [[ $dest_filename == "" ]]; then
dest_filename=$(strip_path "$src")
2020-07-26 17:58:29 +00:00
fi
2021-02-07 20:11:28 +00:00
if [[ -e "$src_expanded" ]]; then
mkdir -p "$dest_path"
local dest="$dest_path/$dest_filename"
mv "$src_expanded" "$dest"
2021-02-18 03:38:53 +00:00
printf "${BOLD}${GREEN}==> ${NORMAL}${BOLD}move: ${YELLOW}'$src'${NORMAL}${BOLD} to ${YELLOW}'$dest'${NORMAL}\n" 2>/dev/null
2020-07-26 17:58:29 +00:00
else
2021-02-18 03:38:53 +00:00
printf "${BOLD}${RED}==> move: ${YELLOW}'$src' ${RED}doesn't exists${NORMAL}\n"
return
2020-07-26 17:58:29 +00:00
fi
2019-07-03 21:08:59 +00:00
}
copy_file() {
2020-07-26 17:58:29 +00:00
local src="$1"
2021-02-07 20:11:28 +00:00
local src_expanded=$(expand_path "$src")
2020-07-26 17:58:29 +00:00
2021-02-07 20:11:28 +00:00
local dest_path=$(windows_to_unix_path_unescaped "$2")
local dest_filename="$3"
if [[ $dest_filename == "" ]]; then
dest_filename=$(strip_path "$src_expanded")
2020-07-26 17:58:29 +00:00
fi
2021-02-07 20:11:28 +00:00
if [[ -e "$src_expanded" ]]; then
mkdir -p "$dest_path"
local dest="$dest_path/$dest_filename"
cp "$src_expanded" "$dest"
2021-02-18 03:38:53 +00:00
printf "${BOLD}${GREEN}==> ${NORMAL}${BOLD}copy: ${YELLOW}'$src'${NORMAL}${BOLD} to\n ${YELLOW}'$dest'${NORMAL}\n" 2>/dev/null
2020-07-26 17:58:29 +00:00
else
2021-02-18 03:38:53 +00:00
printf "${BOLD}${RED}==> copy: ${YELLOW}'$src' ${RED}doesn't exists${NORMAL}\n"
return
2020-07-26 17:58:29 +00:00
fi
2019-07-03 21:08:59 +00:00
}
copy_dir_files() {
2021-02-18 03:38:53 +00:00
local src="$1"
2021-02-07 20:11:28 +00:00
local src_expanded=$(expand_path "$src")
local dest_path=$(windows_to_unix_path_unescaped "$2")
if [[ -d $src_expanded ]]; then
mkdir -p $dest_path
# Need to escape in order to use the wildcard and we have to eval in order to retain the backslash.
local src_escaped=$(escape_unix_path "$src_expanded")
cmd="cp -r $src_escaped/* \"$dest_path\""
eval $cmd
2020-07-26 17:58:29 +00:00
2021-02-18 03:38:53 +00:00
printf "${BOLD}${GREEN}==> ${NORMAL}${BOLD}copy *: ${BOLD}${YELLOW}'$src/*'${NORMAL}${BOLD} into ${YELLOW}'$dest_path'${NORMAL}\n" 2>/dev/null
2020-07-26 17:58:29 +00:00
else
2021-02-18 03:38:53 +00:00
printf "${BOLD}${RED}==> copy *: ${YELLOW}'$src' ${RED}doesn't exists${NORMAL}\n"
return
2020-07-26 17:58:29 +00:00
fi
2019-07-03 21:08:59 +00:00
}
2021-02-18 03:38:53 +00:00
# Only works with Unix paths.
make_link() {
local src=$1
local dest=$2
2019-12-31 05:38:46 +00:00
2021-02-18 03:38:53 +00:00
local debug=0
2019-12-31 05:38:46 +00:00
os_is_windows is_windows
os_is_unix is_unix
2021-02-18 03:38:53 +00:00
if [[ $is_windows -eq 1 ]]; then
if [[ $(is_windows_path "$src") -eq 1 ]]; then
local escaped_path=$(escape_backslashes "$src")
error "Expected a Unix source path, but got '$escaped_path instead.\n"
return
fi
if [[ $(is_windows_path "$dest") -eq 1 ]]; then
local escaped_path=$(escape_backslashes "$dest")
error "Expected a Unix dest path, but got '$escaped_path' instead.\n"
return
fi
fi
local expand_source_symlink=$3
local overwrite_existing=$4
local require_confirmation=$5
2019-12-31 05:38:46 +00:00
2021-02-18 03:38:53 +00:00
if [[ $overwrite_existing -ne 1 || $overwrite_existing -ne 0 ]]; then
overwrite_existing=$MC_OVERWRITE_EXISTING_SYMLINK
fi
2019-12-31 05:38:46 +00:00
if [[ $debug -eq 1 ]]; then
2021-02-18 03:38:53 +00:00
echo source path: $src
echo dest path: $dest
echo abs unix source: $(is_absolute_unix_path "$src")
echo abs unix dest: $(is_absolute_unix_path "$dest")
echo "overwrite existing? $overwrite_existing"
2019-12-31 05:38:46 +00:00
echo "require_confirmation? $require_confirmation"
2021-02-18 03:38:53 +00:00
echo "expand source symlink? $expand_source_symlink"
2019-12-31 05:38:46 +00:00
fi
2021-02-18 03:38:53 +00:00
local final_src=$src
local final_dest=$dest
2020-07-25 23:39:09 +00:00
if [[ $is_windows -eq 1 ]]; then
2021-02-18 03:38:53 +00:00
if [[ $expand_source_symlink -eq 1 ]]; then
final_src=$(expand_path "$final_src")
2020-07-25 23:39:09 +00:00
else
2021-02-18 03:38:53 +00:00
final_src=$(expand_path_if_not_symlink_or_absolute "$final_src")
# Having issues with mingw symlinking a path in the cwd to a dest that's not in the cwd.
# We prepend the cwd when it's not an absolute path in order to work around the issue.
if [[ $(is_absolute_unix_path "$final_dest") -ne 1 && $(is_dotfile "$final_dest") -ne 1 ]]; then
if [[ $(is_absolute_unix_path "$final_src") -eq 0 ]]; then
final_src="$PWD/$final_src"
fi
fi
2020-07-25 23:39:09 +00:00
fi
2021-02-18 03:38:53 +00:00
final_dest=$(expand_path_if_not_symlink_or_absolute "$final_dest")
2019-12-31 05:38:46 +00:00
fi
2021-02-18 03:38:53 +00:00
local source_has_space=$(path_has_a_space "$final_src")
local dest_has_space=$(path_has_a_space "$final_dest")
2019-12-31 05:38:46 +00:00
if [[ $debug -eq 1 ]]; then
2021-02-18 03:38:53 +00:00
echo "final source: $final_src"
echo "final dest: $final_dest"
echo source has space: $source_has_space
echo dest has space: $dest_has_space
2019-12-31 05:38:46 +00:00
fi
# Verify that the source path exists.
2021-02-18 03:38:53 +00:00
! test -e "$final_src" && printf "${BOLD}${RED}==> symlink: ${YELLOW}'$src' ${RED}doesn't exists${NORMAL}\n" && return
# Verify that the dest path doesn't already exist unless we're overwriting.
if [[ -e "$final_dest" ]]; then
if [[ $overwrite_existing -eq 1 ]]; then
echo "DELETING FINAL DEST: $final_dest | orig: $dest ||| final src: $final_src"
rm "$final_dest"
else
printf "==> symlink: ${BOLD}${YELLOW}'$dest'${NORMAL} already linked to ${BOLD}${YELLOW}'$src'${NORMAL}\n"
return
fi
fi
2019-12-31 05:38:46 +00:00
2021-02-18 03:38:53 +00:00
local cmd_source_path=""
local cmd_dest_path=""
local link_cmd=""
2019-12-31 05:38:46 +00:00
if [[ $is_windows -eq 1 ]]; then
2021-02-18 03:38:53 +00:00
cmd_source_path=$(unix_to_windows_path "$final_src")
cmd_dest_path=$(unix_to_windows_path "$final_dest")
2020-09-30 01:51:11 +00:00
if [[ $source_has_space -eq 1 ]]; then cmd_source_path="\"$cmd_source_path\""; fi
if [[ $dest_has_space -eq 1 ]]; then cmd_dest_path="\"$cmd_dest_path\""; fi
link_cmd="cmd //c 'mklink $cmd_dest_path $cmd_source_path'"
2019-12-31 05:38:46 +00:00
else
2020-09-30 01:51:11 +00:00
if [[ $source_has_space -eq 1 ]]; then cmd_source_path="\"$cmd_source_path\""; fi
if [[ $dest_has_space -eq 1 ]]; then cmd_dest_path="\"$cmd_dest_path\""; fi
link_cmd="ln -sf $cmd_source_path $cmd_dest_path"
2019-12-31 05:38:46 +00:00
fi
if [[ $require_confirmation -eq 1 ]]; then
2021-02-18 03:38:53 +00:00
echo "${BOLD}${BLUE}Will attempt to link ${YELLOW}'$src'${BLUE} to ${YELLOW}'$dest'${BLUE}"
2019-12-31 05:38:46 +00:00
printf "${BOLD}Enter 1 to proceed\n${YELLOW}> ${NORMAL}"
read confirm
if [[ $confirm != 1 ]]; then abort; fi
fi
if [[ $debug -eq 1 ]]; then
2020-09-30 01:51:11 +00:00
echo Final cmd source: $cmd_source_path
echo Final cmd dest: $cmd_dest_path
2019-12-31 05:38:46 +00:00
echo Link cmd:: $link_cmd
fi
2021-02-18 03:38:53 +00:00
printf "${BOLD}${GREEN}==> ${NORMAL}${BOLD}symlink: ${YELLOW}'$src'${NORMAL}${BOLD} to ${YELLOW}'$dest'${NORMAL}\n" 2>/dev/null
2020-08-09 19:25:35 +00:00
eval $link_cmd 1>/dev/null
2019-12-31 05:38:46 +00:00
}
2021-02-04 17:18:47 +00:00
create_dir() {
2021-02-18 03:38:53 +00:00
local path=$(strip_trailing_slashes "$1")
if [ ! -d $path ]; then
mkdir $path
2021-02-04 17:18:47 +00:00
fi
2021-02-18 03:38:53 +00:00
printf "${BOLD}${GREEN}==> ${NORMAL}${BOLD}mkdir: ${NORMAL}${BOLD}${YELLOW}'$path'${NORMAL}\n"
2021-02-04 17:18:47 +00:00
}