325 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			325 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			Bash
		
	
	
		
			Executable File
		
	
	
	
	
| #!/bin/bash
 | |
| # 2023-06-13 Hyperling
 | |
| # Compress a video to good-enough quality for high quality streaming.
 | |
| 
 | |
| ## FFMpeg Notes ##
 | |
| # 2024-02-08
 | |
| #   -af parameter comes from Cahlen Lee's recommendation:
 | |
| #   https://odysee.com/@HyperVegan:2/20240205_DesertSilence-Campfire:3?lc=a2439b26d9fe89a950e0a47ec1d110d7156f596843039b4b76a4a2f327afcd2f
 | |
| 
 | |
| ## Setup ##
 | |
| 
 | |
| DIR="$(dirname -- "${BASH_SOURCE[0]}")"
 | |
| PROG="$(basename -- "${BASH_SOURCE[0]}")"
 | |
| echo "Running '$DIR/$PROG'."
 | |
| 
 | |
| filename_flag='compressed'
 | |
| date_YYYYMMDD="`date "+%Y%m%d"`"
 | |
| large_extension='DoNotUse-LargerThanOriginal'
 | |
| large_created=".$PROG.large_created.true"
 | |
| 
 | |
| ## Functions ##
 | |
| 
 | |
| function usage {
 | |
| 	echo -n "Usage: $PROG [-i file/folder] [-v bitrate] [-a bitrate] [-c vcodec]"
 | |
| 	echo " [-s size] [-r] [-f] [-d] [-A] [-m] [-V] [-x] [-h]"
 | |
| 	cat <<- EOF
 | |
| 		  Reduce the filesize of a video file to make it stream well. It also
 | |
| 		  helps with the file size for placing the file into a backup system.
 | |
| 		  Currently only set up for mp4 files.
 | |
| 
 | |
| 		Parameters:
 | |
| 		  -i input : The input file or folder with which to search for video files.
 | |
| 		             If nothing is provided, current directory (.) is assumed.
 | |
| 		  -v bitrate : The video bitrate to convert to.
 | |
| 		               Defaults are based on the size passed.
 | |
| 		                 >= 2160:   '30M'
 | |
| 		                 >= 1440:   '12M'
 | |
| 		                 >= 1080:    '5M'
 | |
| 		                 >=  720: '2000k'
 | |
| 		                 >=  480:  '750k'
 | |
| 		                 >=  360:  '250k'
 | |
| 		                  <  360:  '100k'
 | |
| 		  -a bitrate : The audio bitrate to convert to.
 | |
| 		               Defaults to '192k'.
 | |
| 		  -c vcodec : The video codec you'd like to use, such as libopenh264.
 | |
| 		  -s size : The video size such as 1080 or 720, or1280 for vertical 720p.
 | |
| 		            Defaults to '720'.
 | |
| 		  -r : Recurse the entire directory structure, compressing all video files.
 | |
| 		  -f : Force recompressing any files by deleting it if it already exists.
 | |
| 		  -d : Delete the original video if the compressed version is smaller.
 | |
| 		  -l : Clean the filename of dashes and underscores so dates line up.
 | |
| 		  -A : Recursively Force, Delete, and Clean.
 | |
| 		  -m : Measure the time it takes to compress each video and do the loop.
 | |
| 		  -V : Add verbosity, such as printing all the variable values.
 | |
| 		  -x : Set the shell's x flag to display every action which is taken.
 | |
| 		  -h : Display this help messaging.
 | |
| 	EOF
 | |
| 	exit $1
 | |
| }
 | |
| 
 | |
| ## Parameters ##
 | |
| 
 | |
| while getopts ":i:v:a:c:s:rfdlAmVxh" opt; do
 | |
| 	case $opt in
 | |
| 		i) input="$OPTARG"
 | |
| 			;;
 | |
| 		v) video_bitrate="$OPTARG"
 | |
| 			;;
 | |
| 		a) audio_bitrate="$OPTARG"
 | |
| 			;;
 | |
| 		c) codec="$OPTARG"
 | |
| 			;;
 | |
| 		s) size="$OPTARG"
 | |
| 			;;
 | |
| 		r) search_command="find"
 | |
| 			;;
 | |
| 		f) force="Y"
 | |
| 			;;
 | |
| 		d) delete="Y"
 | |
| 			;;
 | |
| 		l) clean="Y"
 | |
| 			;;
 | |
| 		A) search_command="find" && force="Y" && delete="Y" #&& clean="Y"
 | |
| 			;;
 | |
| 		m) time_command="`which time`"
 | |
| 			;;
 | |
| 		V) verbose="Y"
 | |
| 			;;
 | |
| 		x) set_x="Y"
 | |
| 			;;
 | |
| 		h) usage 0
 | |
| 			;;
 | |
| 		*) echo "ERROR: Option '$OPTARG' not recognized." >&2
 | |
| 			usage 1
 | |
| 			;;
 | |
| 	esac
 | |
| done
 | |
| 
 | |
| if [[ "$set_x" == "Y" ]]; then
 | |
| 	set -x
 | |
| fi
 | |
| 
 | |
| if [[ -z "$input" ]]; then
 | |
| 	echo "WARNING: Program was not passed an input. Using current directory." >&2
 | |
| 	input="."
 | |
| fi
 | |
| 
 | |
| if [[ -z $size ]]; then
 | |
| 	size="720"
 | |
| fi
 | |
| typeset -i size_int
 | |
| size_int="$size"
 | |
| 
 | |
| # Ensure the filenames sort properly between 3 and 4 digit resolutions.
 | |
| while (( ${#size} < 4 )); do
 | |
|     size="0$size"
 | |
| done
 | |
| compress_tags="$size"
 | |
| size="-filter:v scale=-2:$size"
 | |
| 
 | |
| if [[ -z "$video_bitrate" ]]; then
 | |
| 	# Based roughly on the 2.5 step iteration used for 720 and 1080.
 | |
| 	if (( size_int >= 2160 )); then
 | |
| 		video_bitrate="30M"
 | |
| 	elif (( size_int >= 1440 )); then
 | |
| 		video_bitrate="12M"
 | |
| 	elif (( size_int >= 1080 )); then
 | |
| 		video_bitrate="5M"
 | |
| 	elif (( size_int >= 720 )); then
 | |
| 		video_bitrate="2000k"
 | |
| 	elif (( size_int >= 480 )); then
 | |
| 		video_bitrate="750k"
 | |
| 	elif (( size_int >= 360 )); then
 | |
| 		video_bitrate="250k"
 | |
| 	else
 | |
| 		video_bitrate="100k"
 | |
| 	fi
 | |
| fi
 | |
| compress_tags="$compress_tags-$video_bitrate"
 | |
| video_bitrate="-b:v $video_bitrate -minrate 0 -maxrate $video_bitrate -bufsize $video_bitrate"
 | |
| 
 | |
| if [[ -z "$audio_bitrate" ]]; then
 | |
| 	audio_bitrate="192k"
 | |
| fi
 | |
| compress_tags="$compress_tags-$audio_bitrate"
 | |
| audio_bitrate="-b:a $audio_bitrate"
 | |
| 
 | |
| if [[ -z "$codec" ]]; then
 | |
| 	codec=""
 | |
| else
 | |
| 	codec="-vcodec $codec"
 | |
| fi
 | |
| 
 | |
| if [[ -z "$search_command" ]]; then
 | |
| 	search_command="ls"
 | |
| fi
 | |
| 
 | |
| if [[ -z "$time_command" ]]; then
 | |
| 	time_command=""
 | |
| else
 | |
| 	time_command="$time_command -p"
 | |
| fi
 | |
| 
 | |
| ## Main ##
 | |
| 
 | |
| if [[ "$verbose" == "Y" ]]; then
 | |
| 	cat <<- EOF
 | |
| 		VERBOSE: Full list of variables.
 | |
| 		  input='$input'
 | |
| 		  video_bitrate='$video_bitrate'
 | |
| 		  audio_bitrate='$audio_bitrate'
 | |
| 		  codec='$codec'
 | |
| 		  search_command='$search_command'
 | |
| 		  force='$force'
 | |
| 		  delete='$delete'
 | |
| 		  time_command='$time_command'
 | |
| 		  verbose='$verbose'
 | |
| 		  set_x='$set_x'
 | |
| 		  filename_flag='$filename_flag'
 | |
| 		  date_YYYYMMDD='$date_YYYYMMDD'
 | |
| 		  SECONDS='$SECONDS'
 | |
| 	EOF
 | |
| fi
 | |
| 
 | |
| SECONDS=0
 | |
| $search_command "$input" | sort | while read file; do
 | |
| 	echo -e "\n$file"
 | |
| 
 | |
| 	if [[ -n "$time_command" ]]; then
 | |
| 		date
 | |
| 	fi
 | |
| 
 | |
| 	typeset -l extension_lower
 | |
| 	extension="${file##*.}"
 | |
| 	extension_lower="$extension"
 | |
| 
 | |
| 	# Exception checks for the existing file.
 | |
| 	if [[ "$extension_lower" != "mp4" ]]; then
 | |
| 		echo "SKIP: Not an MP4."
 | |
| 		continue
 | |
| 	fi
 | |
| 	if [[ "$file" == *"$filename_flag"* ]]; then
 | |
| 		echo "SKIP: Input is already compressed."
 | |
| 		continue
 | |
| 	fi
 | |
| 
 | |
| 	# Build the new filename to signify it is different than the original.
 | |
| 	newfile="${file//$extension/$filename_flag-$date_YYYYMMDD.$extension_lower}"
 | |
| 
 | |
| 	# Add the size, video, and audio quality it was converted to.
 | |
| 	newfile="${newfile//\.$extension_lower/_$compress_tags.$extension_lower}"
 | |
| 
 | |
| 	if [[ $newfile == $file ]]; then
 | |
| 		echo "ERROR: The new calculated filename matches the old, skipping." >&2
 | |
| 		continue
 | |
| 	fi
 | |
| 
 | |
| 	# TBD / TODO: Test this functionality!
 | |
| 	new_video_clean="${newfile}"
 | |
| 	new_video_clean="${new_video_clean//_/}"
 | |
| 	new_video_clean="${new_video_clean//-/}"
 | |
| 	new_video_clean="${new_video_clean// /}"
 | |
| 
 | |
| 	# More exception checks based on the new file.
 | |
| 	if [[ -e "$newfile" || -e "$new_video_clean" ]]; then
 | |
| 		if [[ "$force" == "Y" ]]; then
 | |
| 			echo "FORCE: Removing '$newfile'."
 | |
| 			if [[ -d ~/TRASH ]]; then
 | |
| 				mv -v "$newfile" "$new_video_clean" ~/TRASH/ 2>/dev/null
 | |
| 			else
 | |
| 				rm -v "$newfile" "$new_video_clean" 2>/dev/null
 | |
| 			fi
 | |
| 		else
 | |
| 			echo "SKIP: Already has a compressed version."
 | |
| 			ls -sh "$newfile" "$new_video_clean" 2>/dev/null
 | |
| 			continue
 | |
| 		fi
 | |
| 	fi
 | |
| 
 | |
| 	# Whether or not to use the cleaned version or the normal version.
 | |
| 	if [[ -n $clean ]]; then
 | |
| 		newfile="$new_video_clean"
 | |
| 	fi
 | |
| 
 | |
| 	# Convert the file.
 | |
| 	echo "Converting to '$newfile'."
 | |
| 	echo "*** `date` ***"
 | |
| 	set -x
 | |
| 	$time_command bash -c "ffmpeg -nostdin -hide_banner -loglevel quiet \
 | |
| 			-i '$file' $size $video_bitrate $audio_bitrate \
 | |
| 			-af 'dynaudnorm=f=33:g=65:p=0.66:m=33.3' \
 | |
| 			$vcodec -movflags +faststart \
 | |
| 			'$newfile'"
 | |
| 	status="$?"
 | |
| 	set +x
 | |
| 	echo "*** `date` ***"
 | |
| 	if [[ "$status" != 0 ]]; then
 | |
| 		echo "SKIP: ffmpeg returned a status of '$status'."
 | |
| 		continue
 | |
| 	fi
 | |
| 
 | |
| 	# Check the filesize compared to the original and note if it is larger.
 | |
| 	echo "Checking file sizes:"
 | |
| 	ls -sh "$file" "$newfile" | sort -hr
 | |
| 	smaller_file=`ls -sh "$file" "$newfile" | sort -h | cut -f 2- -d ' ' | head -n 1`
 | |
| 	if [[ "$smaller_file" == "$file" ]]; then
 | |
| 		echo -n "Conversion had the opposite effect, original was likely lesser "
 | |
| 		echo "quality. Adding a suffix to the file to signify that it grew."
 | |
| 		mv -v "$newfile" "$newfile.$large_extension"
 | |
| 		continue
 | |
| 	fi
 | |
| 
 | |
| 	if [[ -e "$newfile" ]]; then
 | |
| 		echo "Conversion succeeded, file has been compressed."
 | |
| 	else
 | |
| 		echo "ERROR: Converted file '$newfile' could not be found. Aborting." >&2
 | |
| 		break
 | |
| 	fi
 | |
| 
 | |
| 	if [[ -n "$delete" ]]; then
 | |
| 		echo -n "Original has been deleted: "
 | |
| 		if [[ -d ~/TRASH ]]; then
 | |
| 			mv -v "$file" ~/TRASH/
 | |
| 		else
 | |
| 			rm -v "$file"
 | |
| 		fi
 | |
| 	fi
 | |
| done
 | |
| 
 | |
| # If large files do end up being created, allow the user to bulk delete them.
 | |
| if [[ -e "$large_created" ]]; then
 | |
| 	echo -e "\n*********************************************************"
 | |
| 	echo -e "WARNING: The files below are larger than their originals!\n"
 | |
| 	find "$input" -name "*"$large_extension
 | |
| 	echo -e "*********************************************************"
 | |
| 
 | |
| 	echo -en "\nWould you like to delete them? (Y/n): "
 | |
| 	typeset -u confirm_delete
 | |
| 	read confirm_delete
 | |
| 
 | |
| 	if [[ -z "$confirm_delete" || "$confirm_delete" == "Y"* ]]; then
 | |
| 		echo ""
 | |
| 		find "$input" -name "*"$large_extension -exec rm -v {} \;
 | |
| 	else
 | |
| 		echo -e "\nKeeping files. Please use this if you change your mind:"
 | |
| 		echo "  find \"$input\" -name \"*\"$large_extension -exec rm -v {} \;"
 | |
| 	fi
 | |
| 
 | |
| 	rm "$large_created"
 | |
| fi
 | |
| 
 | |
| echo -e "\nDone!"
 | |
| 
 | |
| # Display elapsed time
 | |
| if [[ -n "$time_command" ]]; then
 | |
| 	date
 | |
| 	typeset -i hours minutes seconds
 | |
| 	hours=$(( SECONDS / 3600 ))
 | |
| 	minutes=$(( (SECONDS % 3600) / 60 ))
 | |
| 	seconds=$(( SECONDS % 60 ))
 | |
| 	echo "Loop Performance: ${hours}h ${minutes}m ${seconds}s"
 | |
| fi
 | |
| 
 | |
| exit 0
 |