관련 링크
서버 관리
팰월드 멀티 서버의 경우 아직 불안정한 단계이기에 사용자들의 데이터가 모종의 이유(?)로 날아가면 난감하기에 주기적으로 백업을 해주어야 한다.
백업하는 과정을 스크립트로 작성하고 마무리로 텔레그램으로 알림까지 보내는 간단한 백업용 알림 스크립트등
효과적으로 관리할 수 있는 스크립트들을 작성해보자.
팰월드 세이브 데이터 경로
Palworld의 세이브 데이터는 아래와 같은 경로에 존재한다.
ls -lh game/Pal/Saved/SaveGames/0/4D308CFCDF584AFCAB7E66F8BB5A881D/Players/
total 44K
-rw-r--r-- 1 master master 2.7K Jan 26 00:58 04637D7E000000000000000000000000.sav
-rw-r--r-- 1 master master 3.3K Jan 26 14:02 26BF7B8D000000000000000000000000.sav
-rw-r--r-- 1 master master 2.6K Jan 26 00:58 2EA7D45B000000000000000000000000.sav
-rw-r--r-- 1 master master 3.8K Jan 26 14:02 48FFF339000000000000000000000000.sav
-rw-r--r-- 1 master master 2.4K Jan 26 00:58 62E1ACE8000000000000000000000000.sav
-rw-r--r-- 1 master master 2.8K Jan 26 12:58 7FB221FA000000000000000000000000.sav
-rw-r--r-- 1 master master 3.5K Jan 26 00:58 923FD405000000000000000000000000.sav
-rw-r--r-- 1 master master 3.2K Jan 26 14:02 A15D2445000000000000000000000000.sav
-rw-r--r-- 1 master master 2.6K Jan 26 00:58 A4ABAE08000000000000000000000000.sav
-rw-r--r-- 1 master master 2.6K Jan 26 00:58 C01C1509000000000000000000000000.sav
-rw-r--r-- 1 master master 2.5K Jan 26 00:58 C57E1CD9000000000000000000000000.sav
각 플레이어들의 정보가 저장되어 있지만 누구의 세이브 데이터인지 알 수가 없게 되어있다.
유저 데이터를 쉽게 찾고 싶다면 아래 게시글을 참고하도록 하자
유저 데이터 찾기
스크립트 작성
이제부터 스크립트들을 작성해보자.
▪️ 백업 스크립트
- 유저 데이터 백업 스크립트
~/backup/script.sh
#!/usr/bin/bash
HOST='master'
SERVER_NAME="palworld-server"
if $(docker ps | grep $SERVER_NAME >/dev/null 2>&1); then
{% raw %} if [[ $(docker inspect --format '{{.State.Status }}' "$SERVER_NAME") == 'running' ]]; {% endraw %} then
echo "$SERVER_NAME is running"
# 도커 프로세스에 해당 서버가 있는 경우 & 서버가 실행 중이지 않은 경우
else
echo "$SERVER_NAME is paused"
exit 1
fi
else
echo "$SERVER_NAME is stoped"
exit 1
fi
# 압축할 디렉토리
source_dir="/home/${HOST}/palworld/game/Pal/Saved/SaveGames"
# 압축 파일이 저장될 디렉토리
backup_dir="/tmp/palworld/backup"
# 압축 파일 이름
DATE="$(date +'%Y%m%d_%H%M')"
backup_filename="palworld_backup_${DATE}.tar.gz"
# 백업 디렉토리 없는 경우 생성
if [[ ! -d "${backup_dir}" ]]; then
mkdir -p "${backup_dir}"
fi
# 로그 파일 생성
LOG_FILE_NM="/tmp/backup.log"
touch "${LOG_FILE_NM}"
{
echo -e "\n💿 PalWorld 백업 시작 ... 💿"
# 디렉토리를 gzip 압축
tar -czf "${backup_dir}/${backup_filename}" "${source_dir}" 1>/dev/null 2>&1
NAME=$(ls -lh "${backup_dir}/${backup_filename}" | awk '{print $9}')
SIZE=$(ls -lh "${backup_dir}/${backup_filename}" | awk '{print $5}')
echo "==== 백업 파일 정보 ===="
echo "파일명 : ${NAME}"
echo "파일 크기 : ${SIZE}"
echo
echo "==== 이전 백업파일 삭제 ===="
find "${backup_dir}" -type f -mmin +360 -exec sh -c "ls -l {}; rm -f {}" \;
echo -e "\n💿 PalWorld 백업 종료 ... 💿"
} > "${LOG_FILE_NM}"
# 텔레그램으로 알림 전송
TELEGRAM_FILE="/home/${HOST}/telegram/alarm.sh"
"${TELEGRAM_FILE}" "Master Server" "$(cat "${LOG_FILE_NM}")"
# 로그파일 삭제
rm -rf "${LOG_FILE_NM}"
리소스 체크 스크립트
vim resource.sh
~/resource_check/script.sh
#!/usr/bin/bash
HOST='master'
TELEGRAM_FILE="/home/${HOST}/telegram/alarm.sh"
SWP_MEMORY=$(free -h | grep 'Swap' | awk '{print $4}')
MEMORY=$(free -h | grep 'Mem' | awk '{print $4}')
LOAD_AVERAGE=$(uptime | awk -F 'load average: ' '{print $2}')
MESSAGE="
MEMORY _> ${MEMORY}
SWAP MEMORY -> ${SWP_MEMORY}
uptime -> ${LOAD_AVERAGE}"
"${TELEGRAM_FILE}" "${HOST}" "${MESSAGE}"
알림 스크립트
- 텔레그램 알림 스크립트
~/telegram/alarm.sh
#!/usr/bin/bash
if [[ ${#} -ne 2 ]]; then
clear
echo -e "Usage 👉 ${0} hostname message"
exit 1
fi
HOST='master'
TELE_INFO_FILE="/home/${HOST}/telegram/token.env"
TELE_INFOS=($(awk -F '=' '{print $2}' "${TELE_INFO_FILE}"))
TOKEN="${TELE_INFOS[0]}"
CHAT_ID="${TELE_INFOS[1]}"
URL="https://api.telegram.org/bot${TOKEN}/sendMessage"
TEXT="[${1}] ${2}"
curl -s -d "chat_id=${CHAT_ID}&text=${TEXT}" ${URL} > /dev/null
재부팅 스크립트
- 메모리 점유율 80% 이상 시 서버 재부팅
bc 패키지 설치
sudo apt-get install -y bc
~/palworld/restart.sh
#!/usr/bin/bash
HOST='master'
YAML_FILE="/home/$HOST/palworld/docker-compose.yaml"
CONTAINER_NAME="palworld-server"
THRESHOLD=80
# 변수에 각각 삽입
read total used <<< $(free -m | awk '/Mem:/ {print $2 " " $3}')
# 현재 메모리 사용량 출력
echo "Current Memory usage: $(awk '/MemTotal/{total=$2}/MemAvailable/{available=$2} END {printf "%.0f", (total-available) / total * 100}' /proc/meminfo)%"
# 메모리 사용량 계산
usedPct=$(( used * 100 / total ))
if [[ "$usedPct" -gt $THRESHOLD ]]; then
echo "Memory usage is above $THRESHOLD%."
docker exec -i ${CONTAINER_NAME} rcon-cli "Broadcast Server_will_shut_down_in_5_seconds_for_maintance_Please_log_out"
# 5초 대기
sleep 5
# 세이브
docker exec -i $CONTAINER_NAME rcon-cli save
# 서버 종료
docker exec -i $CONTAINER_NAME rcon-cli "Broadcast Server_is_shutting_down_for_maintance"
# 5초 대기
sleep 5
# 서버 재시작
docker-compose -f "${YAML_FILE}" pull
docker-compose -f "${YAML_FILE}" down
docker-compose -f "${YAML_FILE}" up -d
else
echo "Memory usage is below $THRESHOLD%."
fi
밴 유저 삭제 스크립트
- 밴 당한 유저의 세이브 데이터 자동 삭제 스크립트
~/palworld/deleteBanUserData.sh
#!/usr/bin/bash
HOST='master'
SAVE_PATH="/home/${HOST}/palworld/game/Pal/Saved/SaveGames"
echo -e "PlayerUID를 입력하세요 (여러개의 ID를 쓰는 경우 공백으로 구분)\n: "
while true; do
read -a Playeruids
if [[ -z "${Playeruids}" ]]; then
clear
echo -e "PlayerUID 미입력됨.\n"
continue
fi
break
done
clear
echo "플레이어 데이터 찾는 중 ..."
for playeruid in "${Playeruids[@]}"; do
hex=$(printf "%X\n" "$playeruid" 2>/dev/null)
if [[ "${hex}" == 0 ]]; then
echo -e "${playeruid}: invalid octal number!"
continue
fi
player_file="$(find "${SAVE_PATH}" -type f -iname "$hex*")"
if [[ -z "${player_file}" ]]; then
echo -e "해당되는 플레이어를 찾지 못했습니다."
continue
fi
echo "find! Ban Playeruid : ${hex}"
echo -e "\nIs this really the ban user's file?\n👉 ${player_file}"
read -p "y / Y " answer
while true; do
if [[ "${answer}" == 'y' ]] || [[ "${answer}" == 'Y' ]]; then
echo -e "\nDeleting ban player's data ..."
find "${SAVE_PATH}" -type f -iname "$hex*" -exec sh -c "rm -rf {}" \;
exit 0
else
echo -e "With other data ..."
break
fi
done
done
서버 체크 스크립트
- Palworld 컨테이너 alive 체크 및 서버 메모리 점유율 모니터링 결과를 디스코드 webhook으로 알림 전송
~/palworld-webhook/script.sh
#!/usr/bin/bash
#### Required variables
HOST="server01"
LOG_FILE="/home/${HOST}/discord-webhook/webhook.log"
WEBHOOK_URL_FILE="/home/${HOST}/discord-webhook/.webhook_url.txt"
DISCORD_WEBHOOK_URL=$(awk -F '=' '{print $2}' "$WEBHOOK_URL_FILE")
SERVER_NAME="palworld-server"
SERVER_ADDRESS="waytothem.store:5555"
PLAYERS="$(docker exec -i $SERVER_NAME rcon-cli ShowPlayers 2>/dev/null | awk 'NR > 1 {print}' | wc -l)"
if $(docker ps | grep $SERVER_NAME >/dev/null 2>&1); then
# 도커 프로세스에 해당 서버가 있는 경우 & 서버가 실행 중인 경우
{% raw %} if [[ $(docker inspect --format '{{.State.Status }}' "$SERVER_NAME") == 'running' ]]; {% endraw %} then
SERVER_PROCESS=' 🟢 (온라인)'
# 도커 프로세스에 해당 서버가 있는 경우 & 서버가 실행 중이지 않은 경우
else
SERVER_PROCESS=' 🟠 (중지됨)'
fi
else
SERVER_PROCESS=' 🔴 (오프라인)'
fi
#
#### 메모리 점유율 확인
MEMORY_USAGE=$(free | grep 'Mem' | awk '{print int($3/$2 * 100.0)}')
if [[ "$MEMORY_USAGE" -le 30 ]]; then
color=2003199
elif [[ "$MEMORY_USAGE" -le 50 ]]; then
color=16747520
elif [[ "$MEMORY_USAGE" -le 60 ]]; then
color=16711680
else
color=0
fi
#
#### 보낼 Embed 메시지 작성
MEMORY_MESSAGE="메모리 사용률: $MEMORY_USAGE%"
REBOOT_MESSAGE="메모리 점유율이 70% 이상 올라갈 시 자동 재부팅"
PLAYER_MESSAGE="접속 인원: \`$PLAYERS/10\`"
SERVER_STATUS_MESSAGE="서버 상태: $SERVER_PROCESS"
SERVER_ADDRESS_MESSAGE="서버 주소: \`$SERVER_ADDRESS\`"
BOUNDARY_MESSAGE="────────────────────────────"
#### JSON 데이터 인코딩
DATA=$( jq -n \
--arg bm "$BOUNDARY_MESSAGE" \
--arg mm "$MEMORY_MESSAGE" \
--arg rm "$REBOOT_MESSAGE" \
--arg pm "$PLAYER_MESSAGE" \
--arg ssm "$SERVER_STATUS_MESSAGE" \
--arg sam "$SERVER_ADDRESS_MESSAGE" \
--arg c "$color" \
'{ content: " ", embeds: [ { title: "💊 Server Status", color: $c, fields: [ { name: $bm, value: " ", inline: false }, { name: $ssm, value: " ", inline: false }, { name: $pm, value: " ", inline: false }, { name: $sam, value: " ", inline: false },{ name: $mm, value: " ", inline: false }, { name: "주의사항", value: $rm, inline: false } ] } ] }' )
# curl을 사용하여 웹훅 호출
curl -X POST -H "Content-Type: application/json" -d "$DATA" "$DISCORD_WEBHOOK_URL"
# curl의 반환 코드를 확인하여 로그 파일에 결과 기록
if [[ $? != 0 ]]; then
echo "디스코드 메시지 전송 실패!" >> "$LOG_FILE"
else
echo "디스코드 메시지 전송 성공!" >> "$LOG_FILE"
fi
접속 중인 유저 확인하는 스크립트
도커 컨테이너에서 rcon-cli ShowPlayers
명령을 실행하여 현재 접속 중인 유저의 name
만 추출하고,
이를 디스코드 웹훅을 통해 접속자 정보를 보내는 쉘 스크립트이다.
palworld_check_user.sh
#!/bin/bash
# Docker 컨테이너 이름
CONTAINER_NAME="palworld-server"
# Discord Webhook URL
DISCORD_WEBHOOK_URL="https://discord.com/api/webhooks/your_webhook_url_here"
# rcon-cli 명령 실행
PLAYER_LIST=$(docker exec -it $CONTAINER_NAME rcon-cli ShowPlayers)
# 접속 중인 유저의 name 추출
PLAYER_NAMES=$(echo "$PLAYER_LIST" | awk -F'|' 'NR>2 && $2 != "" {print $2}' | sed 's/^[[:space:]]*//;s/[[:space:]]*$//')
# 접속 중인 유저 수 계산
PLAYER_COUNT=$(echo "$PLAYER_NAMES" | grep -c .)
# 유저 목록 포맷
if [ "$PLAYER_COUNT" -eq 0 ]; then
EMBED_DESCRIPTION="현재 접속 중인 유저가 없습니다."
else
FORMATTED_NAMES=$(echo "$PLAYER_NAMES" | nl -w 2 -s '. ')
EMBED_DESCRIPTION="현재 접속 중인 유저: **$PLAYER_COUNT명**\n\n\`\`\`\n$FORMATTED_NAMES\n\`\`\`"
fi
# Embed JSON 데이터 생성
EMBED_JSON=$(cat <<EOF
{
"embeds": [
{
"title": "Palworld Server 접속 정보",
"description": "$EMBED_DESCRIPTION",
"color": 3066993
}
]
}
EOF
)
# Discord 웹훅으로 메시지 전송
curl -H "Content-Type: application/json" \
-d "$EMBED_JSON" \
$DISCORD_WEBHOOK_URL
함수로 분리
~/palworld/PalworldManager.sh
#!/usr/bin/bash
#### Require variables
HOST='master'
#
#### 알림 함수
alarm() {
if [[ ${#} -ne 2 ]]; then
clear
echo -e "Usage 👉 ${0} hostname message"
exit 1
fi
HOST='master'
TELE_INFO_FILE="/home/${HOST}/telegram/token.env"
TELE_INFOS=($(awk -F '=' '{print $2}' "${TELE_INFO_FILE}"))
TOKEN="${TELE_INFOS[0]}"
CHAT_ID="${TELE_INFOS[1]}"
URL="https://api.telegram.org/bot${TOKEN}/sendMessage"
TEXT="[${1}] ${2}"
curl -s -d "chat_id=${CHAT_ID}&text=${TEXT}" ${URL} > /dev/null
}
####
#### 백업 함수
backup() {
SERVER_NAME="palworld-server"
if $(docker ps | grep $SERVER_NAME >/dev/null 2>&1); then
{% raw %} if [[ $(docker inspect --format '{{.State.Status }}' "$SERVER_NAME") == 'running' ]]; then {% endraw %}
echo "$SERVER_NAME is running"
# 도커 프로세스에 해당 서버가 있는 경우 & 서버가 실행 중이지 않은 경우
else
echo "$SERVER_NAME is paused"
exit 1
fi
else
echo "$SERVER_NAME is stoped"
exit 1
fi
# 압축할 디렉토리
source_dir="/home/${HOST}/palworld/game/Pal/Saved/SaveGames"
# 압축 파일이 저장될 디렉토리
backup_dir="/tmp/palworld/backup"
# 압축 파일 이름
DATE="$(date +'%Y%m%d_%H%M')"
backup_filename="palworld_backup_${DATE}.tar.gz"
# 백업 디렉토리 없는 경우 생성
if [[ ! -d "${backup_dir}" ]]; then
mkdir -p "${backup_dir}"
fi
# 로그 파일 생성
LOG_FILE_NM="/tmp/backup.log"
touch "${LOG_FILE_NM}"
{
echo -e "\n💿 PalWorld 백업 시작 ... 💿"
# 디렉토리를 gzip 압축
tar -czf "${backup_dir}/${backup_filename}" "${source_dir}" 1>/dev/null 2>&1
NAME=$(ls -lh "${backup_dir}/${backup_filename}" | awk '{print $9}')
SIZE=$(ls -lh "${backup_dir}/${backup_filename}" | awk '{print $5}')
echo "==== 백업 파일 정보 ===="
echo "파일명 : ${NAME}"
echo "파일 크기 : ${SIZE}"
echo
echo "==== 이전 백업파일 삭제 ===="
find "${backup_dir}" -type f -mmin +360 -exec sh -c "ls -l {}; rm -f {}" \;
echo -e "\n💿 PalWorld 백업 종료 ... 💿"
} > "${LOG_FILE_NM}"
# 텔레그램으로 알림 전송
alarm "Master Server" "$(cat "${LOG_FILE_NM}")"
# 로그파일 삭제
rm -rf "${LOG_FILE_NM}"
}
####
#### 리소스 체크 함수
resource_check() {
SWP_MEMORY=$(free -h | grep 'Swap' | awk '{print $4}')
MEMORY=$(free -h | grep 'Mem' | awk '{print $4}')
LOAD_AVERAGE=$(uptime | awk -F 'load average: ' '{print $2}')
MESSAGE="
MEMORY _> ${MEMORY}
SWAP MEMORY -> ${SWP_MEMORY}
uptime -> ${LOAD_AVERAGE}"
alarm "${HOST}" "${MESSAGE}"
}
####
#### 재부팅 함수
reboot() {
YAML_FILE="/home/$HOST/palworld/docker-compose.yaml"
CONTAINER_NAME="palworld-server"
THRESHOLD=80
# 변수에 각각 삽입
read total used <<< $(free -m | awk '/Mem:/ {print $2 " " $3}')
# 현재 메모리 사용량 출력
echo "Current Memory usage: $(awk '/MemTotal/{total=$2}/MemAvailable/{available=$2} END {printf "%.0f", (total-available) / total * 100}' /proc/meminfo)%"
# 메모리 사용량 계산
usedPct=$(( used * 100 / total ))
if [[ "$usedPct" -gt $THRESHOLD ]]; then
echo "Memory usage is above $THRESHOLD%."
docker exec -i ${CONTAINER_NAME} rcon-cli "Broadcast Server_will_shut_down_in_5_seconds_for_maintance_Please_log_out"
# 5초 대기
sleep 5
# 세이브
docker exec -i $CONTAINER_NAME rcon-cli save
# 서버 종료
docker exec -i $CONTAINER_NAME rcon-cli "Broadcast Server_is_shutting_down_for_maintance"
# 5초 대기
sleep 5
# 서버 재시작
docker-compose -f "${YAML_FILE}" pull
docker-compose -f "${YAML_FILE}" down
docker-compose -f "${YAML_FILE}" up -d
else
echo "Memory usage is below $THRESHOLD%."
fi
}
####
#### 현재 접속 중인 유저 체크하는 함수
checkUser() {
# Discord Webhook URL
DISCORD_WEBHOOK_URL="https://discord.com/api/webhooks/your_webhook_url_here"
# rcon-cli 명령 실행
PLAYER_LIST=$(docker exec -it $CONTAINER_NAME rcon-cli ShowPlayers)
# 접속 중인 유저의 name 추출
PLAYER_NAMES=$(echo "$PLAYER_LIST" | awk -F'|' 'NR>2 && $2 != "" {print $2}' | sed 's/^[[:space:]]*//;s/[[:space:]]*$//')
# 접속 중인 유저 수 계산
PLAYER_COUNT=$(echo "$PLAYER_NAMES" | grep -c .)
# 유저 목록 포맷
if [ "$PLAYER_COUNT" -eq 0 ]; then
EMBED_DESCRIPTION="현재 접속 중인 유저가 없습니다."
else
FORMATTED_NAMES=$(echo "$PLAYER_NAMES" | nl -w 2 -s '. ')
EMBED_DESCRIPTION="현재 접속 중인 유저: **$PLAYER_COUNT명**\n\n\`\`\`\n$FORMATTED_NAMES\n\`\`\`"
fi
# Embed JSON 데이터 생성
EMBED_JSON=$(cat <<EOF
{
"embeds": [
{
"title": "Palworld Server 접속 정보",
"description": "$EMBED_DESCRIPTION",
"color": 3066993
}
]
}
EOF
)
# Discord 웹훅으로 메시지 전송
curl -H "Content-Type: application/json" \
-d "$EMBED_JSON" \
$DISCORD_WEBHOOK_URL
}
####
# man logic to call functions based on passed argument
case "$1" in
backup)
backup
;;
resource)
resource_check
;;
restart)
reboot
;;
checkUser)
checkUser
;;
*)
echo -e "$0 {backup|resource|restart|checkUser}"
exit 1
;;
esac
crontab 등록
1. 스크립트 각각 등록한 경우
# 등록된 cron job 출력
crontab -l
# 5분마다 rcon으로 조회된 서버 player 목록 파일로 생성
*/5 * * * * bash /home/master/rconUserAlarm/set-rcon-players.sh >/dev/null 2>&1
# 10분마다 재부팅 스크립트 실행
*/10 * * * * bash /home/master/palworld/restart.sh >> /home/master/backup/restart.log 2>&1
# 30분마다 백업 스크립트 실행
*/30 * * * * bash /home/master/backup/script.sh >> /home/master/backup/backup.log 2>&1
# 30분마다 리소스 체크 스크립트 실행
*/30 * * * * bash /home/master/resource_alarm/alarm.sh >> /home/master/backup/resource.log 2>&1
2. 함수 기반으로 기능 분리하여 하나의 스크립트로 등록한 경우
# 등록된 cron job 출력
crontab -l
# 5분마다 rcon으로 조회된 서버 player 목록 파일로 생성
*/5 * * * * bash /home/master/rconUserAlarm/set-rcon-players.sh >/dev/null 2>&1
# 10분마다 재부팅 스크립트 실행
*/10 * * * * bash /home/master/palworld/PalworldManager.sh restart >> /home/master/backup/restart.log 2>&1
# 30분마다 백업 스크립트 실행
*/30 * * * * bash /home/master/palworld/PalworldManager.sh backup >> /home/master/backup/backup.log 2>&1
# 30분마다 리소스 체크 스크립트 실행
*/30 * * * * bash /home/master/palworld/PalworldManager.sh resource >> /home/master/backup/resource.log 2>&1