Merge branch 'master' into dev
This commit is contained in:
@@ -1,5 +1,7 @@
|
|||||||
# Helldivers 2 Mod Manager CLI
|
# Helldivers 2 Mod Manager CLI
|
||||||
|
|
||||||
|
Helldivers 2 Mod Manager CLI is a command line interface for managing Helldivers 2 mods. Since there is no Linux mod manager available and I like being a nerd by using CLI tools instead of GUIs, this project was born.
|
||||||
|
|
||||||
- [Helldivers 2 Mod Manager CLI](#helldivers-2-mod-manager-cli)
|
- [Helldivers 2 Mod Manager CLI](#helldivers-2-mod-manager-cli)
|
||||||
- [Installation](#installation)
|
- [Installation](#installation)
|
||||||
- [Usage](#usage)
|
- [Usage](#usage)
|
||||||
@@ -9,6 +11,7 @@
|
|||||||
- [Uninstall a mod](#uninstall-a-mod)
|
- [Uninstall a mod](#uninstall-a-mod)
|
||||||
- [Enable/disable mods](#enabledisable-mods)
|
- [Enable/disable mods](#enabledisable-mods)
|
||||||
- [List installed mods](#list-installed-mods)
|
- [List installed mods](#list-installed-mods)
|
||||||
|
- [Compatibility](#compatibility)
|
||||||
- [Advanced usage](#advanced-usage)
|
- [Advanced usage](#advanced-usage)
|
||||||
- [Shortcuts](#shortcuts)
|
- [Shortcuts](#shortcuts)
|
||||||
- [Exporting and importing](#exporting-and-importing)
|
- [Exporting and importing](#exporting-and-importing)
|
||||||
@@ -17,14 +20,12 @@
|
|||||||
- [Contributing](#contributing)
|
- [Contributing](#contributing)
|
||||||
- [Planned features](#planned-features)
|
- [Planned features](#planned-features)
|
||||||
|
|
||||||
Helldivers 2 Mod Manager CLI is a command line interface for managing Helldivers 2 mods. Since there is no mod manager GUI for Helldivers 2 on Linux yet, this small script aims to provide a simple way to manage mods on Linux.
|
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
To install/update Helldivers 2 Mod Manager CLI run the following command in your terminal:
|
To install/update Helldivers 2 Mod Manager CLI run the following command in your terminal:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
sh -c "$(curl -fsSL https://raw.githubusercontent.com/v4n00/h2mm-cli/refs/heads/master/install.sh)"
|
bash -c "$(curl -fsSL https://raw.githubusercontent.com/v4n00/h2mm-cli/refs/heads/master/install.sh)"
|
||||||
```
|
```
|
||||||
|
|
||||||
Running this script will require sudo permissions. **DO NOT TRUST** random scripts from the internet. If you want to review the script before running it, check out the mod repository for yourself.
|
Running this script will require sudo permissions. **DO NOT TRUST** random scripts from the internet. If you want to review the script before running it, check out the mod repository for yourself.
|
||||||
@@ -42,8 +43,10 @@ h2mm
|
|||||||
- `install` - Install a mod with files
|
- `install` - Install a mod with files
|
||||||
- `uninstall` - Uninstall a mod by name
|
- `uninstall` - Uninstall a mod by name
|
||||||
- `list` - List all installed mods
|
- `list` - List all installed mods
|
||||||
- `export <zip_name>` - Export installed mods to a zip file
|
- `enable` - Enable a mod by name
|
||||||
- `import <zip_name>` - Import mods from a zip file
|
- `disable` - Disable a mod by name
|
||||||
|
- `export` - Export installed mods to a zip file
|
||||||
|
- `import` - Import mods from a zip file
|
||||||
- `reset` - Reset all installed mods
|
- `reset` - Reset all installed mods
|
||||||
- `help` - Display the help message
|
- `help` - Display the help message
|
||||||
|
|
||||||
@@ -67,7 +70,6 @@ h2mm uninstall "Example mod"
|
|||||||
h2mm uninstall -i 1 # uninstall mod with index 1
|
h2mm uninstall -i 1 # uninstall mod with index 1
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
#### Enable/disable mods
|
#### Enable/disable mods
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
@@ -83,6 +85,18 @@ h2mm disable -i 1 # disable mod with index 1
|
|||||||
h2mm list
|
h2mm list
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Compatibility
|
||||||
|
|
||||||
|
The script is developed and tested on Arch Linux, but it should work on other Linux distributions as well. If you encounter any issues, please open an issue on the repository.
|
||||||
|
|
||||||
|
Status of other platforms:
|
||||||
|
|
||||||
|
- Linux :white_check_mark:
|
||||||
|
- Steam Deck - untested (should work) :grey_question:
|
||||||
|
- WSL :white_check_mark:
|
||||||
|
|
||||||
|
> The script works on WSL, but you need to specify the path to the Helldivers 2 mods directory manually, to find your Windows partition head to `/mnt/` and from there go to your Helldivers 2 data directory, on a typical install it should be on `/mnt/c/Program\ Files\ \(x86\)/Steam/steamapps/common/Helldivers\ 2/data`. You also need to have `unzip` installed, which can be done by running `sudo apt install unzip`.
|
||||||
|
|
||||||
## Advanced usage
|
## Advanced usage
|
||||||
|
|
||||||
### Shortcuts
|
### Shortcuts
|
||||||
@@ -127,6 +141,7 @@ Feel free to contribute to this project by creating a pull request or opening an
|
|||||||
|
|
||||||
- [x] Check for mod updates
|
- [x] Check for mod updates
|
||||||
- [x] Enable/disable mods
|
- [x] Enable/disable mods
|
||||||
|
- [ ] Install mods in batches
|
||||||
- [ ] Easier way to change mod presets
|
- [ ] Easier way to change mod presets
|
||||||
- [ ] Find a way to make use of `manifest.json` and simplify installing variants
|
- [ ] Find a way to make use of `manifest.json` and simplify installing variants
|
||||||
- [x] [DEV] Change to `.tar.gz` for exporting and importing
|
- [x] [DEV] Change to `.tar.gz` for exporting and importing
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
VERSION="0.2.0"
|
VERSION="0.2.2"
|
||||||
|
|
||||||
# --- Globals ---
|
# --- Globals ---
|
||||||
|
|
||||||
@@ -73,8 +73,7 @@ function find_game_directory() {
|
|||||||
|
|
||||||
if [[ -z "$game_dir" ]]; then
|
if [[ -z "$game_dir" ]]; then
|
||||||
echo "Could not find the Helldivers 2 data directory automatically." >&2
|
echo "Could not find the Helldivers 2 data directory automatically." >&2
|
||||||
read -p "Please enter the path to the Helldivers 2 data directory: " game_dir
|
IFS= read -ep "Please enter the path to the Helldivers 2 data directory: " game_dir
|
||||||
game_dir=$(eval echo "$game_dir")
|
|
||||||
if [[ ! -d "$game_dir" ]]; then
|
if [[ ! -d "$game_dir" ]]; then
|
||||||
echo -e "${RED}Error${NC}: Provided path is not a valid directory." >&2
|
echo -e "${RED}Error${NC}: Provided path is not a valid directory." >&2
|
||||||
exit 1
|
exit 1
|
||||||
@@ -115,17 +114,19 @@ function display_help() {
|
|||||||
echo "Helldivers 2 Mod Manager v${VERSION}"
|
echo "Helldivers 2 Mod Manager v${VERSION}"
|
||||||
echo "Usage: h2mm [command] [options]"
|
echo "Usage: h2mm [command] [options]"
|
||||||
echo "Commands:"
|
echo "Commands:"
|
||||||
echo " install Install a mod with files (short form: h2mm i)."
|
echo " install Install a mod with files."
|
||||||
echo " uninstall Uninstall a mod by name (short form: h2mm u)."
|
echo " uninstall Uninstall a mod by name."
|
||||||
echo " list List all installed mods (short form: h2mm l)."
|
echo " list List all installed mods."
|
||||||
echo " export <zip_name> Export installed mods to a zip file (short form: h2mm ex)."
|
echo " enable Enable a mod by name."
|
||||||
echo " import <zip_name> Import mods from a zip file (short form: h2mm im)."
|
echo " disable Disable a mod by name."
|
||||||
echo " reset Reset all installed mods (short form: h2mm rr)."
|
echo " export <zip_name> Export installed mods to a zip file."
|
||||||
echo " help Display this help message (short form: h2mm h)."
|
echo " import <zip_name> Import mods from a zip file."
|
||||||
|
echo " reset Reset all installed mods."
|
||||||
|
echo " help Display this help message."
|
||||||
echo "For more information on usage, use h2mm [command] --help, available for install and uninstall."
|
echo "For more information on usage, use h2mm [command] --help, available for install and uninstall."
|
||||||
echo "Basic Usage:"
|
echo "Basic Usage:"
|
||||||
echo " h2mm install -z /path/to/mod.zip"
|
echo " h2mm install /path/to/mod.zip"
|
||||||
echo " h2mm install -d /path/to/mod/files"
|
echo " h2mm install /path/to/mod/files"
|
||||||
echo " h2mm uninstall \"Example mod\""
|
echo " h2mm uninstall \"Example mod\""
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -210,7 +211,7 @@ function display_import_help() {
|
|||||||
function check_for_updates() {
|
function check_for_updates() {
|
||||||
if [[ -f "$LAST_CHECKED_UPDATE_FILE" ]]; then
|
if [[ -f "$LAST_CHECKED_UPDATE_FILE" ]]; then
|
||||||
last_update=$(cat "$LAST_CHECKED_UPDATE_FILE")
|
last_update=$(cat "$LAST_CHECKED_UPDATE_FILE")
|
||||||
if [[ $(date +%Y-%m-%d) -gt $(date +%Y-%m-%d -d "$last_update + 7 days") ]]; then
|
if [[ $(date +%Y-%m-%d) -gt $(date +%Y-%m-%d -d "$last_update + 3 days") ]]; then
|
||||||
return
|
return
|
||||||
fi
|
fi
|
||||||
else
|
else
|
||||||
@@ -293,38 +294,38 @@ function mod_disable() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function mod_enable() {
|
function mod_enable() {
|
||||||
local mod_name=""
|
local mod_name=""
|
||||||
local mod_index=""
|
local mod_index=""
|
||||||
|
|
||||||
[[ $# -eq 0 ]] && { display_enable_help; exit 0; }
|
[[ $# -eq 0 ]] && { display_enable_help; exit 0; }
|
||||||
|
|
||||||
# parse arguments
|
# parse arguments
|
||||||
while [[ $# -gt 0 ]]; do
|
while [[ $# -gt 0 ]]; do
|
||||||
case "$1" in
|
case "$1" in
|
||||||
-i)
|
-i)
|
||||||
mod_index="$2"; shift 2
|
mod_index="$2"; shift 2
|
||||||
;;
|
;;
|
||||||
--help|-h)
|
--help|-h)
|
||||||
display_enable_help; exit 0
|
display_enable_help; exit 0
|
||||||
;;
|
;;
|
||||||
*)
|
*)
|
||||||
mod_name="$1"; shift 1
|
mod_name="$1"; shift 1
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
done
|
done
|
||||||
|
|
||||||
[[ -z "$mod_name" && -z "$mod_index" ]] && { echo -e "${RED}Error${NC}: Mod name or index is required to enable." >&2; exit 1; }
|
[[ -z "$mod_name" && -z "$mod_index" ]] && { echo -e "${RED}Error${NC}: Mod name or index is required to enable." >&2; exit 1; }
|
||||||
|
|
||||||
# find mod files
|
# find mod files
|
||||||
get_mod_name_and_index "$mod_name" "$mod_index"
|
get_mod_name_and_index "$mod_name" "$mod_index"
|
||||||
|
|
||||||
[[ "$status" == "ENABLED" ]] && { echo -e "${RED}Error${NC}: Mod $mod_name is already enabled." >&2; exit 1; }
|
[[ "$status" == "ENABLED" ]] && { echo -e "${RED}Error${NC}: Mod $mod_name is already enabled." >&2; exit 1; }
|
||||||
|
|
||||||
files=$(get_files_by_entry_from_db "$entry")
|
files=$(get_files_by_entry_from_db "$entry")
|
||||||
|
|
||||||
# enable each mod file by removing disabled_ from the start of the filename
|
# enable each mod file by removing disabled_ from the start of the filename
|
||||||
for file in $files; do
|
for file in $files; do
|
||||||
disabled_file="disabled_$file"
|
disabled_file="disabled_$file"
|
||||||
|
|
||||||
# check if the files exists
|
# check if the files exists
|
||||||
[[ -f "$MODS_DIR/$disabled_file" ]] || { echo -e "${RED}Error${NC}: Mod file $file does not exist." >&2; exit 1; }
|
[[ -f "$MODS_DIR/$disabled_file" ]] || { echo -e "${RED}Error${NC}: Mod file $file does not exist." >&2; exit 1; }
|
||||||
@@ -334,17 +335,17 @@ function mod_enable() {
|
|||||||
# check if the file was moved successfully
|
# check if the file was moved successfully
|
||||||
[[ $? -ne 0 ]] && { echo -e "${RED}Error${NC}: Could not enable mod file $disabled_file." >&2; exit 1; }
|
[[ $? -ne 0 ]] && { echo -e "${RED}Error${NC}: Could not enable mod file $disabled_file." >&2; exit 1; }
|
||||||
echo -e "Enabled ${ORANGE}$disabled_file${NC} (changed to ${GREEN}\$MODS_DIR/$file${NC})." >&2
|
echo -e "Enabled ${ORANGE}$disabled_file${NC} (changed to ${GREEN}\$MODS_DIR/$file${NC})." >&2
|
||||||
done
|
done
|
||||||
|
|
||||||
# update the database
|
# update the database
|
||||||
sed -i "/^$mod_index,/s/DISABLED/ENABLED/" "$DB_FILE"
|
sed -i "/^$mod_index,/s/DISABLED/ENABLED/" "$DB_FILE"
|
||||||
|
|
||||||
if [[ $? -eq 0 ]]; then
|
if [[ $? -eq 0 ]]; then
|
||||||
echo -e "Mod $mod_name ${GREEN}enabled${NC} successfully." >&2
|
echo -e "Mod $mod_name ${GREEN}enabled${NC} successfully." >&2
|
||||||
else
|
else
|
||||||
echo -e "${RED}Error${NC}: Failed to enable mod." >&2
|
echo -e "${RED}Error${NC}: Failed to enable mod." >&2
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
function mod_reset() {
|
function mod_reset() {
|
||||||
@@ -375,23 +376,23 @@ function mod_install() {
|
|||||||
# parse arguments
|
# parse arguments
|
||||||
while [[ $# -gt 0 ]]; do
|
while [[ $# -gt 0 ]]; do
|
||||||
case "$1" in
|
case "$1" in
|
||||||
-n)
|
-n)
|
||||||
mod_name="$2"; shift 2
|
mod_name="$2"; shift 2
|
||||||
;;
|
;;
|
||||||
--help|-h)
|
--help|-h)
|
||||||
display_install_help; exit 0
|
display_install_help; exit 0
|
||||||
;;
|
;;
|
||||||
*)
|
*)
|
||||||
if [[ -f "$1" && "$1" == *.zip ]]; then
|
if [[ -f "$1" && "$1" == *.zip ]]; then
|
||||||
mod_zip="$1"
|
mod_zip="$1"
|
||||||
elif [[ -d "$1" ]]; then
|
elif [[ -d "$1" ]]; then
|
||||||
mod_dir="$1"
|
mod_dir="$1"
|
||||||
else
|
else
|
||||||
mod_files+=("$1")
|
mod_files+=("$1")
|
||||||
fi
|
fi
|
||||||
shift
|
shift
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
done
|
done
|
||||||
|
|
||||||
# zip file containing mod files
|
# zip file containing mod files
|
||||||
@@ -561,17 +562,16 @@ function mod_list() {
|
|||||||
echo "Installed mods:" >&2
|
echo "Installed mods:" >&2
|
||||||
|
|
||||||
awk -v GREEN="$GREEN" -v RED="$RED" -v NC="$NC" -F, '{
|
awk -v GREEN="$GREEN" -v RED="$RED" -v NC="$NC" -F, '{
|
||||||
color = ($2 == "DISABLED") ? RED : GREEN;
|
color = ($2 == "DISABLED") ? RED : GREEN;
|
||||||
if (length($3) > 150) $3 = substr($3, 1, 147) "...";
|
if (length($4) > 150) $4 = substr($4, 1, 147) "...";
|
||||||
printf "%2s. [%s%s%s] %s (%s)\n", $1, color, $2, NC, $3, $4}' "$DB_FILE"
|
printf "%2s. [%s%s%s] %s (%s)\n", $1, color, $2, NC, $3, $4}' "$DB_FILE"
|
||||||
}
|
}
|
||||||
|
|
||||||
function mod_export() {
|
function mod_export() {
|
||||||
[[ "$1" == "--help" || "$1" == "-h" ]] && { display_export_help; exit 0; }
|
[[ "$1" == "--help" || "$1" == "-h" ]] && { display_export_help; exit 0; }
|
||||||
|
|
||||||
echo -ne "Archive file will be saved in the current directory ($(pwd)). Continue? (Y/n): "
|
echo -ne "Archive file will be saved in the current directory ($(pwd)). Continue? (Y/n): "
|
||||||
read -r confirm
|
read -r confirm
|
||||||
|
|
||||||
if [[ "$confirm" == "y" || "$confirm" == "Y" || "$confirm" = "" ]]; then
|
if [[ "$confirm" == "y" || "$confirm" == "Y" || "$confirm" = "" ]]; then
|
||||||
# create a temporary directory to store the mods
|
# create a temporary directory to store the mods
|
||||||
OUT_DIR=$(mktemp -d)
|
OUT_DIR=$(mktemp -d)
|
||||||
|
|||||||
+22
-18
@@ -34,7 +34,7 @@ breaking_changes_patches=(
|
|||||||
["2"]='sed -i "s/^\([0-9]\+\),/\1,ENABLED,/" "$1/mods.csv"'
|
["2"]='sed -i "s/^\([0-9]\+\),/\1,ENABLED,/" "$1/mods.csv"'
|
||||||
)
|
)
|
||||||
|
|
||||||
# Script
|
# Handle breaking changes
|
||||||
|
|
||||||
if [[ -x "$(command -v $SCRIPT_NAME)" ]]; then
|
if [[ -x "$(command -v $SCRIPT_NAME)" ]]; then
|
||||||
installed_version=$($SCRIPT_NAME --version)
|
installed_version=$($SCRIPT_NAME --version)
|
||||||
@@ -63,46 +63,50 @@ if [[ -x "$(command -v $SCRIPT_NAME)" ]]; then
|
|||||||
# find hd2 path
|
# find hd2 path
|
||||||
search_dir="${HOME}"
|
search_dir="${HOME}"
|
||||||
target_dir="Steam/steamapps/common/Helldivers\ 2/data"
|
target_dir="Steam/steamapps/common/Helldivers\ 2/data"
|
||||||
echo "Searching for the Helldivers 2 data directory..." >&2
|
echo "Searching for the Helldivers 2 data directory... (20 seconds timeout)" >&2
|
||||||
game_dir=$(find "$search_dir" -type d -path "*/$target_dir" 2>/dev/null | head -n 1)
|
|
||||||
if [[ -z "$game_dir" ]]; then
|
|
||||||
echo "Could not find the Helldivers 2 data directory automatically." >&2
|
|
||||||
read -p "Please enter the path to the Helldivers 2 data directory: " game_dir
|
|
||||||
game_dir=$(eval echo "$game_dir")
|
|
||||||
|
|
||||||
if [[ ! -d "$game_dir" ]]; then
|
game_dir=$(timeout 20 find "$search_dir" -type d -path "*/$target_dir" 2>/dev/null | head -n 1)
|
||||||
echo -e "${RED}Error${NC}: Provided path is not a valid directory." >&2
|
if [[ -z "$game_dir" ]]; then
|
||||||
exit 1
|
echo "Could not find the Helldivers 2 data directory automatically." >&2
|
||||||
fi
|
IFS= read -ep "Please enter the path to the Helldivers 2 data directory: " game_dir
|
||||||
fi
|
if [[ ! -d "$game_dir" ]]; then
|
||||||
|
echo -e "${RED}Error${NC}: Provided path is not a valid directory." >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
[[ ! -f "$game_dir/mods.csv" ]] && { echo -e "${RED}Error:${NC} mods.csv not found in $game_dir."; exit 1; }
|
[[ ! -f "$game_dir/mods.csv" ]] && { echo -e "${RED}Error:${NC} mods.csv not found in $game_dir."; exit 1; }
|
||||||
|
|
||||||
|
# make backup of mods in case something goes wrong
|
||||||
|
echo "Creating a backup of mods.csv."
|
||||||
|
h2mm export
|
||||||
|
|
||||||
# iterate from installed major number to latest major number
|
# iterate from installed major number to latest major number
|
||||||
for ((i = installed_major + 1; i <= latest_major; i++)); do
|
for ((i = installed_major + 1; i <= latest_major; i++)); do
|
||||||
echo -e "${RED}[ ]${NC} Applying breaking changes patch for version $i."
|
echo -e "Applying breaking changes patch for version $i."
|
||||||
|
|
||||||
[[ -n "${breaking_changes_patches[$i]}" ]] && eval $(echo "${breaking_changes_patches[$i]}" | sed "s:\$1:$game_dir:")
|
[[ -n "${breaking_changes_patches[$i]}" ]] && eval $(echo "${breaking_changes_patches[$i]}" | sed "s:\$1:$game_dir:")
|
||||||
if [[ $? -ne 0 ]]; then
|
if [[ $? -ne 0 ]]; then
|
||||||
echo -ne "${RED}Error:${NC} Failed to apply breaking changes patch for version $i. Do you want to continue? (Y/n): "
|
echo -ne "${RED}Error:${NC} Failed to apply breaking changes patch for version $i. Do you want to continue? (Y/n): "
|
||||||
read -r response
|
read -er response
|
||||||
|
|
||||||
[[ "$response" != "y" && "$response" != "Y" && -n "$response" ]] && { echo "Exiting. Uninstall the script first the retry the install script."; exit 1; }
|
[[ "$response" != "y" && "$response" != "Y" && -n "$response" ]] && { echo "Exiting. Uninstall the script first the retry the install script."; exit 1; }
|
||||||
else
|
else
|
||||||
echo -e "${GREEN}[X]${NC} Breaking changes patch for version $i applied successfully."
|
echo -e "Breaking changes patch for version ${ORANGE}$i${NC} applied ${GREEN}successfully${NC}."
|
||||||
fi
|
fi
|
||||||
done
|
done
|
||||||
fi
|
fi
|
||||||
|
echo
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Install
|
# Install
|
||||||
|
|
||||||
read -p "Install the script to $DESTINATION_PATH or specify another path (must be included in \$PATH)? (Y/path): " response
|
IFS= read -ep "Install the script to $DESTINATION_PATH or specify another path (must be included in \$PATH)? (Y/path): " response
|
||||||
|
|
||||||
if [[ "$response" != "y" && "$response" != "Y" && -n "$response" ]]; then
|
if [[ "$response" != "y" && "$response" != "Y" && -n "$response" ]]; then
|
||||||
DESTINATION_PATH=$(eval echo "$response")
|
DESTINATION_PATH="$response"
|
||||||
if [[ ! -d "$DESTINATION_PATH" ]]; then
|
if [[ ! -d "$DESTINATION_PATH" ]]; then
|
||||||
echo -e "${RED}Error:${NC} Path $DESTINATION_PATH does not exist. Exiting..."
|
echo -e "${RED}Error:${NC} Path $DESTINATION_PATH does not exist."
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|||||||
Reference in New Issue
Block a user