450 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Bash
		
	
	
	
	
	
			
		
		
	
	
			450 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Bash
		
	
	
	
	
	
| # 2024-02-10 Hyperling
 | |
| 
 | |
| if [[ -e ~/.env ]]; then
 | |
| 	source ~/.env
 | |
| fi
 | |
| 
 | |
| ## Aliases ##
 | |
| 
 | |
| # Quickly log onto production server.
 | |
| # Setting up ssh-keygen and ssh-copy-id make this even faster!
 | |
| if [[ -n $PROD_PORT && -n $PROD_USER && -n $PROD_NAME ]]; then
 | |
| 	alias prod="ssh -p $PROD_PORT $PROD_USER@$PROD_NAME"
 | |
| else
 | |
| 	alias prod="echo 'ERROR: .env not set up properly, please fix and reload RC.'"
 | |
| fi
 | |
| 
 | |
| # Quickies
 | |
| alias reload-bash="source ~/.bashrc"
 | |
| alias bash-reload="reload-bash"
 | |
| alias shell-reload="reload-bash"
 | |
| alias reload-shell="reload-bash"
 | |
| alias reload="reload-bash"
 | |
| 
 | |
| alias l='ls'
 | |
| alias ll='ls -alh'
 | |
| alias lh='ls -ash'
 | |
| 
 | |
| # Easily get to storage devices.
 | |
| cd storage/shared
 | |
| export SS="`pwd`/"
 | |
| cd
 | |
| alias ss="cd $SS"
 | |
| alias sd="ss"
 | |
| alias storage="ss"
 | |
| alias home="ss"
 | |
| 
 | |
| # Shortcut to media.
 | |
| export DCIM="$SS/DCIM/Camera"
 | |
| alias dcim="cd $DCIM"
 | |
| 
 | |
| # Shortcut to Code
 | |
| export CODE="$SS/Code"
 | |
| alias code="cd $CODE"
 | |
| 
 | |
| function code-projects {
 | |
|     cat <<- EOF
 | |
|         env-termux dev
 | |
|         env-ansible dev
 | |
|         env-docker dev
 | |
|         nodejs-website dev
 | |
|         hugo-jackanope main
 | |
| EOF
 | |
| }
 | |
| function code-reseed {
 | |
|     cd ~/Code
 | |
|     code-projects | while read project branch; do
 | |
|         git clone ssh://git@$GIT_SERVER:$GIT_PORT/$GIT_USER/$project --branch $branch --recurse-submodules
 | |
|     done
 | |
| }
 | |
| alias reseed-code="code-reseed"
 | |
| 
 | |
| function check-code {
 | |
|     cd ~/Code
 | |
|     ls | while read project; do
 | |
|         echo "Checking $project..."
 | |
|         cd $project
 | |
|         if [[ -d .git ]]; then
 | |
|             git pull
 | |
|             git push
 | |
|         else
 | |
|             echo "* Not a valid git project, skipping."
 | |
|         fi
 | |
|         cd ..
 | |
|     done
 | |
| }
 | |
| alias code-check="check-code"
 | |
| 
 | |
| # Shortcuts for TRASH.
 | |
| export TRASH="$SS/TRASH"
 | |
| alias trash="cd $TRASH"
 | |
| alias clean-trash="bash -c 'rm -rfv $TRASH/*'; clean-trashed"
 | |
| alias trash-clean="clean-trash"
 | |
| alias check-trash="du -h $TRASH"
 | |
| alias trash-check="check-trash"
 | |
| 
 | |
| alias clean-trashed='find "$SS" -name ".trashed*" -exec du -h {} \; -delete | sort -h'
 | |
| alias check-trashed='find "$SS" -name ".trashed*" -exec du -h {} \; -exec mv -v {} ~/TRASH/ \; | sort -h'
 | |
| 
 | |
| # Help prevent mistakes.
 | |
| alias cp="cp -v"
 | |
| alias mv="mv -v"
 | |
| alias rm="echo 'Move to ~/storage/shared/TRASH/ instead!'"
 | |
| 
 | |
| # Quickies
 | |
| alias bye="exit"
 | |
| alias install="pkg install"
 | |
| alias uninstall="pkg uninstall"
 | |
| alias remove="uninstall"
 | |
| alias scan="nmap -A -p- --script=vuln"
 | |
| 
 | |
| ## Functions ##
 | |
| 
 | |
| function update {
 | |
| 	params="$1"
 | |
| 
 | |
| 	echo "******* Apt / Pkg *******" &&
 | |
| 	echo "*** Update Cache ***" &&
 | |
| 	apt update &&
 | |
| 	echo -e "\n*** Auto Remove ***" &&
 | |
| 	apt autoremove $params &&
 | |
| 	echo -e "\n*** Clean ***" &&
 | |
| 	pkg clean $params &&
 | |
| 	echo -e "Cleaned!\n\n*** Fix Broken/Missing ***" &&
 | |
| 	apt --fix-broken --fix-missing install &&
 | |
| 	echo -e "\n*** Upgrade ***" &&
 | |
| 	# No longer using `apt`, in favor of Termux's wrapper `pkg`.
 | |
| 	#apt dist-upgrade --allow-downgrades --fix-broken --fix-missing $params &&
 | |
| 	pkg upgrade
 | |
| 	echo -e "\n******* Mission Complete!! *******"
 | |
| }
 | |
| alias upgrade="update"
 | |
| alias goodbye="update -y && bye"
 | |
| 
 | |
| # Optimize the bitrate and audio levels for an edited video.
 | |
| function process-video-usage {
 | |
| 	echo "USAGE: process-video oldFile newFile [videoBitrate] [audioBitrate] [sizeRating] [numPasses] [forceBitrate]"
 | |
| 	echo -n "Purpose: Call ffmpeg with preferred video posting settings. "
 | |
| 	echo -n "Bitrates default to 2000k and 192k, size is 720, passes is 1, and force is N."
 | |
| 	echo "These work well on Odysee and are fairly small as backups."
 | |
| 	echo "Examples:"
 | |
| 	echo "- Create a small file for quick streaming."
 | |
| 	echo "    process-video youcut.mp4 20240210.mp4 1200k 128k 480"
 | |
| 	echo "- Create a larger file for something like YouTube."
 | |
| 	echo "    process-video youcut.mp4 20240210_1080p.mp4 5000k 256k 1080"
 | |
| }
 | |
| function process-video {
 | |
| 	# Parameters
 | |
| 	file="$1"
 | |
| 	newfile="$2"
 | |
| 	video="$3"
 | |
| 	audio="$4"
 | |
| 	size="$5"
 | |
| 	passes="$6"
 | |
| 	force="$7"
 | |
| 
 | |
| 	# Validations
 | |
| 	if [[ -z $file || ! -e $file ]]; then
 | |
| 		echo "ERROR: Original file '$file' does not exist." >&2
 | |
| 		process-video-usage
 | |
| 		return 1
 | |
| 	fi
 | |
| 
 | |
| 	if [[ -z $newfile ]]; then
 | |
| 		echo "ERROR: New file's name must be provided." >&2
 | |
| 		process-video-usage
 | |
| 		return 1
 | |
| 	elif [[ -e $newfile ]]; then
 | |
| 		echo "ERROR: New file '$newfile' already exists." >&2
 | |
| 		du -h "$newfile"
 | |
| 		process-video-usage
 | |
| 		return 1
 | |
| 	fi
 | |
| 
 | |
| 	echo "`date` - Converting '$file' to '$newfile'."
 | |
| 
 | |
| 	if [[ -z $video ]]; then
 | |
| 		video="2000k"
 | |
| 	fi
 | |
| 	if [[ $force == "Y" ]]; then
 | |
| 		video="-b:v $video"
 | |
| 	else
 | |
| 		video="-b:v $video -minrate 0 -maxrate $video -bufsize $video"
 | |
| 	fi
 | |
| 
 | |
| 	if [[ -z $audio ]]; then
 | |
| 		audio="192k"
 | |
| 	fi
 | |
| 	audio="-b:a $audio"
 | |
| 
 | |
| 	if [[ -z $size ]]; then
 | |
| 		size="720"
 | |
| 	fi
 | |
| 	size="-filter:v scale=-1:$size"
 | |
| 
 | |
| 	if [[ -z $passes ]]; then
 | |
| 		passes=1
 | |
| 	fi
 | |
| 	pass=""
 | |
| 	if [[ $passes != 1 ]]; then
 | |
| 		passes=2
 | |
| 		pass="-pass 2"
 | |
| 	fi
 | |
| 
 | |
| 	## Main ##
 | |
| 	# More information on two-pass processing with ffmpeg
 | |
| 	# https://cinelerra-gg.org/download/CinelerraGG_Manual/Two_pass_Encoding_with_FFmp.html
 | |
| 	if [[ $passes == 2 ]]; then
 | |
| 		set -x
 | |
| 		ffmpeg -nostdin -hide_banner -loglevel quiet \
 | |
| 			-i "$file" $size $video $audio \
 | |
| 			-filter:a "dynaudnorm=f=33:g=65:p=0.66:m=33.3" \
 | |
| 			-vcodec libx264 -movflags +faststart \
 | |
| 			-pass 1 -f mp4 /dev/null -y
 | |
| 		status=$?
 | |
| 		set +x
 | |
| 		echo "`date` - Done with the first pass."
 | |
| 		if [[ $status != 0 ]]; then
 | |
| 			echo "Received unsuccessful status, exiting."
 | |
| 			return 1
 | |
| 		fi
 | |
| 	fi
 | |
| 
 | |
| 	set -x &&
 | |
| 	ffmpeg -nostdin -hide_banner -loglevel quiet \
 | |
| 		-i "$file" $size $video $audio \
 | |
| 		-filter:a "dynaudnorm=f=33:g=65:p=0.66:m=33.3" \
 | |
| 		-vcodec libx264 -movflags +faststart \
 | |
| 		$pass "$newfile"
 | |
| 	status="$?"
 | |
| 	set +x
 | |
| 	echo "`date` - Done with the final pass."
 | |
| 
 | |
| 	if [[ $passes == 2 && $status == 0 ]]; then
 | |
| 		mv -v ffmpeg2pass*.log* ~/TRASH/
 | |
| 	fi
 | |
| 
 | |
| 	sync
 | |
| 	sleep 10
 | |
| 	sync
 | |
| 	if [[ -s $newfile ]]; then
 | |
| 		echo "`date` - Getting file sizes."
 | |
| 		du -h "$file"
 | |
| 		du -h "$newfile"
 | |
| 	else
 | |
| 		echo "ERROR: New file not created or has a 0 size." >&2
 | |
| 	fi
 | |
| 
 | |
| 	echo -e "\n`date` - Finished with status '$status'."
 | |
| 	return $status
 | |
| }
 | |
| alias pv="process-video"
 | |
| 
 | |
| # Quickly commit code to a repo.
 | |
| function commit {
 | |
| 	message="$1"
 | |
| 	if [[ -z $message ]]; then
 | |
| 		echo "ERROR: A message is required." >&2
 | |
| 		echo 'USAGE: commit "My commit message."' >&2
 | |
| 		return 1
 | |
| 	fi
 | |
| 	git add . && git commit -m "$message" && git push
 | |
| }
 | |
| 
 | |
| # Easily test this project after committing changes.
 | |
| # Otherwise can just source this file unless testing setup.sh.
 | |
| function test-termux {
 | |
| 	sh -c "rm -rf ~/termux-deleteme ~/TRASH/termux-deleteme" 2>/dev/null
 | |
| 	git clone https://git.hyperling.com/me/env-termux \
 | |
| 		--branch=dev ~/termux-deleteme
 | |
| 	chmod 755 ~/termux-deleteme/*.sh
 | |
| 	~/termux-deleteme/setup.sh
 | |
| }
 | |
| alias reload-termux="test-termux"
 | |
| alias termux-test="test-termux"
 | |
| alias termux-reload="reload-termux"
 | |
| 
 | |
| # Allow converting video to audio ad other smaller
 | |
| # tasks than what process-video is intended to do.
 | |
| function basic-process-usage {
 | |
| 	echo "basic-process INPUT OUTPUT NORMALIZE [EXTRA]"
 | |
| 	echo -n "Pass a file through ffmpeg with the option"
 | |
| 	echo "to easily normalize the audio with a Y."
 | |
| 	echo "Examples:"
 | |
| 	echo "- Normalize audio on a video."
 | |
| 	echo "    basic-process video.mp4 normalized.mp4 Y"
 | |
| 	echo "- Convert a video to audio at 192k."
 | |
| 	echo "    basic-process video.mp4 audio.mp3 N '-b:a 192k'"
 | |
| }
 | |
| function basic-process {
 | |
| 	# Parameters
 | |
| 	input="$1"
 | |
| 	output="$2"
 | |
| 	typeset -u normalize
 | |
| 	normalize="$3"
 | |
| 	extra="$4"
 | |
| 
 | |
| 	echo "`date` - Starting basic-process"
 | |
| 
 | |
| 	# Validations
 | |
| 	if [[ -z $input || ! -e $input ]]; then
 | |
| 		echo "ERROR: Input file '$input' does not exist." >&2
 | |
| 		basic-process-usage
 | |
| 		return 1
 | |
| 	fi
 | |
| 
 | |
| 	if [[ -z $output ]]; then
 | |
| 		echo "ERROR: Output file's name must be provided." >&2
 | |
| 		basic-process-usage
 | |
| 		return 1
 | |
| 	elif [[ -e $output ]]; then
 | |
| 		echo "ERROR: Output file '$output' already exists." >&2
 | |
| 		du -h "$output"
 | |
| 		basic-process-usage
 | |
| 		return 1
 | |
| 	fi
 | |
| 
 | |
| 	if [[ $normalize == "Y" ]]; then
 | |
| 		echo "Normalize set to TRUE."
 | |
| 		normal="-filter:a dynaudnorm=f=33:g=65:p=0.66:m=33.3"
 | |
| 	else
 | |
| 		echo "No audio normalization is being done."
 | |
| 	fi
 | |
| 
 | |
| 	# Main
 | |
| 	echo "`date` - Converting '$input' to '$output'."
 | |
| 	set -x
 | |
| 	ffmpeg -nostdin -hide_banner -loglevel quiet \
 | |
| 		-i "$input" $extra $normal "$output"
 | |
| 	status=$?
 | |
| 	set +x
 | |
| 
 | |
| 	if [[ $status != 0 ]]; then
 | |
| 		echo "`date` - WARNING: ffmpeg exited with status '$status'." >&2
 | |
| 	fi
 | |
| 
 | |
| 	# Finish
 | |
| 	if [[ ! -s $output ]]; then
 | |
| 		echo "`date` - ERROR: Output '$output' not created or has 0 size." >&2
 | |
| 		return 1
 | |
| 	fi
 | |
| 
 | |
| 	sync
 | |
| 	echo "`date` - '$output' has been created successfully."
 | |
| 	sleep 3
 | |
| 	du -h "$input"
 | |
| 	du -h "$output"
 | |
| 
 | |
| 	echo "`date` - Finished basic-process"
 | |
| 	return 0
 | |
| }
 | |
| alias bp="basic-process"
 | |
| alias bv="bp"
 | |
| 
 | |
| # Function to automatically append tbe Y.
 | |
| function fix-audio {
 | |
| 	basic-process "$1" Y
 | |
| }
 | |
| 
 | |
| # Function to easily turn a video to audio without typing the audio name.
 | |
| function convert-to-audio {
 | |
| 	video="$1"
 | |
| 	audio="$2"
 | |
| 	if [[ -z "$2" || "$2" == "Y" ]]; then
 | |
| 		audio="${1//mp4/mp3}"
 | |
| 	fi
 | |
| 	volume=""
 | |
| 	if [[ ( "$2" == "Y" && -z "$3" ) || "$3" == "Y" ]]; then
 | |
| 		$volume="Y"
 | |
| 	fi
 | |
| 	basic-process "$video" "$audio" $volume
 | |
| }
 | |
| alias v2a="convert-to-audio"
 | |
| alias vta="v2a"
 | |
| alias va="v2a"
 | |
| alias pa="v2a"
 | |
| 
 | |
| # Go to normal storage. DISABLED, use shortcut aliases instead.
 | |
| #cd ~/storage/shared/
 | |
| 
 | |
| # Shortcuts for starting a video project.
 | |
| function init-video {
 | |
| 	dir="$1"
 | |
| 	if [[ ! -z $"dir" && ! -e "$dir" ]]; then
 | |
| 		mkdir -p "$dir"
 | |
| 		cd $dir
 | |
| 	fi
 | |
| 	touch readme.md
 | |
| 	mv -v "$SS/Movies/youcut/*" ./
 | |
| 	mv -v "$SS/Pictures/Screenshots/*" ./
 | |
| }
 | |
| alias pull-video="init-video"
 | |
| alias pull-screenshot="init-video"
 | |
| alias pull-sc="pull-screenshot"
 | |
| alias video-readme="init-video"
 | |
| 
 | |
| # Backup Related Activities #
 | |
| function now {
 | |
| 	date "+%Y%m%d-%H%M%S"
 | |
| }
 | |
| function today {
 | |
| 	date "+%Y%m%d"
 | |
| }
 | |
| export BACKUP_DIR="/srv/backup"
 | |
| export DOCKER="Docker"
 | |
| export SYSTEM="System"
 | |
| function pull_prod_backup {
 | |
| 	if [[ -z "$1" || ("$1" != "$DOCKER" && "$1" != "$SYSTEM") ]]; then
 | |
| 		echo "ERROR: Parameter 1 not correct, expecting '$DOCKER' or '$SYSTEM'."
 | |
| 		return 1
 | |
| 	fi
 | |
| 	backup_type="$1"
 | |
| 	echo "Looking for '*${backup_type}*.zip'"
 | |
| 	ls -lh *${backup_type}*.zip 2>/dev/null
 | |
| 	status="$?"
 | |
| 	if [[ "$status" == "0" ]]; then
 | |
| 		echo "File already downloaded, skipping."
 | |
| 		return 0
 | |
| 	else
 | |
| 		echo "File still needed, downloading to local tmp/."
 | |
| 	fi
 | |
| 	mkdir -pv tmp
 | |
| 	scp -P $PROD_PORT \
 | |
| 		$PROD_USER@$PROD_NAME:"$BACKUP_DIR/*${backup_type}*.zip" \
 | |
| 		tmp/
 | |
| 	if [[ "$?" == "0" ]]; then
 | |
| 		echo "Succeeded!"
 | |
| 		mv -v tmp/*.zip ./
 | |
| 		rmdir -v tmp
 | |
| 	else
 | |
| 		echo "Failed!"
 | |
| 		mv tmp "$TRASH/tmp_`now`"
 | |
| 	fi
 | |
| }
 | |
| function pull_prod_backups {
 | |
| 	dir="ProductionBackups-Pulled`today`"
 | |
| 	mkdir -pv "$dir"
 | |
| 	cd "$dir"
 | |
| 	pull_prod_backup "$DOCKER"
 | |
| 	pull_prod_backup "$SYSTEM"
 | |
| 	du -h *.zip
 | |
| 	status="$?"
 | |
| 	cd ..
 | |
| 	if [[ "$status" != 0 ]]; then
 | |
| 		echo "Failed to find zip files, removing folder."
 | |
| 		rmdir -v "$dir"
 | |
| 		return 1
 | |
| 	fi
 | |
| 	echo "Done!"
 | |
| }
 | |
| alias pull-prod-backups="pull_prod_backups"
 | |
| alias pull-backups="pull_prod_backups"
 | |
| alias prod-backups="pull_prod_backups"
 | |
| alias prod-backup="pull_prod_backups"
 | |
| alias pull-prod="pull_prod_backups"
 | |
| alias prod-pull="pull_prod_backups"
 | |
| 
 | |
| # Complete!
 | |
| PROG="$(basename -- "${BASH_SOURCE[0]}")"
 | |
| echo "'$PROG' completed!"
 |