2022-05-19 08:00:04 +08:00
|
|
|
#!/usr/bin/env bash
|
|
|
|
|
2023-03-12 13:49:59 +08:00
|
|
|
# Grasscutter install script for GNU/Linux
|
2022-05-19 08:00:04 +08:00
|
|
|
# Made by TurtleIdiot
|
|
|
|
|
|
|
|
# Stops the installer if any command has a non-zero exit status
|
|
|
|
set -e
|
|
|
|
|
|
|
|
# Checks for root
|
|
|
|
if [ $EUID != 0 ]; then
|
|
|
|
echo "Please run the installer as root!"
|
|
|
|
exit
|
|
|
|
fi
|
|
|
|
|
|
|
|
is_command() {
|
|
|
|
# Checks if a given command is available
|
|
|
|
local check_command="$1"
|
|
|
|
command -v "${check_command}" > /dev/null 2>&1
|
|
|
|
}
|
|
|
|
|
|
|
|
# IP validation
|
|
|
|
valid_ip() {
|
|
|
|
local ip=$1
|
|
|
|
local stat=1
|
|
|
|
|
|
|
|
if [[ $ip =~ ^[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}$ ]]; then
|
|
|
|
OIFS=$IFS
|
|
|
|
IFS="."
|
|
|
|
ip=($ip)
|
|
|
|
IFS=$OIFS
|
|
|
|
[[ ${ip[0]} -le 255 && ${ip[1]} -le 255 \
|
|
|
|
&& ${ip[2]} -le 255 && ${ip[3]} -le 255 ]]
|
|
|
|
stat=$?
|
|
|
|
fi
|
|
|
|
return $stat
|
|
|
|
}
|
|
|
|
|
2022-05-24 05:16:30 +08:00
|
|
|
# Checks for supported installer(s) (only apt-get and pacman right now, might add more in the future)
|
2022-05-19 08:00:04 +08:00
|
|
|
if is_command apt-get ; then
|
2022-05-24 05:16:30 +08:00
|
|
|
echo -e "Supported package manager found (apt-get)\n"
|
|
|
|
|
|
|
|
GC_DEPS="mongodb openjdk-17-jre"
|
|
|
|
INSTALLER_DEPS="wget openssl unzip git"
|
|
|
|
SYSTEM="deb" # Debian-based (debian, ubuntu)
|
|
|
|
elif is_command pacman ; then
|
|
|
|
echo -e "supported package manager found (pacman)\n"
|
|
|
|
|
|
|
|
GC_DEPS="jre17-openjdk"
|
|
|
|
INSTALLER_DEPS="curl wget openssl unzip git base-devel" # curl is still a dependency here in order to successfully build mongodb
|
|
|
|
SYSTEM="arch" # Arch for the elitists :P
|
2022-05-19 08:00:04 +08:00
|
|
|
else
|
|
|
|
echo "No supported package manager found"
|
|
|
|
exit
|
|
|
|
fi
|
|
|
|
|
2022-05-23 21:40:35 +08:00
|
|
|
BRANCH="stable" # Stable by default
|
|
|
|
# Allows choice between stable and dev branch
|
|
|
|
echo "Please select the branch you wish to install"
|
2022-05-24 05:16:30 +08:00
|
|
|
echo -e "!!NOTE!!: stable is the recommended branch.\nDo *NOT* use development unless you have a reason to and know what you're doing"
|
2022-05-23 21:40:35 +08:00
|
|
|
select branch in "stable" "development" ; do
|
2022-05-24 05:16:30 +08:00
|
|
|
case $branch in
|
|
|
|
stable )
|
|
|
|
BRANCH="stable"
|
|
|
|
break;;
|
|
|
|
development )
|
|
|
|
BRANCH="development"
|
|
|
|
break;;
|
|
|
|
esac
|
2022-05-23 21:40:35 +08:00
|
|
|
done
|
|
|
|
|
2022-05-19 08:00:04 +08:00
|
|
|
echo "The following packages will have to be installed in order to INSTALL grasscutter:"
|
|
|
|
echo -e "$INSTALLER_DEPS \n"
|
|
|
|
echo "The following packages will have to be installed to RUN grasscutter:"
|
|
|
|
echo -e "$GC_DEPS \n"
|
|
|
|
|
|
|
|
echo "Do you wish to proceed and install grasscutter?"
|
|
|
|
select yn in "Yes" "No" ; do
|
|
|
|
case $yn in
|
|
|
|
Yes ) break;;
|
|
|
|
No ) exit;;
|
|
|
|
esac
|
|
|
|
done
|
|
|
|
|
|
|
|
echo "Updating package cache..."
|
2022-05-24 05:16:30 +08:00
|
|
|
case $SYSTEM in # More concise than if
|
|
|
|
deb ) apt-get update -qq;;
|
|
|
|
arch ) pacman -Syy;;
|
|
|
|
esac
|
2022-05-19 08:00:04 +08:00
|
|
|
|
|
|
|
# Starts installing dependencies
|
|
|
|
echo "Installing setup dependencies..."
|
2022-05-24 05:16:30 +08:00
|
|
|
case $SYSTEM in # These are one-liners anyways
|
|
|
|
deb ) apt-get -qq install $INSTALLER_DEPS -y;;
|
|
|
|
arch ) pacman -Sq --noconfirm --needed $INSTALLER_DEPS > /dev/null;;
|
|
|
|
esac
|
2022-05-19 08:00:04 +08:00
|
|
|
echo "Done"
|
|
|
|
|
|
|
|
echo "Installing grasscutter dependencies..."
|
2022-05-24 05:16:30 +08:00
|
|
|
case $SYSTEM in
|
|
|
|
deb) apt-get -qq install $GC_DEPS -y > /dev/null;;
|
|
|
|
arch ) pacman -Sq --noconfirm --needed $GC_DEPS > /dev/null;;
|
|
|
|
esac
|
|
|
|
# *sighs* here we go...
|
|
|
|
INST_ARCH_MONGO="no"
|
|
|
|
if [ $SYSTEM = "arch" ]; then
|
|
|
|
echo -e "-=-=-=-=-=--- !! IMPORTANT !! ---=-=-=-=-=-\n"
|
|
|
|
echo -e " Due to licensing issues with mongodb,\n it is no longer available on the official arch repositiries."
|
|
|
|
echo -e " In order to install mongodb,\n it needs to be fetched from the Arch User Repository.\n"
|
|
|
|
echo -e " As this script is running as root,\n a temporary user will need to be created to run makepkg."
|
|
|
|
echo -e " The temporary user will be deleted once\n makepkg has finished.\n"
|
|
|
|
echo -e " This will be handled automatically.\n"
|
|
|
|
echo -e "-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-\n"
|
|
|
|
echo -e "!!NOTE!!: Only select \"Skip\" if mongodb is already installed on this system"
|
|
|
|
echo "Do you want to continue?"
|
|
|
|
select yn in "Yes" "Skip" "No" ; do
|
|
|
|
case $yn in
|
|
|
|
Yes )
|
|
|
|
INST_ARCH_MONGO="yes"
|
|
|
|
break;;
|
|
|
|
No ) exit;;
|
|
|
|
Skip )
|
|
|
|
INST_ARCH_MONGO="no"
|
|
|
|
break;;
|
|
|
|
esac
|
|
|
|
done
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ $INST_ARCH_MONGO = "yes" ]; then
|
|
|
|
DIR=$(pwd)
|
|
|
|
# Make temp user
|
|
|
|
echo "Creating temporary user..."
|
|
|
|
TEMPUSER="gctempuser"
|
|
|
|
TEMPHOME="/home/$TEMPUSER"
|
|
|
|
useradd -m $TEMPUSER
|
|
|
|
cd $TEMPHOME
|
|
|
|
|
|
|
|
# Do the actual makepkg shenanigans
|
|
|
|
echo "Building mongodb... (this will take a moment)"
|
|
|
|
su $TEMPUSER<<EOF
|
|
|
|
mkdir temp
|
|
|
|
cd temp
|
|
|
|
git clone https://aur.archlinux.org/mongodb-bin.git -q
|
|
|
|
cd mongodb-bin
|
|
|
|
makepkg -s > /dev/null
|
|
|
|
exit
|
|
|
|
EOF
|
|
|
|
mv "$(find -name "mongodb-bin*.pkg.tar.zst" -type f)" ./mongodb-bin.pkg.tar.zst
|
|
|
|
cd $DIR
|
|
|
|
|
|
|
|
# Snatch the file to current working directory
|
|
|
|
mv "$TEMPHOME/mongodb-bin.pkg.tar.zst" ./mongodb-bin.pkg.tar.zst
|
|
|
|
chown root ./mongodb-bin.pkg.tar.zst
|
|
|
|
chgrp root ./mongodb-bin.pkg.tar.zst
|
|
|
|
chmod 775 ./mongodb-bin.pkg.tar.zst
|
|
|
|
|
|
|
|
echo "Installing mongodb..."
|
|
|
|
pacman -U mongodb-bin.pkg.tar.zst --noconfirm > /dev/null
|
|
|
|
rm mongodb-bin.pkg.tar.zst
|
|
|
|
|
|
|
|
echo "Starting mongodb..."
|
|
|
|
systemctl enable mongodb
|
|
|
|
systemctl start mongodb
|
|
|
|
|
|
|
|
echo "Removing temporary account..."
|
|
|
|
userdel -r $TEMPUSER
|
|
|
|
fi
|
2022-05-19 08:00:04 +08:00
|
|
|
echo "Done"
|
|
|
|
|
|
|
|
echo "Getting grasscutter..."
|
|
|
|
|
|
|
|
# Download and rename jar
|
2022-05-23 21:40:35 +08:00
|
|
|
wget -q --show-progress "https://nightly.link/Grasscutters/Grasscutter/workflows/build/$BRANCH/Grasscutter.zip"
|
2022-05-24 05:16:30 +08:00
|
|
|
echo "unzipping"
|
2022-05-19 08:00:04 +08:00
|
|
|
unzip -qq Grasscutter.zip
|
|
|
|
mv $(find -name "grasscutter*.jar" -type f) grasscutter.jar
|
|
|
|
|
|
|
|
# Download resources
|
|
|
|
echo "Downloading resources... (this will take a moment)"
|
|
|
|
wget -q --show-progress https://github.com/Koko-boya/Grasscutter_Resources/archive/refs/heads/main.zip -O resources.zip
|
|
|
|
echo "Extracting..."
|
|
|
|
unzip -qq resources.zip
|
|
|
|
mv ./Grasscutter_Resources-main/Resources ./resources
|
|
|
|
|
|
|
|
# Here we do a sparse checkout to only pull /data and /keys
|
|
|
|
echo "Downloading keys and data..."
|
|
|
|
mkdir repo
|
|
|
|
cd repo
|
|
|
|
git init -q
|
|
|
|
git remote add origin https://github.com/Grasscutters/Grasscutter.git
|
|
|
|
git fetch -q
|
|
|
|
git config core.sparseCheckout true
|
|
|
|
echo "data/" >> .git/info/sparse-checkout
|
|
|
|
echo "keys/" >> .git/info/sparse-checkout
|
|
|
|
git pull origin stable -q
|
|
|
|
cd ../
|
|
|
|
mv ./repo/data ./data
|
|
|
|
mv ./repo/keys ./keys
|
|
|
|
|
|
|
|
# Generate handbook/config
|
2022-05-19 08:28:24 +08:00
|
|
|
echo "Please enter language when *NEXT* prompted (press enter/return to continue to language select)"
|
2022-05-19 08:00:04 +08:00
|
|
|
read
|
|
|
|
java -jar grasscutter.jar -handbook
|
|
|
|
|
|
|
|
# Prompt IP address for config.json and for generating new keystore.p12 file
|
|
|
|
echo "Please enter the IP address that will be used to connect to the server"
|
|
|
|
echo "This can be a local or a public IP address"
|
|
|
|
echo "This IP address will be used to generate SSL certificates so it is important it is correct"
|
|
|
|
|
|
|
|
while : ; do
|
|
|
|
read -p "Enter IP: " SERVER_IP
|
|
|
|
if valid_ip $SERVER_IP; then
|
|
|
|
break;
|
|
|
|
else
|
|
|
|
echo "Invalid IP address. Try again."
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
|
|
|
|
# Replaces "127.0.0.1" with given IP
|
|
|
|
sed -i "s/127.0.0.1/$SERVER_IP/g" config.json
|
|
|
|
|
|
|
|
# Generates new keystore.p12 with the server's IP address
|
|
|
|
# This is done to prevent a "Connection Timed Out" error from appearing
|
|
|
|
# after clicking to enter the door in the main menu/title screen
|
|
|
|
# This issue only exists when connecting to a server *other* than localhost
|
|
|
|
# since the default keystore.p12 has only been made for localhost
|
|
|
|
|
|
|
|
mkdir certs
|
|
|
|
cd certs
|
|
|
|
echo "Generating CA key and certificate pair..."
|
|
|
|
openssl req -x509 -nodes -days 25202 -newkey rsa:2048 -subj "/C=GB/ST=Essex/L=London/O=Grasscutters/OU=Grasscutters/CN=$SERVER_IP" -keyout CAkey.key -out CAcert.crt
|
|
|
|
echo "Generating SSL key and certificate pair..."
|
|
|
|
|
|
|
|
openssl genpkey -out ssl.key -algorithm rsa
|
|
|
|
|
|
|
|
# Creates a conf file in order to generate a csr
|
|
|
|
cat > csr.conf <<EOF
|
|
|
|
[ req ]
|
|
|
|
default_bits = 2048
|
|
|
|
prompt = no
|
|
|
|
default_md = sha256
|
|
|
|
req_extensions = req_ext
|
|
|
|
distinguished_name = dn
|
|
|
|
|
|
|
|
[ dn ]
|
|
|
|
C = GB
|
|
|
|
ST = Essex
|
|
|
|
L = London
|
|
|
|
O = Grasscutters
|
|
|
|
OU = Grasscutters
|
|
|
|
CN = $SERVER_IP
|
|
|
|
|
|
|
|
[ req_ext ]
|
|
|
|
subjectAltName = @alt_names
|
|
|
|
|
|
|
|
[ alt_names ]
|
|
|
|
IP.1 = $SERVER_IP
|
|
|
|
EOF
|
|
|
|
|
|
|
|
# Creates csr using key and conf
|
|
|
|
openssl req -new -key ssl.key -out ssl.csr -config csr.conf
|
|
|
|
|
|
|
|
# Creates conf to finalise creation of certificate
|
|
|
|
cat > cert.conf <<EOF
|
|
|
|
|
|
|
|
authorityKeyIdentifier=keyid,issuer
|
|
|
|
basicConstraints=CA:FALSE
|
|
|
|
keyUsage = digitalSignature, nonRepudiation, keyEncipherment, keyAgreement, dataEncipherment
|
|
|
|
subjectAltName = @alt_names
|
|
|
|
|
|
|
|
[alt_names]
|
|
|
|
IP.1 = $SERVER_IP
|
|
|
|
|
|
|
|
EOF
|
|
|
|
|
|
|
|
# Creates ssl cert
|
|
|
|
openssl x509 -req -in ssl.csr -CA CAcert.crt -CAkey CAkey.key -CAcreateserial -out ssl.crt -days 25202 -sha256 -extfile cert.conf
|
|
|
|
|
|
|
|
echo "Generating keystore.p12 from key and certificate..."
|
|
|
|
openssl pkcs12 -export -out keystore.p12 -inkey ssl.key -in ssl.crt -certfile CAcert.crt -passout pass:123456
|
|
|
|
|
|
|
|
cd ../
|
|
|
|
mv ./certs/keystore.p12 ./keystore.p12
|
|
|
|
echo "Done"
|
|
|
|
|
2022-05-19 08:28:24 +08:00
|
|
|
echo -e "Asking Noelle to clean up...\n"
|
2022-05-19 08:00:04 +08:00
|
|
|
rm -rf Grasscutter.zip resources.zip ./certs ./Grasscutter_Resources-main ./repo
|
2022-05-19 08:28:24 +08:00
|
|
|
echo -e "All done!\n"
|
2022-05-19 08:00:04 +08:00
|
|
|
echo -e "You can now uninstall the following packages if you wish:\n$INSTALLER_DEPS"
|
2022-05-19 08:28:24 +08:00
|
|
|
echo -e "-=-=-=-=-=--- !! IMPORTANT !! ---=-=-=-=-=-\n"
|
|
|
|
echo "Please make sure that ports 443 and 22102 are OPEN (both tcp and udp)"
|
2022-05-19 08:00:04 +08:00
|
|
|
echo -e "In order to run the server, run the following command:\nsudo java -jar grasscutter.jar"
|
|
|
|
echo "You must run it using sudo as port 443 is a privileged port"
|
|
|
|
echo "To play, use the IP you provided earlier ($SERVER_IP) via GrassClipper or Fiddler"
|
|
|
|
|
|
|
|
exit
|