2023-12-27 14:05:07 +03:00
|
|
|
# Rename dirs to `snake_case` and files to `PascalCase`. Careful with structured file names like archives!
|
|
|
|
# Usage: name [FILES]
|
|
|
|
function name() {
|
|
|
|
local IFS=$'\n'
|
|
|
|
local targets=(${@})
|
|
|
|
[[ "${targets}" = "" ]] && targets=($(ls))
|
|
|
|
|
|
|
|
process() {
|
|
|
|
# Skip archive.
|
|
|
|
if $(_is_archive ${target}); then
|
|
|
|
_warn "File is an archive, skip."
|
|
|
|
return 0
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [[ -d "${target}" ]]; then
|
|
|
|
local new_name=$(parse_snake ${target})
|
|
|
|
[[ -e "${new_name}" ]] && return 0
|
|
|
|
|
|
|
|
mv -- ${target} ${new_name} && echo ${new_name}
|
|
|
|
else
|
|
|
|
local ext=".${target##*.}"
|
|
|
|
local name=${target%.*}
|
|
|
|
[[ "${ext}" = ".${target}" ]] && ext=""
|
|
|
|
|
|
|
|
local new_name="$(parse_pascal ${name})${ext}"
|
|
|
|
[[ -e "${new_name}" ]] && return 0
|
|
|
|
|
|
|
|
mv -- ${target} ${new_name} && echo ${new_name}
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
_iterate_targets process ${targets[@]}
|
|
|
|
}
|
|
|
|
|
2023-12-07 04:02:47 +03:00
|
|
|
# Rename files to strip all special characters.
|
|
|
|
# All files by default.
|
2023-12-22 23:00:13 +03:00
|
|
|
# Usage: name_simple [FILES]
|
|
|
|
function name_simple() {
|
2023-12-05 21:50:45 +03:00
|
|
|
local IFS=$'\n'
|
2023-12-17 16:27:21 +03:00
|
|
|
local targets=(${@})
|
|
|
|
[[ "${targets}" = "" ]] && targets=([^.]*)
|
2023-12-05 21:50:45 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
process() {
|
2023-12-27 14:05:07 +03:00
|
|
|
# Skip archive.
|
|
|
|
if $(_is_archive ${target}); then
|
|
|
|
_warn "File is an archive, skip."
|
|
|
|
return 0
|
|
|
|
fi
|
|
|
|
|
2023-12-05 21:50:45 +03:00
|
|
|
# parse new name.
|
|
|
|
local ext=""
|
|
|
|
local name="${target}"
|
|
|
|
|
|
|
|
# ext only for files.
|
|
|
|
if [[ -f "${target}" ]]; then
|
|
|
|
ext=".${target##*.}"
|
|
|
|
name="${target%.*}"
|
|
|
|
fi
|
2023-12-10 07:20:48 +03:00
|
|
|
|
2023-12-05 21:50:45 +03:00
|
|
|
# Files w/o extension support.
|
|
|
|
[[ "${ext#.}" = "${name}" ]] && ext=""
|
|
|
|
|
|
|
|
# Get new name.
|
2023-12-30 20:16:32 +03:00
|
|
|
local new_name=$(echo "${name}" | parse_simplify)${ext,,}
|
2023-12-05 21:50:45 +03:00
|
|
|
|
|
|
|
# check if same name.
|
2023-12-17 16:27:21 +03:00
|
|
|
[[ "${target}" = "${new_name}" ]] && return 0
|
2023-12-05 21:50:45 +03:00
|
|
|
|
|
|
|
# check if target name already exists.
|
|
|
|
if [[ -f "${new_name}" ]]; then
|
2023-12-17 16:27:21 +03:00
|
|
|
_error "${new_name}: Already exists!"
|
|
|
|
return 1
|
2023-12-05 21:50:45 +03:00
|
|
|
fi
|
|
|
|
|
|
|
|
# rename target.
|
|
|
|
mv -- "${target}" "${new_name}"
|
2023-12-17 16:27:21 +03:00
|
|
|
}
|
2023-12-05 21:50:45 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
_iterate_targets process ${targets[@]}
|
2023-08-08 16:24:15 +03:00
|
|
|
}
|
|
|
|
|
2023-12-07 04:02:47 +03:00
|
|
|
# Rename all files to their hashes while keeping extensions.
|
|
|
|
# All files by default.
|
|
|
|
# Usage: name_hash [FILES]
|
2023-12-07 01:44:42 +03:00
|
|
|
function name_hash() {
|
2023-12-05 21:50:45 +03:00
|
|
|
local IFS=$'\n'
|
2023-12-17 16:27:21 +03:00
|
|
|
local targets=(${@})
|
|
|
|
[[ "${targets}" = "" ]] && targets=($(_ls_file))
|
2023-12-05 21:50:45 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
process() {
|
2023-12-05 21:50:45 +03:00
|
|
|
# extract extension.
|
|
|
|
local extension="${target##*.}"
|
|
|
|
if [[ "${extension}" = "${target}" ]]; then
|
|
|
|
extension=""
|
|
|
|
else
|
|
|
|
extension=".${extension}"
|
|
|
|
fi
|
|
|
|
|
|
|
|
# hash the new name.
|
2023-12-07 22:37:08 +03:00
|
|
|
local hash=$(pv "${target}" | sha1sum | cut -d\ -f1)
|
2023-12-05 21:50:45 +03:00
|
|
|
new_name="${hash,,}${extension,,}"
|
2023-12-10 07:20:48 +03:00
|
|
|
|
2023-12-05 21:50:45 +03:00
|
|
|
# check if same name.
|
2023-12-17 16:27:21 +03:00
|
|
|
[[ "${target}" = "${new_name}" ]] && return 0
|
2023-12-05 21:50:45 +03:00
|
|
|
|
|
|
|
# rename target.
|
2023-12-17 16:27:21 +03:00
|
|
|
mv -- ${target} ${new_name} && echo ${new_name}
|
|
|
|
}
|
2023-12-10 07:20:48 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
_iterate_targets process ${targets[@]}
|
2023-08-08 16:24:15 +03:00
|
|
|
}
|
|
|
|
|
2023-12-07 04:02:47 +03:00
|
|
|
# Check hashes for previously renamed files.
|
|
|
|
# All files by default.
|
|
|
|
# Usage: name_hash_check [FILES]
|
2023-12-07 01:44:42 +03:00
|
|
|
function name_hash_check() {
|
2023-12-05 21:50:45 +03:00
|
|
|
local IFS=$'\n'
|
2023-12-17 16:27:21 +03:00
|
|
|
local targets=(${@})
|
|
|
|
[[ "${targets}" = "" ]] && targets=([^.]*)
|
2023-12-05 21:50:45 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
process() {
|
2023-12-05 21:50:45 +03:00
|
|
|
# extract hashes.
|
|
|
|
local stored="${target%%.*}"
|
2023-12-07 22:37:08 +03:00
|
|
|
local actual=$(pv "${target}" | sha1sum | cut -d\ -f1)
|
2023-12-05 21:50:45 +03:00
|
|
|
|
|
|
|
# compare hashes.
|
|
|
|
if [[ "${stored}" != "${actual}" ]]; then
|
2023-12-17 20:12:33 +03:00
|
|
|
_error "Failed."
|
2023-12-17 16:27:21 +03:00
|
|
|
return 1
|
2023-12-05 21:50:45 +03:00
|
|
|
fi
|
2023-12-17 16:27:21 +03:00
|
|
|
}
|
2023-12-05 21:50:45 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
_iterate_targets process ${targets[@]}
|
2023-08-08 16:24:15 +03:00
|
|
|
}
|
|
|
|
|
2023-12-07 05:11:26 +03:00
|
|
|
# Rename files for Jellyfin series, i.e. `Episode S01E01.mkv`
|
2023-12-07 04:02:47 +03:00
|
|
|
# All files by default.
|
2023-12-30 22:17:44 +03:00
|
|
|
# Usage: name_series [FILES]
|
2023-12-07 01:44:42 +03:00
|
|
|
function name_series() {
|
2023-12-05 21:50:45 +03:00
|
|
|
local IFS=$'\n'
|
2023-12-30 22:17:44 +03:00
|
|
|
local season="$(realpath .)"; season="${season##*\ }"
|
2023-12-05 23:56:04 +03:00
|
|
|
local episode=0
|
2023-12-30 22:17:44 +03:00
|
|
|
local targets=(${@})
|
2023-12-17 16:27:21 +03:00
|
|
|
[[ "${targets}" = "" ]] && targets=($(_ls_file))
|
2023-12-05 21:50:45 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
# Error when no season number specified.
|
2023-12-05 21:50:45 +03:00
|
|
|
if [[ "${season}" = "" ]]; then
|
2023-12-30 22:17:44 +03:00
|
|
|
_error "Could not determine season number."
|
2023-12-05 21:50:45 +03:00
|
|
|
return 2
|
|
|
|
fi
|
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
process() {
|
2023-12-05 21:50:45 +03:00
|
|
|
((episode++))
|
|
|
|
|
|
|
|
# extract new name.
|
|
|
|
local new_name="Episode S${season}E$(printf %02d ${episode}).${target##*.}"
|
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
# Skip on no change.
|
|
|
|
[[ "${target}" = "${new_name}" ]] && return 0
|
2023-12-05 21:50:45 +03:00
|
|
|
|
|
|
|
# rename target.
|
2023-12-17 16:27:21 +03:00
|
|
|
mv -- ${target} ${new_name} && echo ${new_name}
|
|
|
|
}
|
2023-12-10 07:20:48 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
_iterate_targets process ${targets[@]}
|
2023-08-08 16:24:15 +03:00
|
|
|
}
|
|
|
|
|
2023-12-07 04:02:47 +03:00
|
|
|
# Rename files for Kavita manga format.
|
|
|
|
# All files by default.
|
|
|
|
# Usage: name_manga <SEASON> [FILES]
|
2023-12-07 01:44:42 +03:00
|
|
|
function name_manga() {
|
2023-12-05 21:50:45 +03:00
|
|
|
local IFS=$'\n'
|
2023-12-17 16:27:21 +03:00
|
|
|
local manga=${PWD##*/}
|
|
|
|
local season=${1}
|
2023-12-05 23:56:04 +03:00
|
|
|
local episode=0
|
2023-12-17 16:27:21 +03:00
|
|
|
local targets=(${@:2})
|
|
|
|
[[ "${targets}" = "" ]] && targets=($(_ls_file))
|
2023-12-05 21:50:45 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
# Error when no season number specified.
|
2023-12-05 21:50:45 +03:00
|
|
|
if [[ "${season}" = "" ]]; then
|
2023-12-30 22:17:44 +03:00
|
|
|
help name_manga
|
2023-12-05 21:50:45 +03:00
|
|
|
return 2
|
|
|
|
fi
|
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
process() {
|
2023-12-05 21:50:45 +03:00
|
|
|
((episode++))
|
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
# Extract new name.
|
2023-12-05 21:50:45 +03:00
|
|
|
local new_name="${manga} Vol.${season} Ch.${episode}.${target##*.}"
|
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
# Skip on no change.
|
|
|
|
[[ "${target}" = "${new_name}" ]] && return 0
|
2023-12-05 21:50:45 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
# Rename target.
|
|
|
|
mv -- ${target} ${new_name} && echo ${new_name}
|
|
|
|
}
|
2023-12-05 21:50:45 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
_iterate_targets process ${targets[@]}
|
2023-08-08 16:24:15 +03:00
|
|
|
}
|
|
|
|
|
2023-12-07 04:02:47 +03:00
|
|
|
# Rename files with new extension.
|
|
|
|
# All files by default.
|
|
|
|
# Usage: name_ext <EXTENSION> [FILES]
|
2023-12-07 01:44:42 +03:00
|
|
|
function name_ext() {
|
2023-12-05 21:50:45 +03:00
|
|
|
local IFS=$'\n'
|
2023-12-17 16:27:21 +03:00
|
|
|
local extension=${1}
|
|
|
|
local targets=(${@:2})
|
|
|
|
[[ "${targets}" = "" ]] && targets=($(_ls_file))
|
2023-12-05 21:50:45 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
# Error when no new extension specified.
|
2023-12-05 21:50:45 +03:00
|
|
|
if [[ "${extension}" = "" ]]; then
|
2023-12-07 04:31:18 +03:00
|
|
|
help name_ext
|
2023-12-05 21:50:45 +03:00
|
|
|
return 2
|
|
|
|
fi
|
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
process() {
|
|
|
|
# Extract new name.
|
2023-12-05 21:50:45 +03:00
|
|
|
local new_name="${target%.*}"."${extension}"
|
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
# Skip on no change.
|
|
|
|
[[ "${target}" = "${new_name}" ]] && return 0
|
2023-12-05 21:50:45 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
# Rename target.
|
|
|
|
mv -- ${target} ${new_name} && echo ${new_name}
|
|
|
|
}
|
2023-12-05 21:50:45 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
_iterate_targets process ${targets[@]}
|
2023-08-08 16:24:15 +03:00
|
|
|
}
|
|
|
|
|
2023-12-07 04:02:47 +03:00
|
|
|
# Change file name prefix.
|
|
|
|
# All matching files by default.
|
2023-11-26 00:51:57 +03:00
|
|
|
# Usage: name_prefix <OLD> <NEW> [FILES]
|
2023-12-07 01:44:42 +03:00
|
|
|
function name_prefix() {
|
2023-12-05 21:50:45 +03:00
|
|
|
local IFS=$'\n'
|
2023-12-17 16:27:21 +03:00
|
|
|
local old=${1}
|
|
|
|
local new=${2}
|
|
|
|
local targets=(${@:3})
|
|
|
|
[[ "${targets}" = "" ]] && targets=(${old}*)
|
2023-12-05 21:50:45 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
process() {
|
2023-12-05 21:50:45 +03:00
|
|
|
# Create new name.
|
|
|
|
local new_name="${new}${target#$old}"
|
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
# Skip on no change.
|
|
|
|
[[ "${target}" = "${new_name}" ]] && return 0
|
2023-12-10 07:20:48 +03:00
|
|
|
|
2023-12-05 21:50:45 +03:00
|
|
|
# Rename.
|
2023-12-17 16:27:21 +03:00
|
|
|
mv -- ${target} ${new_name} && echo ${new_name}
|
|
|
|
}
|
2023-12-10 07:20:48 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
_iterate_targets process ${targets[@]}
|
2023-11-08 16:47:12 +03:00
|
|
|
}
|
|
|
|
|
2023-12-07 04:02:47 +03:00
|
|
|
# Change file name postfix.
|
|
|
|
# All matching files by default.
|
2023-11-26 00:51:57 +03:00
|
|
|
# Usage: name_postfix <OLD> <NEW> [FILES]
|
2023-12-07 01:44:42 +03:00
|
|
|
function name_postfix() {
|
2023-12-05 21:50:45 +03:00
|
|
|
local IFS=$'\n'
|
2023-12-17 16:27:21 +03:00
|
|
|
local old=${1}
|
|
|
|
local new=${2}
|
|
|
|
local targets=(${@:3})
|
|
|
|
[[ "${targets}" = "" ]] && targets=(*${old})
|
2023-12-05 21:50:45 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
process() {
|
2023-12-05 21:50:45 +03:00
|
|
|
# Create new name.
|
|
|
|
local new_name="${target%$old}${new}"
|
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
# Skip on no change.
|
|
|
|
[[ "${target}" = "${new_name}" ]] && return 0
|
2023-12-10 07:20:48 +03:00
|
|
|
|
2023-12-05 21:50:45 +03:00
|
|
|
# Rename.
|
2023-12-17 16:27:21 +03:00
|
|
|
mv -- ${target} ${new_name} && echo ${new_name}
|
|
|
|
}
|
2023-12-05 21:50:45 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
_iterate_targets process ${targets[@]}
|
2023-11-26 00:51:57 +03:00
|
|
|
}
|
|
|
|
|
2023-11-30 06:07:01 +03:00
|
|
|
# Replace part of the name.
|
2023-12-07 04:02:47 +03:00
|
|
|
# All matching files by default.
|
|
|
|
# Usage: name_replace <OLD> <NEW> [FILES]
|
2023-12-07 01:44:42 +03:00
|
|
|
function name_replace() {
|
2023-12-05 21:50:45 +03:00
|
|
|
local IFS=$'\n'
|
2023-12-17 16:27:21 +03:00
|
|
|
local old=${1}
|
|
|
|
local new=${2}
|
|
|
|
local targets=(${@:3})
|
|
|
|
[[ "${targets}" = "" ]] && targets=(*${old}*)
|
2023-12-05 21:50:45 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
process() {
|
2023-12-05 21:50:45 +03:00
|
|
|
# Create new name.
|
|
|
|
local new_name="${target//$old/$new}"
|
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
# Skip on no change.
|
|
|
|
[[ "${target}" = "${new_name}" ]] && return 0
|
2023-12-10 07:20:48 +03:00
|
|
|
|
2023-12-05 21:50:45 +03:00
|
|
|
# Rename.
|
2023-12-17 16:27:21 +03:00
|
|
|
mv -- ${target} ${new_name} && echo ${new_name}
|
|
|
|
}
|
2023-12-10 07:20:48 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
_iterate_targets process ${targets[@]}
|
2023-12-10 07:20:48 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
# Fix numbering for numbered files. I.e if there are 10 items and some of them start without zero, then append zero to it. 1..10 -> 01..10.
|
|
|
|
# Usage: name_fix_numbering [FILES]
|
|
|
|
function name_fix_numbering() {
|
|
|
|
local IFS=$'\n'
|
2023-12-17 16:27:21 +03:00
|
|
|
local highest=0
|
2023-12-10 07:20:48 +03:00
|
|
|
local power=0
|
2023-12-17 16:27:21 +03:00
|
|
|
local targets=(${@})
|
|
|
|
[[ "${targets}" = "" ]] && targets=($(ls | grep "^[0-9]"))
|
2023-12-10 07:20:48 +03:00
|
|
|
|
|
|
|
# Count leading zeroes.
|
|
|
|
for target in "${targets[@]}"; do
|
2023-12-10 07:26:04 +03:00
|
|
|
# Check that starts with a digit.
|
|
|
|
[[ "${target}" =~ ^[0-9] ]] || continue
|
|
|
|
|
2024-01-03 18:50:59 +03:00
|
|
|
local digits=($(parse_ints "${target}"))
|
2023-12-10 07:20:48 +03:00
|
|
|
local digit="${digits[0]}"
|
|
|
|
digit=$((10#${digit}))
|
|
|
|
|
|
|
|
[[ "${digit}" -gt "${highest}" ]] && highest="${digit}"
|
|
|
|
done
|
|
|
|
|
|
|
|
local i=${highest}
|
|
|
|
while [[ i -gt 0 ]]; do
|
|
|
|
((power++))
|
|
|
|
i=$((${i}/10))
|
|
|
|
done
|
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
process() {
|
2023-12-10 07:20:48 +03:00
|
|
|
# Check that starts with a digit.
|
|
|
|
if [[ ! "${target}" =~ ^[0-9] ]]; then
|
2023-12-17 16:27:21 +03:00
|
|
|
_error "Does not start with a digit!"
|
|
|
|
return 1
|
2023-12-10 07:20:48 +03:00
|
|
|
fi
|
|
|
|
|
|
|
|
# Prepare new name.
|
2024-01-03 18:50:59 +03:00
|
|
|
local digits=($(parse_ints "${target}"))
|
2023-12-10 07:20:48 +03:00
|
|
|
local digit="${digits[0]}"
|
|
|
|
digit=$((10#${digit}))
|
|
|
|
local new_name=$(printf "%0${power}d" "${digit}")"${target#${digits[0]}}"
|
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
# Skip if the same name.
|
|
|
|
[[ "${target}" = "${new_name}" ]] && return 0
|
2023-12-10 07:20:48 +03:00
|
|
|
|
|
|
|
# Check that file does not exist.
|
|
|
|
if [[ -e "${new_name}" ]]; then
|
2023-12-17 16:27:21 +03:00
|
|
|
_error "${new_name}: File exists!"
|
2023-12-10 07:20:48 +03:00
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
mv -- ${target} ${new_name} && echo ${new_name}
|
|
|
|
}
|
2023-12-10 07:20:48 +03:00
|
|
|
|
2023-12-17 16:27:21 +03:00
|
|
|
_iterate_targets process ${targets[@]}
|
2023-11-30 06:07:01 +03:00
|
|
|
}
|
2023-12-17 23:46:41 +03:00
|
|
|
|