1
0
Fork 0
mirror of https://github.com/aclist/dztui.git synced 2025-04-04 11:33:01 +02:00

feat: save loader WIP

This commit is contained in:
aclist 2025-03-21 02:08:49 +09:00
parent bba793cc52
commit 62ce5d08ec
2 changed files with 429 additions and 37 deletions

228
dzgui.sh
View file

@ -36,6 +36,7 @@ history_file="$state_path/$prefix.history"
versions_file="$state_path/$prefix.versions" versions_file="$state_path/$prefix.versions"
lock_file="$state_path/$prefix.lock" lock_file="$state_path/$prefix.lock"
cols_file="$state_path/$prefix.cols.json" cols_file="$state_path/$prefix.cols.json"
scroll_fifo="$state_path/$prefix.fifo"
#CACHE FILES #CACHE FILES
coords_file="$cache_path/$prefix.coords" coords_file="$cache_path/$prefix.coords"
@ -53,6 +54,7 @@ geo_file="$helpers_path/ips.csv"
km_helper="$helpers_path/latlon" km_helper="$helpers_path/latlon"
sums_path="$helpers_path/sums.md5" sums_path="$helpers_path/sums.md5"
func_helper="$helpers_path/funcs" func_helper="$helpers_path/funcs"
scroll_helper="$helpers_path/scroll.py"
#REMOTE #REMOTE
remote_host=gh remote_host=gh
@ -65,7 +67,6 @@ releases_url="https://github.com/$author/$repo/releases/download/browser"
km_helper_url="$releases_url/latlon" km_helper_url="$releases_url/latlon"
geo_file_url="$releases_url/ips.csv.gz" geo_file_url="$releases_url/ips.csv.gz"
set_im_module(){ set_im_module(){
#TODO: drop pending SteamOS changes #TODO: drop pending SteamOS changes
pgrep -a gamescope | grep -q "generate-drm-mode" pgrep -a gamescope | grep -q "generate-drm-mode"
@ -90,13 +91,16 @@ logger(){
| redact >> "$debug_log" | redact >> "$debug_log"
} }
setup_dirs(){ setup_dirs(){
scroll "STATUS" "Checking directories"
for dir in "$state_path" "$cache_path" "$share_path" "$helpers_path" "$freedesktop_path" "$config_path" "$log_path"; do for dir in "$state_path" "$cache_path" "$share_path" "$helpers_path" "$freedesktop_path" "$config_path" "$log_path"; do
if [[ ! -d $dir ]]; then if [[ ! -d $dir ]]; then
mkdir -p "$dir" mkdir -p "$dir"
fi fi
done done
scroll "RESULT" "OK"
} }
setup_state_files(){ setup_state_files(){
scroll STATUS "Checking state files"
if [[ -f "$debug_log" ]]; then if [[ -f "$debug_log" ]]; then
rm "$debug_log" && touch $debug_log rm "$debug_log" && touch $debug_log
logger INFO "Initializing DZGUI version $version" logger INFO "Initializing DZGUI version $version"
@ -113,6 +117,7 @@ setup_state_files(){
done done
logger INFO "Wiped cache files" logger INFO "Wiped cache files"
fi fi
scroll RESULT OK
} }
print_config_vals(){ print_config_vals(){
local keys=( local keys=(
@ -134,13 +139,14 @@ print_config_vals(){
} }
test_gobject(){ test_gobject(){
scroll STATUS "Checking PyGObject"
python3 -c "import gi" python3 -c "import gi"
if [[ ! $? -eq 0 ]]; then if [[ ! $? -eq 0 ]]; then
logger CRITICAL "Missing PyGObject" raise_error_and_quit "Requires PyGObject (python-gobject)" "silent"
fdialog "Requires PyGObject (python-gobject)"
exit 1 exit 1
fi fi
logger INFO "Found PyGObject in Python env" logger INFO "Found PyGObject in Python env"
scroll RESULT OK
} }
update_config(){ update_config(){
# handling for legacy files # handling for legacy files
@ -222,11 +228,12 @@ src_path="$src_path"
END END
} }
depcheck(){ depcheck(){
scroll STATUS "Checking dependencies"
for dep in "${!deps[@]}"; do for dep in "${!deps[@]}"; do
command -v "$dep" 2>&1>/dev/null command -v "$dep" 2>&1>/dev/null
if [[ $? -eq 1 ]]; then if [[ $? -eq 1 ]]; then
local msg="Requires $dep >= ${deps[$dep]}" local msg="Requires $dep >= ${deps[$dep]}"
raise_error_and_quit "$msg" raise_error_and_quit "$msg" "silent"
fi fi
done done
local jqmsg="jq must be compiled with support for oniguruma" local jqmsg="jq must be compiled with support for oniguruma"
@ -234,22 +241,27 @@ depcheck(){
jqtest=$(echo '{"test": "foo"}' | jq '.test | test("^foo$")') jqtest=$(echo '{"test": "foo"}' | jq '.test | test("^foo$")')
[[ $? -ne 0 ]] && raise_error_and_quit "$jqmsg" [[ $? -ne 0 ]] && raise_error_and_quit "$jqmsg"
logger INFO "Initial dependencies satisfied" logger INFO "Initial dependencies satisfied"
scroll RESULT OK
} }
check_pyver(){ check_pyver(){
scroll STATUS "Checking Python version"
local pyver=$(python3 --version | awk '{print $2}') local pyver=$(python3 --version | awk '{print $2}')
local minor=$(<<< $pyver awk -F. '{print $2}') local minor=$(<<< $pyver awk -F. '{print $2}')
if [[ -z $pyver ]] || [[ ${pyver:0:1} -lt 3 ]] || [[ $minor -lt 10 ]]; then if [[ -z $pyver ]] || [[ ${pyver:0:1} -lt 3 ]] || [[ $minor -lt 10 ]]; then
local msg="Requires Python >=3.10" local msg="Requires Python >=3.10"
raise_error_and_quit "$msg" raise_error_and_quit "$msg" "silent"
fi fi
logger INFO "Found Python version: $pyver" logger INFO "Found Python version: $pyver"
scroll RESULT OK
} }
watcher_deps(){ watcher_deps(){
scroll STATUS "Checking dialog dependencies"
if [[ ! $(command -v wmctrl) ]] && [[ ! $(command -v xdotool) ]]; then if [[ ! $(command -v wmctrl) ]] && [[ ! $(command -v xdotool) ]]; then
raise_error_and_quit "Missing dependency: requires 'wmctrl' or 'xdotool'" raise_error_and_quit "Missing dependency: requires 'wmctrl' or 'xdotool'" "silent"
exit 1 exit 1
fi fi
logger INFO "Found DZG Watcher dependencies" logger INFO "Found DZG Watcher dependencies"
scroll "RESULT" "OK"
} }
format_version_url(){ format_version_url(){
[[ -z "$branch" ]] && branch="stable" [[ -z "$branch" ]] && branch="stable"
@ -277,6 +289,7 @@ Categories=Game
END END
} }
freedesktop_dirs(){ freedesktop_dirs(){
#scroll STATUS "Setting up Freedesktop dirs"
local version_url=$(format_version_url) local version_url=$(format_version_url)
local img_url="$stable_url/images" local img_url="$stable_url/images"
curl -s "$version_url" > "$script_path" curl -s "$version_url" > "$script_path"
@ -287,22 +300,21 @@ freedesktop_dirs(){
write_desktop_file > "$freedesktop_path/$app_name.desktop" write_desktop_file > "$freedesktop_path/$app_name.desktop"
[[ $is_steam_deck -eq 0 ]] && return [[ $is_steam_deck -eq 0 ]] && return
write_desktop_file > "$HOME/Desktop/$app_name.desktop" write_desktop_file > "$HOME/Desktop/$app_name.desktop"
#scroll RESULT OK
} }
legacy_vars(){ legacy_vars(){
scroll "STATUS" "Checking for old config format"
local suffix="fav" local suffix="fav"
local hr_msg="Config file contains values based on old API. Please update and re-run setup." local msg="Config file contains legacy API value '$suffix'. Please update and re-run setup."
local msg="Config file contains legacy API value: '$suffix'"
if [[ -n $fav ]]; then if [[ -n $fav ]]; then
logger WARN "$msg" raise_error_and_quit "$msg" "silent"
fdialog "$hr_msg"
exit 1
fi fi
if [[ -n $whitelist ]]; then if [[ -n $whitelist ]]; then
suffix="whitelist" suffix="whitelist"
logger WARN "$msg" msg="Config file contains legacy API value '$suffix'. Please update and re-run setup."
fdialog "$hr_msg" raise_error_and_quit "$msg" "silent"
exit 1
fi fi
scroll "RESULT" "OK"
} }
merge_config(){ merge_config(){
[[ -z $staging_dir ]] && staging_dir="/tmp" [[ -z $staging_dir ]] && staging_dir="/tmp"
@ -316,10 +328,11 @@ check_unmerged(){
fi fi
} }
check_version(){ check_version(){
scroll "STATUS" "Looking for version updates"
local version_url=$(format_version_url) local version_url=$(format_version_url)
local upstream=$(curl -Ls "$version_url" | awk -F= '/^version=/ {print $2}') local upstream=$(curl -Ls "$version_url" | awk -F= '/^version=/ {print $2}')
local res=$(get_response_code "$version_url") local res=$(get_response_code "$version_url")
[[ $res -ne 200 ]] && raise_error_and_quit "Remote resource unavailable: '$version_url'" [[ $res -ne 200 ]] && raise_error_and_quit "Remote resource unavailable: '$version_url'" "silent"
logger INFO "Local branch: '$branch', local version: $version" logger INFO "Local branch: '$branch', local version: $version"
if [[ $branch == "stable" ]]; then if [[ $branch == "stable" ]]; then
version_url="$stable_url/dzgui.sh" version_url="$stable_url/dzgui.sh"
@ -332,8 +345,11 @@ check_version(){
check_unmerged check_unmerged
else else
logger WARN "Local and remote version mismatch: $version != $upstream" logger WARN "Local and remote version mismatch: $version != $upstream"
scroll "RESULT" "WARN"
prompt_dl prompt_dl
return 0
fi fi
scroll "RESULT" "OK"
} }
download_new_version(){ download_new_version(){
local version_url="$(format_version_url)" local version_url="$(format_version_url)"
@ -373,6 +389,7 @@ prompt_dl(){
if [[ $? -eq 1 ]]; then if [[ $? -eq 1 ]]; then
return 0 return 0
else else
scroll "EXIT"
download_new_version download_new_version
fi fi
} }
@ -393,11 +410,13 @@ test_display_mode(){
fi fi
} }
check_architecture(){ check_architecture(){
scroll "STATUS" "Setting architecture"
local cpu=$(< /proc/cpuinfo awk -F": " '/AMD Custom APU [0-9]{4}$/ {print $2; exit}') local cpu=$(< /proc/cpuinfo awk -F": " '/AMD Custom APU [0-9]{4}$/ {print $2; exit}')
read -a APU_MODEL <<< "$cpu" read -a APU_MODEL <<< "$cpu"
if [[ ${APU_MODEL[3]} != "0932" ]] && [[ ${APU_MODEL[3]} != "0405" ]]; then if [[ ${APU_MODEL[3]} != "0932" ]] && [[ ${APU_MODEL[3]} != "0405" ]]; then
is_steam_deck=0 is_steam_deck=0
logger INFO "Setting architecture to 'desktop'" logger INFO "Setting architecture to 'desktop'"
scroll "RESULT" "OK"
return return
fi fi
@ -407,20 +426,27 @@ check_architecture(){
is_steam_deck=1 is_steam_deck=1
fi fi
logger INFO "Setting architecture to 'Steam Deck'" logger INFO "Setting architecture to 'Steam Deck'"
scroll "RESULT" "OK"
} }
check_map_count(){ check_map_count(){
[[ $is_steam_deck -gt 0 ]] && return 0 scroll "STATUS" "Checking system map count"
if [[ $is_steam_deck -gt 0 ]]; then
scroll "RESULT" "OK"
return 0
fi
local map_count_file="/proc/sys/vm/max_map_count" local map_count_file="/proc/sys/vm/max_map_count"
local min_count=1048576 local min_count=1048576
local conf_file="/etc/sysctl.d/dayz.conf" local conf_file="/etc/sysctl.d/dayz.conf"
local current_count local current_count
if [[ ! -f ${map_count_file} ]]; then if [[ ! -f ${map_count_file} ]]; then
scroll "RESULT" "FAIL"
logger WARN "File '${map_count_file}' doesn't exist!" logger WARN "File '${map_count_file}' doesn't exist!"
return 1 return 1
fi fi
current_count=$(cat ${map_count_file}) current_count=$(cat ${map_count_file})
if [[ $current_count -ge $min_count ]]; then if [[ $current_count -ge $min_count ]]; then
logger DEBUG "System map count is set to ${current_count}" logger DEBUG "System map count is set to ${current_count}"
scroll "RESULT" "OK"
return 0 return 0
fi fi
qdialog "sudo password required to set system vm map count." "OK" "Cancel" qdialog "sudo password required to set system vm map count." "OK" "Cancel"
@ -430,6 +456,7 @@ check_map_count(){
pass=$($steamsafe_zenity --password) pass=$($steamsafe_zenity --password)
if [[ $? -eq 1 ]]; then if [[ $? -eq 1 ]]; then
logger WARN "User aborted password prompt" logger WARN "User aborted password prompt"
scroll "RESULT" "FAIL"
return 1 return 1
fi fi
logger DEBUG "Old map count is $current_count" logger DEBUG "Old map count is $current_count"
@ -437,8 +464,10 @@ check_map_count(){
sudo -S <<< "$pass" sh -c "echo 'vm.max_map_count=${current_count}' > $conf_file" sudo -S <<< "$pass" sh -c "echo 'vm.max_map_count=${current_count}' > $conf_file"
sudo sysctl -p "$conf_file" sudo sysctl -p "$conf_file"
logger DEBUG "Updated map count to $min_count" logger DEBUG "Updated map count to $min_count"
scroll "RESULT" "OK"
else else
logger WARN "User aborted map count prompt" logger WARN "User aborted map count prompt"
scroll "RESULT" "FAIL"
return 1 return 1
fi fi
} }
@ -458,14 +487,14 @@ tdialog(){
$steamsafe_zenity --info --text="$1" "${zenity_flags[@]}" $steamsafe_zenity --info --text="$1" "${zenity_flags[@]}"
} }
steam_deps(){ steam_deps(){
scroll "STATUS" "Checking Steam dependencies"
local flatpak local flatpak
local steam local steam
[[ $(command -v flatpak) ]] && flatpak=$(flatpak list | grep valvesoftware.Steam) [[ $(command -v flatpak) ]] && flatpak=$(flatpak list | grep valvesoftware.Steam)
steam=$(command -v steam) steam=$(command -v steam)
if [[ -z "$steam" ]] && [[ -z "$flatpak" ]]; then if [[ -z "$steam" ]] && [[ -z "$flatpak" ]]; then
local msg="Found neither Steam nor Flatpak Steam" local msg="Found neither Steam nor Flatpak Steam"
raise_error_and_quit "$msg" raise_error_and_quit "$msg" "silent"
exit 1
elif [[ -n "$steam" ]] && [[ -n "$flatpak" ]]; then elif [[ -n "$steam" ]] && [[ -n "$flatpak" ]]; then
[[ -n $preferred_client ]] && return 0 [[ -n $preferred_client ]] && return 0
if [[ -z $preferred_client ]]; then if [[ -z $preferred_client ]]; then
@ -478,24 +507,35 @@ steam_deps(){
fi fi
update_config update_config
logger INFO "Preferred client set to '$preferred_client'" logger INFO "Preferred client set to '$preferred_client'"
scroll "RESULT" "OK"
} }
migrate_files(){ migrate_files(){
scroll "STATUS" "Checking legacy API files"
if [[ ! -f $config_path/dztuirc.oldapi ]]; then if [[ ! -f $config_path/dztuirc.oldapi ]]; then
cp $config_file $config_path/dztuirc.oldapi cp $config_file $config_path/dztuirc.oldapi
logger INFO "Migrated old API file" logger INFO "Migrated old API file"
fi fi
[[ ! -f $hist_file ]] && return if [[ ! -f $hist_file ]]; then
scroll "RESULT" "OK"
return
fi
rm $hist_file rm $hist_file
logger INFO "Wiped old history file" logger INFO "Wiped old history file"
scroll "RESULT" "OK"
} }
stale_symlinks(){ stale_symlinks(){
#TODO: test progress update
scroll "STATUS" "Cleaning stale symlinks"
local game_dir="$steam_path/steamapps/common/DayZ" local game_dir="$steam_path/steamapps/common/DayZ"
for l in $(find "$game_dir" -xtype l); do for l in $(find "$game_dir" -xtype l); do
scroll "PROGRESS" "Updating symlink '$l'"
logger DEBUG "Updating stale symlink '$l'" logger DEBUG "Updating stale symlink '$l'"
unlink "$l" unlink "$l"
done done
scroll "RESULT" "OK"
} }
local_latlon(){ local_latlon(){
scroll "STATUS" "Getting geolocation"
if [[ -z $(command -v dig) ]]; then if [[ -z $(command -v dig) ]]; then
local local_ip=$(curl -Ls "https://ipecho.net/plain") local local_ip=$(curl -Ls "https://ipecho.net/plain")
else else
@ -505,29 +545,40 @@ local_latlon(){
local res=$(curl -Ls "$url" | jq -r '"\(.lat)\n\(.lon)"') local res=$(curl -Ls "$url" | jq -r '"\(.lat)\n\(.lon)"')
if [[ -z "$res" ]]; then if [[ -z "$res" ]]; then
logger WARN "Failed to get local coordinates" logger WARN "Failed to get local coordinates"
return 1 tdialog "Failed to get local coordinates"
scroll "RESULT" "WARN"
#this is non blocking, so scroller does not abort
return 0
fi fi
echo "$res" > "$coords_file" echo "$res" > "$coords_file"
scroll "RESULT" "OK"
} }
lock(){ lock(){
scroll "STATUS" "Checking for existing DZGUI"
#TODO: check for existing scroller from other context
#is_scroller_running
#[[ $? -eq 0 ]] && raise_error_and_quit "DZGUI is already running ($scroller_pid)" "silent"
[[ ! -f $lock_file ]] && touch $lock_file [[ ! -f $lock_file ]] && touch $lock_file
local pid=$(cat $lock_file) local pid=$(cat $lock_file)
ps -p $pid -o pid= >/dev/null 2>&1 ps -p $pid -o pid= >/dev/null 2>&1
res=$? res=$?
if [[ $res -eq 0 ]]; then if [[ $res -eq 0 ]]; then
local msg="DZGUI already running ($pid)" local msg="DZGUI already running ($pid)"
raise_error_and_quit "$msg" raise_error_and_quit "$msg" "silent"
elif [[ $pid == $$ ]]; then elif [[ $pid == $$ ]]; then
: :
else else
echo $$ > $lock_file echo $$ > $lock_file
fi fi
scroll "RESULT" "OK"
} }
get_hash(){ get_hash(){
local file="$1" local file="$1"
md5sum "$1" | awk '{print $1}' md5sum "$1" | awk '{print $1}'
} }
fetch_a2s(){ fetch_a2s(){
#scroll STATUS "Checking query helper"
#scroll PROGRESS "Checking A2S"
# this file is currently monolithic # this file is currently monolithic
[[ -d $helpers_path/a2s ]] && { logger INFO "A2S helper is current"; return 0; } [[ -d $helpers_path/a2s ]] && { logger INFO "A2S helper is current"; return 0; }
local sha=c7590ffa9a6d0c6912e17ceeab15b832a1090640 local sha=c7590ffa9a6d0c6912e17ceeab15b832a1090640
@ -537,13 +588,15 @@ fetch_a2s(){
local prefix="${author^}-$repo-${sha:0:7}" local prefix="${author^}-$repo-${sha:0:7}"
local file="$prefix.tar.gz" local file="$prefix.tar.gz"
local res=$(get_response_code "$url") local res=$(get_response_code "$url")
[[ $res -ne 200 ]] && raise_error_and_quit "Remote resource unavailable: '$file'" [[ $res -ne 200 ]] && raise_error_and_quit "Remote resource unavailable: '$file'" "silent"
curl -Ls "$url" > "$helpers_path/$file" curl -Ls "$url" > "$helpers_path/$file"
tar xf "$helpers_path/$file" -C "$helpers_path" "$prefix/a2s" --strip=1 tar xf "$helpers_path/$file" -C "$helpers_path" "$prefix/a2s" --strip=1
rm "$helpers_path/$file" rm "$helpers_path/$file"
logger INFO "Updated A2S helper to sha '$sha'" logger INFO "Updated A2S helper to sha '$sha'"
#scroll RESULT OK
} }
fetch_dzq(){ fetch_dzq(){
#scroll PROGRESS "Checking DZQ"
local sum="9caed1445c45832f4af87736ba3f9637" local sum="9caed1445c45832f4af87736ba3f9637"
local file="$helpers_path/a2s/dayzquery.py" local file="$helpers_path/a2s/dayzquery.py"
if [[ -f $file ]] && [[ $(get_hash "$file") == $sum ]]; then if [[ -f $file ]] && [[ $(get_hash "$file") == $sum ]]; then
@ -555,11 +608,13 @@ fetch_dzq(){
local repo="dayzquery" local repo="dayzquery"
local url="https://raw.githubusercontent.com/$author/$repo/$sha/dayzquery.py" local url="https://raw.githubusercontent.com/$author/$repo/$sha/dayzquery.py"
local res=$(get_response_code "$url") local res=$(get_response_code "$url")
[[ $res -ne 200 ]] && raise_error_and_quit "Remote resource unavailable: 'dayzquery.py'" [[ $res -ne 200 ]] && raise_error_and_quit "Remote resource unavailable: 'dayzquery.py'" "silent"
curl -Ls "$url" > "$file" curl -Ls "$url" > "$file"
logger INFO "Updated DZQ to sha '$sha'" logger INFO "Updated DZQ to sha '$sha'"
#scroll RESULT OK
} }
fetch_icons(){ fetch_icons(){
#scroll STATUS "Fetching icons"
res=( res=(
"16" "16"
"24" "24"
@ -572,6 +627,7 @@ fetch_icons(){
) )
url="$stable_url/images/icons" url="$stable_url/images/icons"
for i in "${res[@]}"; do for i in "${res[@]}"; do
#scroll PROGRESS "Fetching size ${i}x${i}"
size="${i}x${i}" size="${i}x${i}"
dir="$HOME/.local/share/icons/hicolor/$size/apps" dir="$HOME/.local/share/icons/hicolor/$size/apps"
icon="$dir/$app_name.png" icon="$dir/$app_name.png"
@ -582,8 +638,10 @@ fetch_icons(){
logger INFO "Updating $size Freedesktop icon" logger INFO "Updating $size Freedesktop icon"
curl -Ls "${url}/${i}.png" > "$icon" curl -Ls "${url}/${i}.png" > "$icon"
done done
#scroll RESULT OK
} }
fetch_helpers_by_sum(){ fetch_helpers_by_sum(){
#scroll STATUS "Checking other helpers"
[[ -f "$config_file" ]] && source "$config_file" [[ -f "$config_file" ]] && source "$config_file"
declare -A sums declare -A sums
sums=( sums=(
@ -616,16 +674,17 @@ fetch_helpers_by_sum(){
full_path="$helpers_path/$file" full_path="$helpers_path/$file"
url="${url_prefix}/$realbranch/helpers/$file" url="${url_prefix}/$realbranch/helpers/$file"
#scroll PROGRESS "Checking $file"
if [[ -f "$full_path" ]] && [[ $(get_hash "$full_path") == $sum ]]; then if [[ -f "$full_path" ]] && [[ $(get_hash "$full_path") == $sum ]]; then
logger INFO "$file is current" logger INFO "$file is current"
else else
logger WARN "File '$full_path' checksum != '$sum'" logger WARN "File '$full_path' checksum != '$sum'"
local res=$(get_response_code "$url") local res=$(get_response_code "$url")
[[ $res -ne 200 ]] && raise_error_and_quit "Remote resource unavailable: '$url'" [[ $res -ne 200 ]] && raise_error_and_quit "Remote resource unavailable: '$url'" "silent"
curl -Ls "$url" > "$full_path" curl -Ls "$url" > "$full_path"
if [[ ! $? -eq 0 ]]; then if [[ ! $? -eq 0 ]]; then
raise_error_and_quit "Failed to fetch the file '$file'. Possible timeout?" raise_error_and_quit "Failed to fetch the file '$file'. Possible timeout?" "silent"
fi fi
if [[ $(get_hash $full_path) != $sum ]]; then if [[ $(get_hash $full_path) != $sum ]]; then
logger WARN "Downloaded new '$file', but checksum != '$sum'" logger WARN "Downloaded new '$file', but checksum != '$sum'"
@ -635,26 +694,31 @@ fetch_helpers_by_sum(){
[[ $file == "funcs" ]] && chmod +x "$full_path" [[ $file == "funcs" ]] && chmod +x "$full_path"
[[ $file == "lan" ]] && chmod +x "$full_path" [[ $file == "lan" ]] && chmod +x "$full_path"
done done
#scroll RESULT OK
return 0 return 0
} }
fetch_geo_file(){ fetch_geo_file(){
#scroll STATUS "Checking geo file"
# for binary releases # for binary releases
local geo_sum="9824e9b9a75a4830a2423932cc188b06" local geo_sum="9824e9b9a75a4830a2423932cc188b06"
local km_sum="b038fdb8f655798207bd28de3a004706" local km_sum="b038fdb8f655798207bd28de3a004706"
local gzip="$helpers_path/ips.csv.gz" local gzip="$helpers_path/ips.csv.gz"
if [[ ! -f $geo_file ]] || [[ $(get_hash $geo_file) != $geo_sum ]]; then if [[ ! -f $geo_file ]] || [[ $(get_hash $geo_file) != $geo_sum ]]; then
local res=$(get_response_code "$geo_file_url") local res=$(get_response_code "$geo_file_url")
[[ $res -ne 200 ]] && raise_error_and_quit "Remote resource unavailable: '$geo_file_url'" [[ $res -ne 200 ]] && raise_error_and_quit "Remote resource unavailable: '$geo_file_url'" "silent"
curl -Ls "$geo_file_url" > "$gzip" curl -Ls "$geo_file_url" > "$gzip"
#force overwrite #force overwrite
gunzip -f "$gzip" gunzip -f "$gzip"
fi fi
#scroll RESULT OK
#scroll STATUS "Checking km helper"
if [[ ! -f $km_helper ]] || [[ $(get_hash $km_helper) != $km_sum ]]; then if [[ ! -f $km_helper ]] || [[ $(get_hash $km_helper) != $km_sum ]]; then
local res=$(get_response_code "$km_helper_url") local res=$(get_response_code "$km_helper_url")
[[ $res -ne 200 ]] && raise_error_and_quit "Remote resource unavailable: '$km_helper_url'" [[ $res -ne 200 ]] && raise_error_and_quit "Remote resource unavailable: '$km_helper_url'" "silent"
curl -Ls "$km_helper_url" > "$km_helper" curl -Ls "$km_helper_url" > "$km_helper"
chmod +x "$km_helper" chmod +x "$km_helper"
fi fi
#scroll RESULT OK
} }
fetch_helpers(){ fetch_helpers(){
fetch_a2s fetch_a2s
@ -667,7 +731,8 @@ fetch_helpers(){
raise_error_and_quit(){ raise_error_and_quit(){
local msg="$1" local msg="$1"
logger CRITICAL "$msg" logger CRITICAL "$msg"
fdialog "$msg" scroll "RESULT" "FAIL" "$msg"
[[ ! $2 == "silent" ]] && fdialog "$msg"
exit 1 exit 1
} }
test_steam_api(){ test_steam_api(){
@ -828,9 +893,12 @@ create_config(){
done done
} }
varcheck(){ varcheck(){
#TODO: test branching scroller here
scroll "STATUS" "Testing config variables"
local msg="Config file '$config_file' missing. Start first-time setup now?" local msg="Config file '$config_file' missing. Start first-time setup now?"
local msg2="The Steam paths set in your config file appear to be invalid. Restart first-time setup now?" local msg2="The Steam paths set in your config file appear to be invalid. Restart first-time setup now?"
if [[ ! -f $config_file ]]; then if [[ ! -f $config_file ]]; then
scroll "EXIT"
qdialog "$msg" "Yes" "Exit" qdialog "$msg" "Yes" "Exit"
if [[ $? -eq 1 ]]; then if [[ $? -eq 1 ]]; then
logger CRITICAL "Config file missing, but user aborted setup" logger CRITICAL "Config file missing, but user aborted setup"
@ -844,6 +912,9 @@ varcheck(){
if [[ ! -d $steam_path ]] || [[ ! -d $game_dir ]] || [[ ! $(find $game_dir -type f) ]]; then if [[ ! -d $steam_path ]] || [[ ! -d $game_dir ]] || [[ ! $(find $game_dir -type f) ]]; then
logger WARN "DayZ path resolved to '$game_dir'" logger WARN "DayZ path resolved to '$game_dir'"
logger WARN "Workshop path resolved to '$workshop_dir'" logger WARN "Workshop path resolved to '$workshop_dir'"
#scroller might already be destroyed from prior step
is_scroller_running
[[ $? -eq 0 ]] && scroll "EXIT"
qdialog "$msg2" "Yes" "Exit" qdialog "$msg2" "Yes" "Exit"
if [[ $? -eq 1 ]]; then if [[ $? -eq 1 ]]; then
logger CRITICAL "Malformed Steam path, but user aborted setup" logger CRITICAL "Malformed Steam path, but user aborted setup"
@ -856,40 +927,54 @@ varcheck(){
src_path=$(realpath "$0") src_path=$(realpath "$0")
update_config update_config
fi fi
is_scroller_running
[[ $? -eq 0 ]] && scroll "RESULT" "OK"
} }
is_dzg_downloading(){ is_dzg_downloading(){
scroll "STATUS" "Checking if DayZ is up to date"
local msg="DayZ may be scheduling updates on Steam"
if [[ -d $steam_path ]] && [[ -d $steam_path/downloading/$aid ]]; then if [[ -d $steam_path ]] && [[ -d $steam_path/downloading/$aid ]]; then
logger WARN "DayZ may be scheduling updates" #this is non blocking, so scroller does not abort
#TODO: add a new scroller warning category
scroll "RESULT" "WARN"
tdialog "$msg"
logger WARN "$msg"
return 0 return 0
fi fi
scroll "RESULT" "OK"
} }
is_steam_running(){ is_steam_running(){
scroll "STATUS" "Checking for Steam"
local res=$(ps aux | grep "steamwebhelper" | grep -v grep) local res=$(ps aux | grep "steamwebhelper" | grep -v grep)
if [[ -z $res ]]; then if [[ -z $res ]]; then
#this is non blocking, so scroller does not abort
#TODO: add a new scroller warning category
logger WARN "Steam may not be running" logger WARN "Steam may not be running"
tdialog "Is Steam running? For best results, make sure Steam is open in the background." tdialog "Is Steam running? For best results, make sure Steam is open in the background."
return 0 return 0
fi fi
scroll "RESULT" "OK"
} }
get_response_code(){ get_response_code(){
local url="$1" local url="$1"
curl -Ls -I -o /dev/null -w "%{http_code}" "$url" curl -Ls -I -o /dev/null -w "%{http_code}" "$url"
} }
test_connection(){ test_connection(){
scroll "STATUS" "Testing connection"
source "$config_file" source "$config_file"
declare -A hr declare -A hr
local res1 local res1
local res2 local res2
local str="No connection could be established to the remote server" local str="No connection could be established to the remote server"
hr=( hr=(
["steampowered.com"]="https://api.steampowered.com/IGameServersService/GetServerList/v1/?key=$steam_api" ["steampowered.com"]="https://api.steampowered.com/IGameServersService/GetServerList/v1/?filter=\appid\221100&limit=10&key=$steam_api"
["github.com"]="https://github.com/$author" ["github.com"]="https://github.com/$author"
["codeberg.org"]="https://codeberg.org/$author" ["codeberg.org"]="https://codeberg.org/$author"
) )
# steam API is mandatory, except on initial setup # steam API is mandatory, except on initial setup
if [[ -n $steam_api ]]; then if [[ -n $steam_api ]]; then
res=$(get_response_code "${hr["steampowered.com"]}") res=$(get_response_code "${hr["steampowered.com"]}")
[[ $res -ne 200 ]] && raise_error_and_quit "$str ("steampowered.com")" [[ $res -ne 200 ]] && raise_error_and_quit "$str ("steampowered.com")" "silent"
fi fi
res=$(get_response_code "${hr["github.com"]}") res=$(get_response_code "${hr["github.com"]}")
@ -898,8 +983,9 @@ test_connection(){
remote_host=cb remote_host=cb
logger INFO "Set remote host to '${hr["codeberg.org"]}'" logger INFO "Set remote host to '${hr["codeberg.org"]}'"
res=$(get_response_code "${hr["codeberg.org"]}") res=$(get_response_code "${hr["codeberg.org"]}")
[[ $res -ne 200 ]] && raise_error_and_quit "$str (${hr["codeberg.org"]})" [[ $res -ne 200 ]] && raise_error_and_quit "$str (${hr["codeberg.org"]})" "silent"
fi fi
scroll "RESULT" "OK"
if [[ $remote_host == "cb" ]]; then if [[ $remote_host == "cb" ]]; then
url_prefix="https://codeberg.org/$author/$repo/raw/branch" url_prefix="https://codeberg.org/$author/$repo/raw/branch"
releases_url="https://codeberg.org/$author/$repo/releases/download/browser" releases_url="https://codeberg.org/$author/$repo/releases/download/browser"
@ -910,28 +996,38 @@ test_connection(){
fi fi
} }
legacy_cols(){ legacy_cols(){
scroll "STATUS" "Checking column preferences"
[[ ! -f $cols_file ]] && return [[ ! -f $cols_file ]] && return
local has=$(< "$cols_file" jq '.cols|has("Queue")') local has=$(< "$cols_file" jq '.cols|has("Queue")')
[[ $has == "true" ]] && return if [[ $has == "true" ]]; then
scroll "RESULT" "OK"
return
fi
< $cols_file jq '.cols += { "Queue": 120 }' > $cols_file.new && < $cols_file jq '.cols += { "Queue": 120 }' > $cols_file.new &&
mv $cols_file.new $cols_file mv $cols_file.new $cols_file
scroll "RESULT" "OK"
} }
stale_mod_signatures(){ stale_mod_signatures(){
scroll "STATUS" "Cleaning old mod signatures"
local workshop_dir="$steam_path/steamapps/workshop/content/$aid" local workshop_dir="$steam_path/steamapps/workshop/content/$aid"
if [[ -d $workshop_dir ]]; then if [[ -d $workshop_dir ]]; then
readarray -t old_mod_ids < <(awk -F, '{print $1}' $versions_file) readarray -t old_mod_ids < <(awk -F, '{print $1}' $versions_file)
for ((i=0; i<${#old_mod_ids[@]}; ++i)); do for ((i=0; i<${#old_mod_ids[@]}; ++i)); do
if [[ ! -d $workshop_dir/${old_mod_ids[$i]} ]]; then if [[ ! -d $workshop_dir/${old_mod_ids[$i]} ]]; then
scroll "PROGRESS" "Updating mod '${old_mod_ids[$i]}'"
"$func_helper" "align_local" "${old_mod_ids[$i]}" "$func_helper" "align_local" "${old_mod_ids[$i]}"
fi fi
done done
fi fi
scroll "RESULT" "OK"
} }
create_new_links(){ create_new_links(){
scroll "STATUS" "Updating symlinks"
"$func_helper" "update_symlinks" "$func_helper" "update_symlinks"
scroll "RESULT" "OK"
} }
initial_setup(){ initial_setup(){
lock
setup_dirs setup_dirs
setup_state_files setup_state_files
depcheck depcheck
@ -939,11 +1035,11 @@ initial_setup(){
test_gobject test_gobject
watcher_deps watcher_deps
check_architecture check_architecture
test_connection
fetch_helpers > >(pdialog "Checking helper files")
varcheck varcheck
test_connection
#TODO: test scroller with helper files
# fetch_helpers > >(pdialog "Checking helper files")
source "$config_file" source "$config_file"
lock
legacy_vars legacy_vars
legacy_cols legacy_cols
check_version check_version
@ -1000,6 +1096,40 @@ uninstall(){
rm "$self" rm "$self"
echo "Uninstall routine complete" echo "Uninstall routine complete"
} }
is_scroller_running(){
[[ -z $scroller_pid ]] && return
ps -p $scroller_pid > /dev/null
[[ $? -eq 1 ]] && return 1
ps aux | grep scroll.py | grep -v grep > /dev/null
[[ $? -eq 1 ]] && return 1
return 0
}
scroll(){
#blocking process; abort if fifo is absent
[[ ! -p "$scroll_fifo" ]] && return
local flag="$1"
local str="$2"
local msg="$3"
sleep 0.1s
printf "%s␞%s␞%s" "$flag" "$str" "$msg" > "$scroll_fifo"
}
destroy_scroller(){
is_scroller_running
[[ $? -eq 0 ]] && scroll "EXIT"
logger INFO "Caught abort/hangup signal"
[[ -p "$scroll_fifo" ]] && rm "$scroll_fifo"
exit 1
}
write_pid(){
local dzg_pid=$$
echo "$dzg_pid" > "$state_path/$prefix.pid"
}
main(){ main(){
local zenv=$(zenity --version 2>/dev/null) local zenv=$(zenity --version 2>/dev/null)
[[ -z $zenv ]] && { echo "Requires zenity >= ${deps[$steamsafe_zenity]}"; exit 1; } [[ -z $zenv ]] && { echo "Requires zenity >= ${deps[$steamsafe_zenity]}"; exit 1; }
@ -1011,11 +1141,35 @@ main(){
set_im_module set_im_module
printf "Initializing setup...\n" printf "Initializing setup...\n"
#TODO: download scroller if missing
#cf. fetch_helpers
write_pid
[[ -p "$scroll_fifo" ]] && rm "$scroll_fifo"
if [[ -f "$scroll_helper" ]]; then
mkfifo "$scroll_fifo"
python3 "$scroll_helper" &
scroller_pid=$!
fi
initial_setup initial_setup
#if all ok, destroy scroller
scroll "EXIT"
printf "All OK. Kicking off UI...\n" printf "All OK. Kicking off UI...\n"
python3 "$ui_helper" "--init-ui" "$version" "$is_steam_deck" python3 "$ui_helper" "--init-ui" "$version" "$is_steam_deck"
} }
quit_from_scroller(){
logger INFO "User triggered early abort from loading dialog"
exit 1
}
#early halt signals from script must destroy loader
trap destroy_scroller SIGINT SIGHUP SIGQUIT SIGTERM
#cancel button in loader will trigger this signal
trap quit_from_scroller SIGUSR1
main "$@" main "$@"
#TODO: tech debt: cruddy handling for steam forking #TODO: tech debt: cruddy handling for steam forking
[[ $? -eq 1 ]] && pkill -f dzgui.sh [[ $? -eq 1 ]] && pkill -f dzgui.sh

238
helpers/scroll.py Normal file
View file

@ -0,0 +1,238 @@
import gi
import signal
import subprocess
import sys
import textwrap
import threading
import time
import os
user_path = os.path.expanduser('~')
fifo_path = '%s/.local/state/dzgui/dzg.fifo' %(user_path)
pid_path = "%s/.local/state/dzgui/dzg.pid" %(user_path)
FIFO = fifo_path
gi.require_version("Gtk", "3.0")
from gi.repository import Gtk, GLib, Gdk, GObject, Pango
main_app = "DZGUI"
app = "DZGUI Loader"
version = "5.6.0"
class Dialog(Gtk.MessageDialog):
def __init__(self, parent):
Gtk.MessageDialog.__init__(
self,
transient_for=parent,
buttons=Gtk.ButtonsType.OK,
flags=0,
text="Some text",
title=app,
modal=True,
)
self.connect("delete-event", self._on_dialog_delete)
self.set_size_request(1000, 0)
self.set_title(app)
self.set_default_response(Gtk.ResponseType.OK)
def _on_dialog_delete(self, resp_id, some):
return True
class OuterWindow(Gtk.Window):
def __init__(self):
super().__init__(title=app)
self.set_border_width(10)
outervbox = Gtk.Box()
outervbox.set_margin_top(50)
outervbox.set_margin_start(50)
outervbox.set_margin_end(50)
outervbox.set_orientation(Gtk.Orientation.VERTICAL)
self.model = Gtk.ListStore(str, str, str)
self.treeview = Gtk.TreeView(model=self.model)
columns = ["Process", "Result", "Color"]
for i, column_title in enumerate(columns):
renderer = Gtk.CellRendererText()
if i == 1:
renderer.set_property("weight", Pango.Weight.BOLD)
column = Gtk.TreeViewColumn(column_title, renderer, text=i, foreground=2)
else:
column = Gtk.TreeViewColumn(column_title, renderer, text=i)
column.set_fixed_width(400)
if i != 2:
self.treeview.append_column(column)
self.treeview.set_headers_visible(False)
self.treeview.get_selection().set_select_function(self.select_function)
self.scrolled = Gtk.ScrolledWindow()
self.scrolled.set_propagate_natural_width(True)
self.scrolled.set_vexpand(True)
self.scrolled.set_hexpand(True)
# vbox > label > button box > left buttons > right button
self.early_cancel_box = Gtk.Box()
self.early_cancel_box.set_orientation(Gtk.Orientation.HORIZONTAL)
self.cancel = Gtk.Button(label="Cancel")
self.early_cancel_box.add(self.cancel)
self.cancel.connect("clicked", self._on_cancel_clicked)
self.box = Gtk.Box(halign=Gtk.Align.FILL)
self.box.set_orientation(Gtk.Orientation.HORIZONTAL)
self.box2 = Gtk.Box(hexpand=True, halign=Gtk.Align.END)
self.aa = Gtk.Button(label="Submit a bug report ⧉")
self.cc = Gtk.Button(label="Exit")
self.cc.connect("clicked", self._on_exit_clicked)
self.box.add(self.aa)
self.box.add(self.box2)
self.box2.add(self.cc)
self.box.set_spacing(10)
self.box.set_margin_top(10)
self.scrolled.add(self.treeview)
self.label = Gtk.Label()
self.label.set_text(main_app + " " + version)
self.label2 = Gtk.Label()
self.label2.set_text(main_app + " is starting up")
self.label2.set_margin_top(10)
self.label2.set_margin_bottom(10)
self.spinner = Gtk.Spinner()
self.spinner.start()
self.aa.connect("clicked", self._on_button_clicked)
self.spinner_box = Gtk.Box(orientation=Gtk.Orientation.VERTICAL)
self.spinner_box.add(self.label)
self.spinner_box.add(Gtk.Separator())
self.spinner_box.add(self.label2)
self.spinner_box.add(self.spinner)
self.pg = Gtk.ProgressBar()
self.grid = Gtk.Grid()
#self.grid.set_column_homogeneous(True)
#self.grid.set_row_homogeneous(True)
self.grid.attach(self.spinner_box, 0, 0, 4, 1)
#self.grid.attach(self.scrolled, 0, 1, 4, 1)
self.errors = Gtk.Box()
self.errors_label = Gtk.Label()
self.errors.add(self.errors_label)
self.grid.attach_next_to(self.scrolled, self.spinner_box, Gtk.PositionType.BOTTOM, 4, 20)
self.grid.attach_next_to(self.early_cancel_box, self.scrolled, Gtk.PositionType.BOTTOM, 4, 1)
self.grid.attach_next_to(self.errors, self.scrolled, Gtk.PositionType.BOTTOM, 4, 1)
self.grid.attach_next_to(self.box, self.early_cancel_box, Gtk.PositionType.BOTTOM, 4, 1)
self.grid.set_row_spacing(20)
self.add(self.grid)
self.show_all()
self.box.set_visible(False)
self.errors.set_visible(False)
self.thread = threading.Thread(target=self.log, args=())
self.thread.start()
def _on_cancel_clicked(self, button):
with open (pid_path) as f:
pid = int(f.read())
os.kill(pid, signal.SIGUSR1)
self.destroy()
os.remove(FIFO)
Gtk.main_quit()
#self._on_exit_clicked(button)
def _on_exit_clicked(self, button):
self.destroy()
os.remove(FIFO)
Gtk.main_quit()
def select_function(self, treeselection, model, path, current):
state = True
def scroll_to_end(self):
adj = self.scrolled.get_vadjustment()
adj.set_value(adj.get_upper() + adj.get_page_size())
def log(self):
d = ""
def update_gui():
tip = len(self.model)
if (tip - 1) < 0:
tip = 0
else:
tip = tip -1
s = d.split("")
label = s[1]
match s[0]:
case "RESULT":
if s[1] == "OK":
color = "#00FF00"
self.model[tip][1] = label
self.model[tip][2] = color
elif s[1] == "FAIL":
self.box.set_visible(True)
self.errors.set_visible(True)
color = "#FF0000"
self.model[tip][1] = label
self.model[tip][2] = color
self.early_cancel_box.set_visible(False)
self.errors_label.set_text(textwrap.fill(s[2], 100))
self.label2.set_text("Errors occurred: see details below.")
self.spinner.stop()
else:
color = "ORANGE"
self.model[tip][1] = label
self.model[tip][2] = color
case "STATUS":
self.model.append([label, "", "#FF0000"])
case "PROGRESS":
self.label2.set_text(s[1])
case _:
return
#FIXME: not scrolling down on failure msg
self.scroll_to_end();
while True:
with open(FIFO) as fifo:
newdata = fifo.read()
if len(newdata) == 0:
break
else:
d = newdata
if d.startswith("EXIT"):
#TODO: some GTK errors when exiting and launching UI
self.destroy()
os.remove(FIFO)
Gtk.main_quit()
break
else:
GLib.idle_add(update_gui)
def _on_button_clicked(self, button):
url="https://github.com/aclist/dztui/issues/new/choose"
subprocess.Popen(['/usr/bin/env', 'bash', "xdg-open", url])
class App(Gtk.Application):
def __init__(self):
GLib.set_prgname("DZGUI Loader")
self.win = OuterWindow()
GLib.set_prgname("DZGUI Loader")
Gtk.main()
def main():
App()
if __name__ == '__main__':
main()