2021-12-11 19:40:43 +01:00
|
|
|
#!/bin/bash
|
|
|
|
set -eo pipefail
|
2022-01-11 10:51:57 +01:00
|
|
|
version=0.2.3
|
2022-01-20 21:40:12 +01:00
|
|
|
release_url="https://raw.githubusercontent.com/aclist/dztui/main/dztui.sh"
|
2021-12-11 19:40:43 +01:00
|
|
|
aid=221100
|
|
|
|
game="dayz"
|
2022-01-09 12:44:23 +01:00
|
|
|
workshop="https://steamcommunity.com/sharedfiles/filedetails/?id="
|
2021-12-11 19:40:43 +01:00
|
|
|
api="https://api.battlemetrics.com/servers"
|
|
|
|
|
|
|
|
#BEGIN CONFIG================
|
|
|
|
steam_path="/path/to/steam"
|
|
|
|
workshop_dir="$steam_path/steamapps/workshop/content/$aid"
|
|
|
|
game_dir="$steam_path/steamapps/common/DayZ"
|
|
|
|
key="APIKEY"
|
|
|
|
whitelist="8039514,8789747,4363928,8199330,11359652,12862329"
|
|
|
|
fav="8789747"
|
|
|
|
name="player"
|
|
|
|
separator="│"
|
|
|
|
ping=1
|
|
|
|
debug=0
|
|
|
|
#END CONFIG================
|
|
|
|
|
|
|
|
#STEAMCMD CONFIG===========
|
|
|
|
auto_install_mods=0
|
|
|
|
steamcmd_user="steam"
|
|
|
|
steam_username="STEAMUSER"
|
|
|
|
staging_dir="/tmp"
|
|
|
|
#END STEAMCMD CONFIG=======
|
|
|
|
|
|
|
|
declare -A deps
|
|
|
|
deps=([awk]="5.1.1" [curl]="7.80.0" [jq]="1.6" [column]="2.37.2" [tr]="9.0" [comm]="9.0")
|
|
|
|
max_range=$(awk -F, '{print NF}' <<< $whitelist)
|
|
|
|
|
2022-01-20 21:40:12 +01:00
|
|
|
err(){
|
|
|
|
printf "[ERROR] %s\n" "$1"
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
version_check(){
|
|
|
|
upstream=$(curl -Ls "$release_url" | awk -F= 'NR==3 {print $2}')
|
|
|
|
printf "[INFO] Checking for new version\n"
|
|
|
|
if [[ ! $upstream == "$version" ]]; then
|
|
|
|
printf "[INFO] A newer version of DZTUI is available at:\n$release_url.\n"
|
|
|
|
while true; do
|
|
|
|
read -p "Continue anyway? [Y/n] " response
|
|
|
|
if [[ $response == "Y" ]]; then
|
|
|
|
break
|
|
|
|
elif [[ $response == "n" ]]; then
|
|
|
|
exit_msg
|
|
|
|
else
|
|
|
|
:
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
fi
|
|
|
|
}
|
2021-12-11 19:40:43 +01:00
|
|
|
depcheck(){
|
|
|
|
for dep in "${!deps[@]}"; do
|
|
|
|
command -v $dep 2>&1>/dev/null || (printf "[ERROR] Requires %s >= %s\n" $dep ${deps[$dep]} ; exit 1)
|
|
|
|
done
|
|
|
|
}
|
2022-01-20 21:41:03 +01:00
|
|
|
column_check(){
|
|
|
|
echo foo | column -o$'\t' &>/dev/null || err "column version >= 2.37.2 required"
|
2021-12-11 19:40:43 +01:00
|
|
|
}
|
|
|
|
varcheck(){
|
|
|
|
[[ -z $key ]] && (err "Missing API key")
|
|
|
|
[[ -z $whitelist ]] && (err "Missing server IDs")
|
|
|
|
[[ ! -d $workshop_dir ]] && (err "Malformed workshop path")
|
|
|
|
[[ ! -d $game_dir ]] && (err "Malformed game path")
|
|
|
|
[[ $whitelist =~ [[:space:]] ]] && (err "Separate whitelist values with commas")
|
|
|
|
IFS=,
|
|
|
|
[[ ! "${whitelist[*]}" =~ "${fav}" ]] && (err "Fav key value not in whitelist")
|
|
|
|
unset IFS
|
|
|
|
}
|
|
|
|
checks() {
|
|
|
|
depcheck
|
2022-01-20 21:41:03 +01:00
|
|
|
column_check
|
|
|
|
version_check
|
2021-12-11 19:40:43 +01:00
|
|
|
varcheck
|
|
|
|
}
|
|
|
|
check_ping(){
|
|
|
|
if [[ $ping -eq 1 ]]; then
|
|
|
|
ping_ip=$(echo -e "$i" | awk -F'\t' '{print $2}' | awk -F: '{print $1}')
|
|
|
|
ms=$(ping -c 1 "$ping_ip" | awk -Ftime= '/time=/ {print $2}')
|
2022-01-11 10:48:48 +01:00
|
|
|
[[ -z $ms ]] && ms="Timeout" || :
|
2021-12-11 19:40:43 +01:00
|
|
|
printf "%s\t%s\n" "$i" "$ms"
|
|
|
|
else
|
|
|
|
printf "%s\n" "$i"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
parse_json(){
|
|
|
|
list=$(jq -r '.data[] .attributes | "\(.name)\t\(.ip):\(.port)\t\(.players)/\(.maxPlayers)\t\(.status)\t\(.id)"')
|
|
|
|
readarray -t list <<< $list
|
|
|
|
for i in "${list[@]}"; do
|
|
|
|
check_ping
|
|
|
|
done
|
|
|
|
}
|
|
|
|
symlinks(){
|
|
|
|
for d in "$workshop_dir"/*; do
|
2021-12-30 05:23:17 +01:00
|
|
|
id=$(awk -F"= " '/publishedid/ {print $2}' "$d"/meta.cpp | awk -F\; '{print $1}')
|
|
|
|
mod=$(awk -F\" '/name/ {print $2}' "$d"/meta.cpp | sed -E 's/[^[:alpha:]0-9]+/_/g; s/^_|_$//g')
|
|
|
|
link="@$id-$mod"
|
|
|
|
[[ -h "$game_dir/$link" ]] && : ||
|
2021-12-11 19:40:43 +01:00
|
|
|
printf "[INFO] Creating symlink for $mod\n"
|
2021-12-30 05:23:17 +01:00
|
|
|
ln -fs "$d" "$game_dir/$link"
|
2021-12-11 19:40:43 +01:00
|
|
|
done
|
|
|
|
}
|
|
|
|
installed_mods(){
|
|
|
|
ls -1 "$workshop_dir"
|
|
|
|
}
|
|
|
|
|
|
|
|
list_mods(){
|
|
|
|
printf "Installed mods: "
|
|
|
|
for d in $(installed_mods); do
|
|
|
|
awk -F\" '/name/ {print $2}' "$workshop_dir"/$d/meta.cpp
|
|
|
|
done | sort | awk 'NR > 1 { printf(", ") } {printf("%s",$0)}'
|
|
|
|
printf "\n"
|
|
|
|
}
|
|
|
|
|
|
|
|
columnize(){
|
|
|
|
column -t -s$'\t' -o$" $separator "
|
|
|
|
}
|
|
|
|
test_fav(){
|
|
|
|
if [[ -n $fav ]]; then
|
|
|
|
if [[ $(echo -e "${tabled[$i]}" | awk -F'\t' -v fav=$fav '$5 == fav') ]] ; then
|
|
|
|
printf "%s│▶%s\n" "$i" "${tabled[$i]}"
|
|
|
|
else
|
|
|
|
printf "%s│ %s\n" "$i" "${tabled[$i]}"
|
|
|
|
fi
|
|
|
|
else
|
|
|
|
printf "%s│ %s\n" "$i" "${tabled[$i]}"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
table(){
|
|
|
|
range=$((${#tabled[@]} - 1))
|
|
|
|
for ((i=0;i<="$range";i++)); do
|
|
|
|
test_fav
|
|
|
|
done
|
|
|
|
}
|
|
|
|
concat_mods(){
|
|
|
|
readarray -t serv <<< "$remote_mods"
|
|
|
|
for i in "${serv[@]}"; do
|
2021-12-30 05:23:17 +01:00
|
|
|
id=$(awk -F"= " '/publishedid/ {print $2}' "$workshop_dir"/$i/meta.cpp | awk -F\; '{print $1}')
|
|
|
|
mod=$(awk -F\" '/name/ {print $2}' "$workshop_dir"/$i/meta.cpp | sed -E 's/[^[:alpha:]0-9]+/_/g; s/^_|_$//g')
|
|
|
|
link="@$id-$mod;"
|
|
|
|
echo -e "$link"
|
|
|
|
done | tr -d '\n' | perl -ple 'chop'
|
2021-12-11 19:40:43 +01:00
|
|
|
}
|
|
|
|
launch(){
|
|
|
|
mods=$(concat_mods)
|
|
|
|
ip=$(echo -e "${tabled[$sel]}" | awk -F'\t' '{print $2}')
|
|
|
|
printf "[INFO] Connecting to: $connecting_to\n"
|
|
|
|
if [[ $debug -eq 1 ]]; then
|
2021-12-31 07:43:54 +01:00
|
|
|
printf "[DEBUG] steam -applaunch $aid -connect=$ip -nolauncher -nosplash -skipintro \"-mod=$mods\"\n"
|
2021-12-11 19:40:43 +01:00
|
|
|
else
|
2021-12-31 07:43:54 +01:00
|
|
|
steam -applaunch $aid -connect=$ip -nolauncher -nosplash -skipintro -name=$name \"-mod=$mods\"
|
2021-12-11 19:40:43 +01:00
|
|
|
printf "Good luck out there. DZTUI $version\n"
|
|
|
|
exit
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
launch_fav(){
|
|
|
|
if [[ -n $fav ]]; then
|
|
|
|
sorted_id=$fav
|
|
|
|
sel=$(table | awk -F'\t' -v fav=$fav '$5 == fav {print substr($1,1,1)}')
|
|
|
|
connect
|
|
|
|
else
|
|
|
|
printf "[INFO] No favorite set\n"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
manual_mod_install(){
|
|
|
|
printf "[ERROR] Missing mods. Open these links and subscribe to each one, then reconnect\n"
|
|
|
|
for i in $diff; do
|
|
|
|
printf "%s%s\n" "$workshop" $i
|
|
|
|
done
|
|
|
|
}
|
|
|
|
steamcmd_modlist(){
|
|
|
|
for i in $diff; do
|
|
|
|
printf "+workshop_download_item %s %s " $aid $i
|
|
|
|
done
|
|
|
|
}
|
|
|
|
move_files(){
|
|
|
|
sudo chown -R $USER:$gid "$staging_dir"/steamapps
|
2022-01-20 21:41:35 +01:00
|
|
|
cp -R "$staging_dir"/steamapps/workshop/content/$aid/* "$workshop_dir"
|
2021-12-11 19:40:43 +01:00
|
|
|
rm -r "$staging_dir"/steamapps
|
|
|
|
}
|
|
|
|
auto_mod_download(){
|
2022-01-20 21:42:56 +01:00
|
|
|
sudo su -c "steamcmd +force_install_dir $staging_dir +login $steam_username $(steamcmd_modlist) +quit" $steamcmd_user
|
2021-12-14 23:48:05 +01:00
|
|
|
[[ "$(ls -A $staging_dir/steamapps)" ]] && move_files || return 1
|
2021-12-11 19:40:43 +01:00
|
|
|
}
|
|
|
|
auto_mod_install(){
|
|
|
|
printf "[ERROR] Missing mods. Invoking steamcmd for user $steamcmd_user\n"
|
|
|
|
if [[ -z $steamcmd_user ]]; then
|
|
|
|
err "steamcmd user value was empty. Reverting to manual mode"
|
|
|
|
elif
|
|
|
|
id $steamcmd_user &>/dev/null
|
|
|
|
[[ $? -eq 1 ]]; then
|
|
|
|
err "Invalid steamcmd user. Reverting to manual mode"
|
2021-12-14 23:48:05 +01:00
|
|
|
elif
|
|
|
|
command -v steamcmd &>/dev/null
|
|
|
|
[[ $? -eq 1 ]]; then
|
2022-01-20 21:42:56 +01:00
|
|
|
err "steamcmd not installed. See: https://developer.valvesoftware.com/wiki/SteamCMD"
|
2021-12-11 19:40:43 +01:00
|
|
|
else
|
|
|
|
printf "[INFO] Found steamcmd user. Downloading mods\n"
|
|
|
|
revert_msg="Something went wrong. Reverting to manual mode"
|
|
|
|
auto_mod_download
|
|
|
|
[[ $? -eq 0 ]] && printf "\n"; init_table || err "$revert_msg"
|
|
|
|
fi
|
|
|
|
|
|
|
|
}
|
|
|
|
failed_mod_check(){
|
|
|
|
[[ $auto_install_mods -eq 1 ]] && auto_mod_install || manual_mod_install
|
|
|
|
}
|
|
|
|
passed_mod_check(){
|
|
|
|
printf "[INFO] Mod check passed\n"
|
|
|
|
connecting_to=$(echo -e "${tabled[$sel]}" | awk -F'\t' '{print $1,$2}')
|
|
|
|
symlinks
|
|
|
|
launch
|
|
|
|
|
|
|
|
}
|
2022-01-11 10:51:57 +01:00
|
|
|
check_workshop(){
|
|
|
|
curl -Ls "$url${modlist[$i]}" | grep data-appid | awk -F\" '{print $8}'
|
|
|
|
}
|
|
|
|
validate_mods(){
|
|
|
|
url="https://steamcommunity.com/sharedfiles/filedetails/?id="
|
|
|
|
aid=221100
|
|
|
|
tput civis
|
|
|
|
newlist=()
|
|
|
|
readarray -t modlist <<< $remote_mods
|
|
|
|
for ((i=0;i<=${#modlist[@]};i++)); do
|
|
|
|
printf "[INFO] Verifying integrity of server modlist manifest [$i/${#modlist[@]}]\r"
|
|
|
|
[[ $(check_workshop) -eq $aid ]] && newlist+=("${modlist[$i]}") || :
|
|
|
|
sleep 2s
|
|
|
|
done
|
|
|
|
tput cnorm
|
|
|
|
printf "\n"
|
|
|
|
}
|
2022-01-20 21:43:24 +01:00
|
|
|
server_modlist(){
|
|
|
|
for i in "${newlist[@]}"; do
|
|
|
|
printf "$i\n"
|
|
|
|
done
|
|
|
|
}
|
2021-12-11 19:40:43 +01:00
|
|
|
compare(){
|
|
|
|
fetch_mods
|
2022-01-11 10:51:57 +01:00
|
|
|
validate_mods
|
2022-01-20 21:43:24 +01:00
|
|
|
diff=$(comm -23 <(server_modlist | sort) <(installed_mods | sort))
|
2021-12-11 19:40:43 +01:00
|
|
|
}
|
|
|
|
connect(){
|
|
|
|
compare
|
|
|
|
if [[ -n $diff ]]; then
|
|
|
|
failed_mod_check
|
|
|
|
else
|
|
|
|
passed_mod_check
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
fetch_mods(){
|
|
|
|
remote_mods=$(curl -s "$api" -H "Authorization: Bearer "$key"" -G -d filter[ids][whitelist]="$sorted_id" -d "sort=-players" \
|
|
|
|
| jq -r '.data[] .attributes .details .modIds[]')
|
|
|
|
}
|
|
|
|
query_api(){
|
|
|
|
response=$(curl -s "$api" -H "Authorization: Bearer "$key"" -G -d "sort=-players" \
|
|
|
|
-d "filter[game]=$game" -d "filter[ids][whitelist]=$whitelist")
|
|
|
|
if [[ "$(jq -r 'keys[]' <<< $response)" == "errors" ]]; then
|
|
|
|
printf "[ERROR] %s: check API key\n" "$(jq -r '.errors[] .status' <<< $response)"
|
|
|
|
return 1
|
|
|
|
elif
|
|
|
|
[[ -z "$(jq -r '.data[]' <<< $response)" ]]; then
|
|
|
|
printf "[ERROR] Check server ID\n"
|
|
|
|
return 1
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
init_table(){
|
|
|
|
printf "\n[INFO] Polling servers. Please wait.\n"
|
|
|
|
query_api
|
|
|
|
readarray -t tabled <<< $(parse_json <<< $response)
|
|
|
|
tput cuu1
|
|
|
|
tput el
|
|
|
|
table | columnize
|
|
|
|
}
|
|
|
|
get_sorted_id(){
|
|
|
|
sorted_id=$(echo -e "${tabled[$sel]}" | awk -F'\t' '{print $5}')
|
|
|
|
}
|
2022-01-11 10:49:37 +01:00
|
|
|
menu(){
|
|
|
|
printf "\n"
|
|
|
|
printf "f$separator Launch favorite\n"
|
|
|
|
printf "l$separator List installed mods\n"
|
|
|
|
printf "r$separator Refresh\n"
|
|
|
|
printf "q$separator Quit\n"
|
|
|
|
printf "\n"
|
|
|
|
}
|
|
|
|
exit_msg(){
|
|
|
|
printf "DZTUI $version\n"
|
|
|
|
exit
|
|
|
|
}
|
2021-12-11 19:40:43 +01:00
|
|
|
main(){
|
|
|
|
checks
|
|
|
|
init_table
|
|
|
|
while true; do
|
2022-01-11 10:50:42 +01:00
|
|
|
menu
|
2021-12-11 19:40:43 +01:00
|
|
|
read -p "Selection: " sel
|
|
|
|
if [[ $sel =~ ^[0-9]+$ ]]; then
|
|
|
|
if [[ $sel -gt $max_range ]]; then
|
|
|
|
:
|
|
|
|
else
|
|
|
|
get_sorted_id
|
|
|
|
connect
|
|
|
|
fi
|
|
|
|
else
|
|
|
|
case $sel in
|
|
|
|
r) init_table ;;
|
|
|
|
f) launch_fav ;;
|
|
|
|
l) list_mods ;;
|
2022-01-11 10:50:42 +01:00
|
|
|
q) exit_msg ;;
|
2021-12-11 19:40:43 +01:00
|
|
|
*) : ;;
|
|
|
|
esac
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
}
|
|
|
|
main
|